Sidebar
ein-/ausblenden

Keine Sub-Kategorien im Loop von WordPress

Plugin für WordPress SEO

Anzeige

Nicht immer soll die Ausgabe des Loop innerhalb einer Kategorie die Unterkategorien enthalten. Um das zu bewerkstelligen kann man einen eigene WP Query bauen und jeweils ein Template erstellen, siehe einige Tipps und Hinweis dazu in einem anderen Beitrag von mir. Alternativ kann man aber auch einen Hook nutzen und alle Unterkategorien zu filtern. Die folgende Syntax ablegen in der functions.php des Themes oder in ein Plugin auslagern und fertig.

Alternativ kann man diesen Hook aber auch zu anderen Sachen benutzen. Auch dazu möchte ich hier einige Worte verlieren.

Der Hook posts_where ist eine wichtige Möglichkeit um auf den Query von WordPress Einfluss zu nehmen und daher ein Bestandteil des Query: $where = apply_filters('posts_where', $where);. Er dient beispielsweise dazu, den Query von WordPress zu erweitern. Dabei wird immer ein Parameter übergeben. Dieser kann mit jede Art von SQL-Anweisungen befüllt werden.

Ein bekanntes Beispiel ist eventuell mein Artikel zum verspäteten Veröffentlichen von Feeds: „WordPress Feed zeitversetzt befüllen“.

Eine weitere Möglichkeit ist es, wie schon im Eingang angesprochen, dass bei der Ausgabe innerhalb von Kategorien die Unterkategorien nicht gelistet werden sollen. Die folgende Syntax tut genau dies.


if ( !function_exists('fb_filter_child_cats') ) {
	function fb_filter_child_cats( $cats= '' ) {
		global $wp_query, $wpdb;

		if ( is_category() ) {

			// get children ID's
			if ( $excludes = get_categories( "child_of=" . $wp_query->get('cat') ) ) {

				// set array with ID's
				foreach ( $excludes as $key => $value ) {
					$exclude[] = $value->cat_ID;
				}
			}

			// remove child cats
			if ( isset($exclude) && is_array($exclude) ) {
				$cats .= " AND " . $wpdb->prefix . "term_taxonomy.term_id NOT IN (" . implode(",", $exclude) . ") ";
			}
		}

		return $cats;
	}

	if ( !is_admin() ) {
		add_filter( 'posts_where', 'fb_filter_child_cats' );
	}
}

Um ein stabileres Ergebnis zu erzielen, sollte man möglichst SQL Selects umgehen und damit stabiler in WP zu bleiben. Daher hier eine weitere Lösung für die Filterung der Kinder-Kategorien:


function fb_filter_child_cats($query) {

	$cat = get_term_by('name', $query->query_vars['category_name'], 'category');
	$child_cats = (array) get_term_children( &$cat->term_id, 'category' );

	if ( !$query->is_admin )
		$query->set( 'category__not_in', array_merge($child_cats) );

	return $query;
}
add_filter( 'pre_get_posts', 'fb_filter_child_cats' );

Ein weiteres Beispiel soll zeigen, wie man den Query erweitert. Dabei beziehe ich mich auf einige Fragen aus der WP Hacker Mailingliste, die eigentlich dort nicht hingehören, dazu ist das Forum da.
In die Funktion habe ich mal drei Beispiele integriert.

  • Beiträge der Zukunft im Loop
  • Seiten im Loop
  • Nur Beiträge im Loop, die älter als 5 Tage sind

Ich hoffe, dass die Möglichkeiten damit recht gut dargestellt werden.


if ( !function_exists('fb_filter_where') ) {
	function fb_filter_where( $where = '' ) {
		global $wpdb;

		// allow future posts in loop
		$where .= " OR $wpdb->posts.post_status = 'future' ";

		// show pages in loop
		$where .= " OR $wpdb->posts.post_type = 'page' ";

		// only older (<) as 5 days
		$where .= " AND post_date < '" . date('Y-m-d', strtotime('-5 days')) . "'";

		return $where;
	}

	if ( !is_admin() ) {
		add_filter( 'posts_where', 'fb_filter_where' );
	}
}

Die Möglichkeiten sind sehr flexibel und im Grunde nur abhängig von den SQL-Abfragen. Damit sollte der Hook ein wenig verständlicher sein und seine vielseitigen Möglichkeiten lassen sich nur annähernd darstellen. Dieser Hook ist nur einer von vielen Hooks, die die Flexibilität von WordPress auszeichnen.

22 Kommentare zu „Keine Sub-Kategorien im Loop von WordPress“

  1. 1
    Kommentar von Daniel

    Dieser Post machte meinen Tag! Danke.

  2. 2
    Kommentar von Jens

    An welcher Stelle der functions.php füge ich den Code ein?
    habe ihn ans Ende gestellt und erhalte die Fehlermeldung über die Zeile:
    add_filter( 'posts_where', 'fb_filter_where' );

  3. 3
    Kommentar von Frank Bültge

    @Jens: wo ist egal, Hauptsache innerhalb von PHP <?php ?>

  4. 4
    Kommentar von Helmut

    Jetzt wollte ich mich fast beim falschen bedanken....
    Diese Erweiterung der functions.php ist aus meiner Sicht ein muss will ich WordPress als CMS System einsetzen und die Kategorien als hierarchische Verzeichnisstruktur nutzen.

  5. 5
    Kommentar von Wolf Larsen

    Ist es denn möglich der oberen Funktion auch zu sagen, welche Unterkategorien ausgeschlossen werden sollen? Ich möchte z.B. nur Artikel von zwei Unterkategorien ausschließen.

    Danke,
    Wolf

  6. 6
    Kommentar von Frank Bültge

    @Wolf Larsen: diese solltest du über den Query steuern, siehe dieser Beitrag.

  7. 7
    Kommentar von Wolf Larsen

    Servus Frank,

    der Code für das Ausnehmen der Subkategorien scheint bei mir nach dem Update auf 3.1 nicht mehr zu funktionieren. Die Oberkategorie bleibt leer, obwohl dort Artikel liegen. Wenn ich den Code aus meiner function lösche, dann habe ich wieder den gewohnten Anblick, also den WordPress-Standard. Danke!

    Wolf

  8. 8
    Kommentar von Michael

    Hi Frank!

    Habe Deinen Artikel zu diesem Thema auf wpengineer.com gelesen. Hast Du eine Idee, wieso der Code nicht mehr mit WP 3.1 funktioniert? Beim Einsatz des Codes in der functions.php wird in der Hauptkategorie überhaupt kein Artikel mehr angezeigt.

  9. 9
    Kommentar von Frank Bültge

    @Michael: prüfe mal diesen, sollte gehen:

    
    if ( !function_exists('fb_filter_child_cats') ) {
    	function fb_filter_child_cats( $cats ) {
    		global $wp_query, $wpdb;
    		if ( is_category() ) {
    			// get children ID's
    			if ( $excludes = get_categories( "child_of=" . $wp_query->get('cat') ) ) {
    				// set array with ID's
    				foreach ( $excludes as $key => $value ) {
    					$exclude[] = $value->cat_ID;
    				}
    			}
    			// remove child cats
    			if ( isset($exclude) && is_array($exclude) ) {
    				$cats .= " AND " . $wpdb->prefix . "term_relationships.term_taxonomy_id NOT IN (" . implode(",", $exclude) . ") ";
    			}
    		}
    		return $cats;
    	}
    	if ( !is_admin() ) {
    		add_filter( 'posts_where', 'fb_filter_child_cats' );
    	}
    }
    
  10. 10
    Kommentar von Michael

    Hallo Frank!

    Danke für Deine Rückmeldung, aber leider ist der von Dir gepostete Code mit dem von der wpengineer Seite identisch. Das funktioniert leider nicht.

  11. 11
    Kommentar von Frank Bültge

    @Michael: ja, der ist so gut wie immer der gleiche, da ich ja beide Blogs betreibe und update. Ich habe aber in einem Beispiel getestet und in den Archiven waren die child-cats nicht mehr gelistet. Was funktioniert nicht, wie sieht deine Ausleitung aus; also was passiert?

  12. 12
    Kommentar von Michael

    Hi!
    Wie in #8 beschrieben erscheinen mit diesem Code keine Artikel mehr in der Elternkategorie. Stattdessen erhalte ich dort eine Meldung, dass keine passenden Beiträge gefunden werden können (Meldung kommt aus der archive.php). Klicke ich auf eine Unterkategorie, werden mir dort die zugeordneten Artikel angezeigt. Wie gesagt, ist das erst seit WP 3.1 so. Vorher funktionierte es wie von Dir beschrieben.
    Beispiel WP 3.0.4
    Beispiel WP 3.1

  13. 13
    Kommentar von Frank Bültge

    @Michael: ich schaue nochmal, aber genau das habe ich im Test versucht - bei Darstellung der Elternkategorie waren nur die Artikel in dieser Kategorie enthalten, nicht die aus der Kinder-Kategorie.

  14. 14
    Kommentar von Frank Bültge

    @Michael: irgendwie hatte ich via Copy/paste den gleichen Code nochmal rein gepackt, teste nun mal meinen Kommentar

  15. 15
    Kommentar von Frank Bültge

    @Michale: ich denke, dass ich eine bessere Lösung habe, ohne DB-select, was es stabil gegenüber den WP Versionen macht:

    
    function fb_filter_child_cats($query) {
    	$child_cats = (array) get_term_children( $query->get('cat'), 'category');
    
    	if ( !$query->is_admin )
    		$query->set( 'category__not_in', array_merge($child_cats) );
    
    	return $query;
    }
    add_filter( 'pre_get_posts', 'fb_filter_child_cats' );
    
  16. 16
    Kommentar von Michael

    Hallo Frank!
    Sorry, für die späte Rückmeldung! Hab in letzter Zeit einiges zu tun und komm nur noch Abends zu den WP-Geschichten...

    Vielen vielen Dank! Der Code aus #9 funktioniert nun wieder. Super!

    Natürlich habe ich auch Deinen zweiten Versuch mal getestet, aber da werden die Unterkategorien nicht herausgefiltert und erscheinen demnach noch in den Elternkategorien.

    Trotzdem ein dickes Dankeschön für Deine Mühe! Ich schau aber die Tage nochmal rein, vielleicht hast Du ja noch eine Idee bezgl. #15.

    Gruß
    Michael

  17. 17
    Kommentar von Frank Bültge

    @Michael: ich denke ich habe eine Lösung gefunden, wenn man WP in De betreibt, dann habe ich die ID nicht in $query, daher ging es bei mir im englischen Blog, meine neue Version holt die Id aus dem Namen und ging eben in allen Tests.

    
    function fb_filter_child_cats($query) {
    
    	$cat = get_term_by('name', $query->query_vars['category_name'], 'category');
    	$child_cats = (array) get_term_children( &$cat->term_id, 'category' );
    	// auch möglich
    	//$child_cats = (array) get_term_children( get_cat_id($query-%gt;query_vars['category_name']), 'category' );
    
    	if ( !$query->is_admin )
    		$query->set( 'category__not_in', array_merge($child_cats) );
    
    	return $query;
    }
    add_filter( 'pre_get_posts', 'fb_filter_child_cats' );
    
  18. 18
    Kommentar von Michael

    Hallo Frank!
    Sorry für die späte Rückmeldung! Habe nun mal Deine 2te Variante aus #17 getestet. Die scheint auch problemlos zu funktionieren. Super! Herzlichen Dank nochmal für Deine Arbeit!

  19. 19
    Kommentar von Wolf Larsen

    Hallo Frank!

    ich bin immer noch am tun wegen diesen ausgenommenen Unterkategorien. Mit 3.0.5 und dem alten Code ging alles wunderbar. Mit 3.1 brachte ich nichts zu stande. Nun mit 3.1.1 habe ich folgendes Verhalten: Der Code aus Beitrag 15 und 17 filtert überhaupt nichts, wie Standard also.

    Der Code aus Beitrag 9 ist besser, aber er lässt trotzdem eine andere Unterkategorie zu. Ich habe also alle meine Beiträge aus der Oberkategorie und zwei Beiträge aus einer Unterkategorie. Ich habe schon die Unterkategorie umbenannt, gelöscht und unter anderem Namen angelegt, nichts. Wenn ich die beiden Beiträge in eine andere schon angelegte Unterkategorie verschiebe, dann geht es. Sowas macht mich fast verrückt :-)

    Ich schwöre, dass es mit 3.0.5 schon ging :-)
    Die Kategorie zum schauen ist hier: http://tinyurl.com/67l3trl

    Danke,
    Wolf

  20. 20
    Kommentar von Wolf Larsen

    Hallo Frank,

    ich habe nochmal geforscht. Komischerweise ist es so, dass neue oder bis vor einiger Zeit angelegte Unterkategorien einfach nicht aus der Oberkategorie verschwinden. Schon ältere bestehende Unterkategorien werden aber ausgenommen. Da bin ich echt ratlos.

    Wolf

  21. 21
    Kommentar von Mark

    Hallo,

    ich habe das Problem, dass ich genau das umgekehrte benötige. Die Kategorie-Seiten zeigen bei mir nur die Artikel der Hauptkategorie und nicht auch die der Sub-Kategorien an.

    Weisst Du da evt. Rat?

    Danke

    Mark

  22. 22
    Kommentar von Frank Bültge

    @Mark: es sollte reichen, wenn du category__not_in in category__in änderst.

Kommentar schreiben

Kommentarregeln: Bleib cool, kritisch ist in Ordnung, aber wenn du unhöflich bist, dann lösche ich deinen Kommentar. Bitte benutze deinen persönlichen Namen oder Initialen und nicht den Namen eines Unternehmens, dies würde als Spam gewertet und wird gelöscht. Der Zusammenhang zwischen Namen und URL sollte nicht offensichtlich auf Spam hindeuten! ♥ Ansonsten, vielen Dank für den Kommentar und viel Spaß mit meinem Blog.

E-Mail-Benachrichtigung bei weiteren Kommentaren.
Auch möglich: Abo ohne Kommentar.

Kommentar-Hilfe

händischer Spam:
Beachte die Kommentarregeln, jede Form von versuchtem Spam wird gelöscht. Warum und wieso steht in einem meiner Beiträge.

Bezug auf Textstellen:
Du kannst direkt bezug auf Textstellen im Beitrag nehmen. Dazu muss lediglich der Bereich im Artikel markiert werden; daraufhin erscheint ein Button, der den markierten Text in das Kommentarfeld übernimmt und als Zitat auszeichnet. Die Funktion ist nur bei aktivem JavaScript nutzbar.

xHTML:
Du kannst folgende Tags benutzen: <a href="" title=""> <abbr title=""> <acronym title=""> <blockquote cite=""> <code> <pre> <em> <strong> <strike> <ul> <ul> <li>

Achte darauf, wenn du Code im Kommentar hinterlegen willst, dann muss der Code maskiert sein. Dann wird er nicht interpretiert. Der Code muss mit Hilfe von HTML-Entities dargestellt werden, d.h. dass man z.B. < als &lt; und > als &gt; einfügt.

E-Mail-Benachrichtigung bei neuen Kommentaren ?
Wenn der Haken in der Checkbox gesetzt ist, dann wirst du über neue Kommentare vie E-Mail informiert. Der Versand erfolgt nur, wenn du die URL in der Bestätigungs-E-Mail genutzt hast oder schon Abonnent hier im Blog bist.

Kommentar erscheint nicht:
Alle Kommentare werden manuell geprüft, freigegeben und nach Möglichkeit beantwortet. Bitte um etwas Geduld und Nachsicht.

Identifikationsbilder (Avatare):
Auf Gravatar.com kann man sich mit seiner E-Mail-Adresse registrieren und ein Bild hochladen, dann erscheint dieses Gravatar hier und in vielen weiteren Blogs.

Spamschutz:
Das Kommentarformular ist mit einem Spamschutz ausgerüstet. Solltest du diesen Artikel ohne JavaScript besuchen und kommentieren wollen, so muss du die Frage beantworten und das jeweilige Wort in das Textfeld eingeben.