Includerea tipurilor personalizate de postări în widget-ul "Postări recente"
Pot include cu ușurință tipurile mele personalizate de postări în bucla principală prin mici ajustări cu query_posts()
, dar nu sunt sigur cum aș putea include tipurile personalizate de postări în widget-ul "Postări recente" din bara laterală (sau în oricare alt widget).
Cum ar trebui să procedez pentru a extinde domeniul widget-ului "Postări recente" să includă mai mult decât doar tipul de postare nativ?

Va trebui să editați codul pentru widget-ul Postări Recente sau să creați propria versiune bazată pe cea implicită. Codul se află în fișierul wp-includes/default-widgets.php
în jurul liniei 513. Dar, deoarece nu ar trebui să faceți niciodată modificări în nucleul WordPress, recomandarea mea este să copiați codul pentru a crea propriul widget Postări Recente Personalizate și să-l folosiți pe site-ul dvs. Pur și simplu adăugați noua clasă de widget în fișierul functions.php
al temei sau folosiți-o într-un plugin.
Singura modificare reală pe care trebuie să o faceți este la numele clasei widget-ului și la funcțiile și opțiunile încapsulate (pentru a evita conflicte de nume cu widget-ul original Postări Recente). După aceea, va trebui să editați apelul către WP_Query
în constructorul widget()
astfel încât să includă tipul personalizat de postare.
Pentru acest exemplu, am setat post_type
egal cu array('post, 'page', 'custom-post-type')
... va trebui să modificați acest lucru pentru a se potrivi nevoilor dumneavoastră specifice.
Iată codul complet al widget-ului pentru referință:
/**
* Clasa widget-ului My_Custom_Recent_Posts
*
*/
class WP_Widget_My_Custom_Recent_Posts extends WP_Widget {
function __construct() {
$widget_ops = array('classname' => 'widget_my_custom_recent_entries', 'description' => __( "Cele mai recente postări de pe site-ul tău") );
$this->WP_Widget('my-custom-recent-posts', __('Postări Recente Personalizate'), $widget_ops);
$this->alt_option_name = 'widget_my_custom_recent_entries';
add_action( 'save_post', array(&$this, 'flush_widget_cache') );
add_action( 'deleted_post', array(&$this, 'flush_widget_cache') );
add_action( 'switch_theme', array(&$this, 'flush_widget_cache') );
}
function widget($args, $instance) {
$cache = wp_cache_get('widget_my_custom_recent_posts', 'widget');
if ( !is_array($cache) )
$cache = array();
if ( isset($cache[$args['widget_id']]) ) {
echo $cache[$args['widget_id']];
return;
}
ob_start();
extract($args);
$title = apply_filters('widget_title', empty($instance['title']) ? __('Postări Recente Personalizate') : $instance['title'], $instance, $this->id_base);
if ( !$number = (int) $instance['number'] )
$number = 10;
else if ( $number < 1 )
$number = 1;
else if ( $number > 15 )
$number = 15;
$r = new WP_Query(array('showposts' => $number, 'nopaging' => 0, 'post_status' => 'publish', 'ignore_sticky_posts' => true, 'post_type' => array('post', 'page', 'custom-post-type')));
if ($r->have_posts()) :
?>
<?php echo $before_widget; ?>
<?php if ( $title ) echo $before_title . $title . $after_title; ?>
<ul>
<?php while ($r->have_posts()) : $r->the_post(); ?>
<li><a href="<?php the_permalink() ?>" title="<?php echo esc_attr(get_the_title() ? get_the_title() : get_the_ID()); ?>"><?php if ( get_the_title() ) the_title(); else the_ID(); ?></a></li>
<?php endwhile; ?>
</ul>
<?php echo $after_widget; ?>
<?php
// Resetează variabila globală $the_post deoarece această interogare a modificat-o
wp_reset_postdata();
endif;
$cache[$args['widget_id']] = ob_get_flush();
wp_cache_set('widget_my_custom_recent_posts', $cache, 'widget');
}
function update( $new_instance, $old_instance ) {
$instance = $old_instance;
$instance['title'] = strip_tags($new_instance['title']);
$instance['number'] = (int) $new_instance['number'];
$this->flush_widget_cache();
$alloptions = wp_cache_get( 'alloptions', 'options' );
if ( isset($alloptions['widget_my_custom_recent_entries']) )
delete_option('widget_my_custom_recent_entries');
return $instance;
}
function flush_widget_cache() {
wp_cache_delete('widget_my_custom_recent_posts', 'widget');
}
function form( $instance ) {
$title = isset($instance['title']) ? esc_attr($instance['title']) : '';
if ( !isset($instance['number']) || !$number = (int) $instance['number'] )
$number = 5;
?>
<p><label for="<?php echo $this->get_field_id('title'); ?>"><?php _e('Titlu:'); ?></label>
<input class="widefat" id="<?php echo $this->get_field_id('title'); ?>" name="<?php echo $this->get_field_name('title'); ?>" type="text" value="<?php echo $title; ?>" /></p>
<p><label for="<?php echo $this->get_field_id('number'); ?>"><?php _e('Număr de postări de afișat:'); ?></label>
<input id="<?php echo $this->get_field_id('number'); ?>" name="<?php echo $this->get_field_name('number'); ?>" type="text" value="<?php echo $number; ?>" size="3" /></p>
<?php
}
}

În loc să redenumești widget-ul, cred că ar fi mai bine să folosești unregister_widget
pentru widget-ul implicit înainte de a înregistra cel nou (mai puțin confuz în administrare).

Da, ai putea folosi unregister_widget
, dar această metodă îți oferă acces la ambele dacă dorești să le folosești pe ambele (de exemplu, să ai un widget Postări Recente pentru a afișa postările recente de pe blog și un widget Recenzii Recente de Filme pentru a afișa doar intrările recente ale unui anumit tip de postare personalizată).

doar o notă pentru cei care încerce codul de mai sus: Nu uitați să adăugați aceasta: register_widget('WP_Widget_My_Custom_Recent_Posts');

Mulțumesc pentru furnizarea codului pentru clasa personalizată de widget și pentru explicații. A fost foarte util. Totuși, mă întrebam dacă există o modalitate de a-l modifica pentru a permite HTML încorporat în câmpul de titlu? Practic, am nevoie să transform titlul widget-ului într-un link pe care se poate face clic, dar implicit HTML-ul încorporat este eliminat din titlurile widget-urilor. Orice ajutor ar fi foarte apreciat.

Ian, te rog să pui asta ca o întrebare separată, astfel încât să putem avea o discuție completă...

Nu știu nimic despre PHP, dar nu s-ar putea doar să Extend
clasa implicită și să actualizezi variabilele necesare? Decât un întreg copy-paste?

Începând cu versiunea 3.6 (cel puțin), puteți utiliza următorul cod pentru a modifica interogarea folosită:
add_filter('widget_posts_args', 'widget_posts_args_add_custom_type');
function widget_posts_args_add_custom_type($params) {
$params['post_type'] = array('post','custom_type');
return $params;
}
Pur și simplu adăugați tipurile dorite în array-ul pentru post_type și acestea ar trebui să apară.
Actualizare: Conform http://core.trac.wordpress.org/ticket/16159, această funcționalitate este disponibilă începând cu versiunea 3.4

Tocmai am dat peste un plugin minunat unde munca grea este deja făcută și are o documentație excelentă și suportul autorului. Am fost foarte impresionat.
Acesta permite suprascrieri WP_Query (permițându-vă să filtrați după tipuri personalizate de postări și orice altceva ați dori) și oferă instrucțiuni clare despre cum să-l utilizați.
Documentație http://www.pjgalbraith.com/2011/08/recent-posts-plus/
URL-ul pluginului Wordpress http://wordpress.org/extend/plugins/recent-posts-plus/
Mi-a făcut munca cu atât de mult mai ușoară!

Am creat și un widget plugin pentru aceasta care este mai personalizabil decât widget-ul Recent Posts. Dacă sunteți interesat, îl puteți descărca de aici http://new2wp.com/pro/latest-custom-post-type-posts-sidebar-widget/

Acest Cod Creează un Widget Recent Posts Personalizat Care Include și CPT-urile Tale
Sunt implicați 2 pași pentru extinderea widget-ului nativ de articole recente:
i. Creează o nouă clasă pentru widget-ul tău personalizat de articole recente, pe care o poți face copiind și redenumind codul widget-ului din defaults-widgets.php din directorul wp-includes.
ii. Apoi va trebui să înregistrezi noul widget și poți alege să dezînregistrezi widget-ul nativ sau să le folosești pe amândouă.
Tot codul poate fi copiat în fișierul functions.php folosind un child theme sau poți crea un alt fișier și să-l incluzi în fișierul functions.php al child theme-ului.
<?php
class WPSites_Recent_Posts extends WP_Widget {
public function __construct() {
$widget_ops = array('classname' => 'wpsites_recent_posts', 'description' => __( "Ultimele CPT-uri & Articole.") );
parent::__construct('wpsites-recent-posts', __('Articole Recente WP Sites'), $widget_ops);
$this->alt_option_name = 'wpsites_recent_posts';
add_action( 'save_post', array($this, 'flush_widget_cache') );
add_action( 'deleted_post', array($this, 'flush_widget_cache') );
add_action( 'switch_theme', array($this, 'flush_widget_cache') );
}
public function widget($args, $instance) {
$cache = array();
if ( ! $this->is_preview() ) {
$cache = wp_cache_get( 'wpsites_widget_recent_posts', 'widget' );
}
if ( ! is_array( $cache ) ) {
$cache = array();
}
if ( ! isset( $args['widget_id'] ) ) {
$args['widget_id'] = $this->id;
}
if ( isset( $cache[ $args['widget_id'] ] ) ) {
echo $cache[ $args['widget_id'] ];
return;
}
ob_start();
$title = ( ! empty( $instance['title'] ) ) ? $instance['title'] : __( 'Articole Recente' );
/** This filter is documented in wp-includes/default-widgets.php */
$title = apply_filters( 'widget_title', $title, $instance, $this->id_base );
$number = ( ! empty( $instance['number'] ) ) ? absint( $instance['number'] ) : 5;
if ( ! $number )
$number = 5;
$show_date = isset( $instance['show_date'] ) ? $instance['show_date'] : false;
$r = new WP_Query( apply_filters( 'widget_posts_args', array(
'posts_per_page' => $number,
'no_found_rows' => true,
'post_status' => 'publish',
'post_type' => array('post', 'portfolio'),
'ignore_sticky_posts' => true
) ) );
if ($r->have_posts()) :
?>
<?php echo $args['before_widget']; ?>
<?php if ( $title ) {
echo $args['before_title'] . $title . $args['after_title'];
} ?>
<ul>
<?php while ( $r->have_posts() ) : $r->the_post(); ?>
<li>
<a href="<?php the_permalink(); ?>"><?php get_the_title() ? the_title() : the_ID(); ?></a>
<?php if ( $show_date ) : ?>
<span class="post-date"><?php echo get_the_date(); ?></span>
<?php endif; ?>
</li>
<?php endwhile; ?>
</ul>
<?php echo $args['after_widget']; ?>
<?php
wp_reset_postdata();
endif;
if ( ! $this->is_preview() ) {
$cache[ $args['widget_id'] ] = ob_get_flush();
wp_cache_set( 'wpsites_widget_recent_posts', $cache, 'widget' );
} else {
ob_end_flush();
}
}
public function update( $new_instance, $old_instance ) {
$instance = $old_instance;
$instance['title'] = strip_tags($new_instance['title']);
$instance['number'] = (int) $new_instance['number'];
$instance['show_date'] = isset( $new_instance['show_date'] ) ? (bool) $new_instance['show_date'] : false;
$this->flush_widget_cache();
$alloptions = wp_cache_get( 'alloptions', 'options' );
if ( isset($alloptions['wpsites_recent_posts']) )
delete_option('wpsites_recent_posts');
return $instance;
}
public function flush_widget_cache() {
wp_cache_delete('wpsites_widget_recent_posts', 'widget');
}
public function form( $instance ) {
$title = isset( $instance['title'] ) ? esc_attr( $instance['title'] ) : '';
$number = isset( $instance['number'] ) ? absint( $instance['number'] ) : 5;
$show_date = isset( $instance['show_date'] ) ? (bool) $instance['show_date'] : false;
?>
<p><label for="<?php echo $this->get_field_id( 'title' ); ?>"><?php _e( 'Titlu:' ); ?></label>
<input class="widefat" id="<?php echo $this->get_field_id( 'title' ); ?>" name="<?php echo $this->get_field_name( 'title' ); ?>" type="text" value="<?php echo $title; ?>" /></p>
<p><label for="<?php echo $this->get_field_id( 'number' ); ?>"><?php _e( 'Număr de articole de afișat:' ); ?></label>
<input id="<?php echo $this->get_field_id( 'number' ); ?>" name="<?php echo $this->get_field_name( 'number' ); ?>" type="text" value="<?php echo $number; ?>" size="3" /></p>
<p><input class="checkbox" type="checkbox" <?php checked( $show_date ); ?> id="<?php echo $this->get_field_id( 'show_date' ); ?>" name="<?php echo $this->get_field_name( 'show_date' ); ?>" />
<label for="<?php echo $this->get_field_id( 'show_date' ); ?>"><?php _e( 'Afișează data articolului?' ); ?></label></p>
<?php
}
}
Înregistrează noul widget personalizat de articole recente
function wpsites_widgets_init() {
if ( !is_blog_installed() )
return;
register_widget('WPSites_Recent_Posts');
do_action( 'widgets_init' );
}
add_action( 'init', 'wpsites_widgets_init', 2 );
Codul include o interogare WP_Query modificată care include un array pentru tipurile de postări, inclusiv CPT-ul portfolio pe care îl poți redenumi pentru a se potrivi cu tipul tău personalizat de postare.
Iată linia de cod care trebuie modificată:
'post_type' => array('post', 'portfolio'),

Este 2020 și am venit aici pentru a găsi o soluție la "cele mai recente 10 postări personalizate de tipul XYZ". Am găsit acel plugin care face asta și chiar mai mult.
Widget-uri pentru Tipuri de Postări Personalizate extinde funcționalitățile standard ale widget-urilor pentru postări obișnuite (cele mai recente, arhive lunare, categorii utilizate, comentarii recente, căutare, calendar) la tipurile de postări personalizate.
Selectezi widget-ul de care ai nevoie (în cazul meu "cele mai recente") și primești mai întâi o listă derulantă unde specifici tipul de postare personalizată pe care trebuie să o vizeze widget-ul. Opțiunea implicită este vechea bună 'post', așadar acest plugin este un înlocuitor pentru widget-urile standard legate de postări din WP.
