'cat-post-widget', 'description' => __( 'List single category posts', 'category-posts' ), ); parent::__construct( WIDGET_BASE_ID, __( 'Category Posts', 'category-posts' ), $widget_ops ); } /** * Calculate the HTML for showing the thumb of a post item. * * Used as a filter for the thumb wordpress API to add css based stretching and cropping * when the image is not at the requested dimensions * * @param string $html The original HTML generated by the core APIS. * @param int $post_id the ID of the post of which the thumb is a featured image. * @param int $post_thumbnail_id The id of the featured image attachment. * @param string|array $size The requested size identified by name or (width, height) array. * @param mixed $attr ignored in this context. * @return string The HTML for the thumb related to the post * * @since 4.1 */ public function post_thumbnail_html( $html, $post_id, $post_thumbnail_id, $size, $attr ) { $use_css_cropping = isset( $this->instance['use_css_cropping'] ) && $this->instance['use_css_cropping']; $thumb = isset( $this->instance['template'] ) && preg_match( '/%thumb%/', $this->instance['template'] ); if ( ! ( $use_css_cropping && $thumb ) ) { return $html; } $meta = image_get_intermediate_size( $post_thumbnail_id, $size ); if ( empty( $meta ) ) { $post_img = wp_get_attachment_metadata( $post_thumbnail_id, $size ); $meta['file'] = basename( $post_img['file'] ); } $origfile = get_attached_file( $post_thumbnail_id, true ); // the location of the full file. $file = dirname( $origfile ) . '/' . $meta['file']; // the location of the file displayed as thumb. if ( file_exists( $file ) ) { list( $width, $height ) = getimagesize( $file ); // get actual size of the thumb file. if ( isset( $this->instance['use_css_cropping'] ) && $this->instance['use_css_cropping'] ) { $show_post_format = isset( $this->instance['show_post_format'] ) && ( 'none' !== $this->instance['show_post_format'] ); if ( $show_post_format || $this->instance['thumb_hover'] ) { $format = get_post_format() ? : 'standard'; $post_format_class = 'cat-post-format cat-post-format-' . $format; } $html = '' . $html . ''; } else { // use_css_cropping is not used. // wrap span. $html = '' . $html . ''; } } return $html; } /* * wrapper to execute the the_post_thumbnail with filters. */ /** * Calculate the HTML for showing the thumb of a post item. * * It is a wrapper to execute the the_post_thumbnail with filters * * @param string|array $size The requested size identified by name or (width, height) array. * * @return string The HTML for the thumb related to the post and empty string if it can not be calculated * * @since 4.1 */ public function the_post_thumbnail( $size = 'post-thumbnail' ) { if ( empty( $size ) ) { // if junk value, make it a normal thumb. $size = 'post-thumbnail'; } elseif ( is_array( $size ) && ( 2 === count( $size ) ) ) { // good format at least. // normalize to ints first. $size[0] = (int) $size[0]; $size[1] = (int) $size[1]; if ( ( 0 === $size[0] ) && ( 0 === $size[1] ) ) { // Both values zero then revert to thumbnail. $size = array( get_option( 'thumbnail_size_w', 150 ), get_option( 'thumbnail_size_h', 150 ) ); } elseif ( ( 0 === $size[0] ) && ( 0 !== $size[1] ) ) { // if thumb width 0 set to max/full widths for wp rendering $post_thumb = get_the_post_thumbnail( get_the_ID(), 'full' ); preg_match( '/(?<=width=")[\d]*/', $post_thumb, $thumb_full_w ); $size[0] = $thumb_full_w[0]; } elseif ( ( 0 !== $size[0] ) && ( 0 === $size[1] ) ) { // if thumb height 0 get full thumb for ratio and calc height with ratio $post_thumb = get_the_post_thumbnail( get_the_ID(), 'full' ); preg_match( '/(?<=width=")[\d]*/', $post_thumb, $thumb_full_w ); preg_match( '/(?<=height=")[\d]*/', $post_thumb, $thumb_full_h ); $ratio = $thumb_full_w[0] / $thumb_full_h[0]; $size[1] = intval( $size[0] / $ratio ); } } else { $size = array( get_option( 'thumbnail_size_w', 150 ), get_option( 'thumbnail_size_h', 150 ) ); // yet another form of junk. } $post_thumbnail_id = get_post_thumbnail_id( get_the_ID() ); if ( ! $post_thumbnail_id && $this->instance['default_thunmbnail'] ) { $post_thumbnail_id = $this->instance['default_thunmbnail']; } do_action( 'begin_fetch_post_thumbnail_html', get_the_ID(), $post_thumbnail_id, $size ); $html = wp_get_attachment_image( $post_thumbnail_id, $size, false, '' ); if ( ! $html ) { $ret = ''; } else { $ret = $this->post_thumbnail_html( $html, get_the_ID(), $post_thumbnail_id, $size, '' ); } do_action( 'end_fetch_post_thumbnail_html', get_the_ID(), $post_thumbnail_id, $size ); return $ret; } /** * Excerpt more link filter * * @param string $more The "more" text passed by the filter. * * @return string The link to the post with the "more" text configured in the widget. */ public function excerpt_more_filter( $more ) { return ' ' . esc_html( $this->instance['excerpt_more_text'] ) . ''; } /** * Apply the_content filter for excerpt * This should show sharing buttons which comes with other widgets in the widget output in the same way as on the main content * * @param string $text The HTML with other applied excerpt filters. * * @return string If option hide_social_buttons is unchecked applay the_content filter. * * @since 4.6 */ public function apply_the_excerpt( $text ) { $ret = apply_filters( 'the_content', $text ); return $ret; } /** * Calculate the wp-query arguments matching the filter settings of the widget * * @param array $instance Array which contains the various settings. * @return array The array that can be fed to wp_Query to get the relevant posts * * @since 4.6 */ public function queryArgs( $instance ) { $valid_sort_orders = array( 'date', 'title', 'comment_count', 'rand' ); if ( isset( $instance['sort_by'] ) && in_array( $instance['sort_by'], $valid_sort_orders, true ) ) { $sort_by = $instance['sort_by']; } else { $sort_by = 'date'; } $sort_order = ( isset( $instance['asc_sort_order'] ) && $instance['asc_sort_order'] ) ? 'ASC' : 'DESC'; // Get array of post info. $args = array( 'orderby' => $sort_by, 'order' => $sort_order, 'ignore_sticky_posts' => 1, // Make sure we do not get stickies out of order. 'no_found_rows' => true, // Do not count the total numbers of rows by default. ); $non_default_valid_status = array( 'publish', 'future', 'publish,future', 'private', 'private,publish', 'private,publish,future', ); if ( isset( $instance['status'] ) && in_array( $instance['status'], $non_default_valid_status, true ) ) { $args['post_status'] = $instance['status']; } if ( isset( $instance['num'] ) ) { $args['showposts'] = (int) $instance['num']; } if ( isset( $instance['offset'] ) && ( (int) $instance['offset'] > 1 ) ) { $args['offset'] = (int) $instance['offset'] - 1; } if ( isset( $instance['cat'] ) ) { if ( isset( $instance['no_cat_childs'] ) && $instance['no_cat_childs'] ) { $args['category__in'] = (int) $instance['cat']; } else { $args['cat'] = (int) $instance['cat']; } } if ( is_singular() && isset( $instance['exclude_current_post'] ) && $instance['exclude_current_post'] ) { $args['post__not_in'] = array( get_the_ID() ); } if ( isset( $instance['hideNoThumb'] ) && $instance['hideNoThumb'] ) { $args = array_merge( $args, array( 'meta_query' => array( array( 'key' => '_thumbnail_id', 'compare' => 'EXISTS', ), ), ) ); } switch ( $instance['date_range'] ) { case 'days_ago': $ago = (int) $instance['days_ago']; // If there is no valid integer value given, bail. if ( 0 === $ago ) { break; } $date = date( 'Y-m-d', strtotime( '-' . $ago . ' days' ) ); $args['date_query'] = array( 'after' => $date, 'inclusive' => true, ); break; case 'between_dates': // Validation note - not doing any, assuming the query will // fail gracefully enough for now as it is not clear what Should // the validation be right now. $start_date = $instance['start_date']; $end_date = $instance['end_date']; $args['date_query'] = array( 'after' => $start_date, 'before' => $end_date, 'inclusive' => true, ); break; } return $args; } /** * Calculate the HTML of the title based on the widget settings * * @param string $before_title The sidebar configured HTML that should come * before the title itself. * @param string $after_title The sidebar configured HTML that should come * after the title itself. * @param array $instance Array which contains the various settings. * @return string The HTML for the title area * * @since 4.6 */ public function titleHTML( $before_title, $after_title, $instance ) { $ret = ''; // If no title, use the name of the category. if ( ! isset( $instance['title'] ) || ! $instance['title'] ) { $instance['title'] = ''; if ( 0 !== (int) $instance['cat'] ) { $category_info = get_category( $instance['cat'] ); if ( $category_info && ! is_wp_error( $category_info ) ) { $instance['title'] = $category_info->name; } else { $instance['cat'] = 0; // For further processing treat it like "all categories". $instance['title'] = __( 'Recent Posts', 'category-posts' ); } } else { $instance['title'] = __( 'Recent Posts', 'category-posts' ); } } if ( ! ( isset( $instance['hide_title'] ) && $instance['hide_title'] ) ) { $ret = $before_title; if ( isset( $instance['is_shortcode'] ) ) { $title = esc_html( $instance['title'] ); } else { $title = apply_filters( 'widget_title', $instance['title'] ); } if ( isset( $instance['title_link'] ) && $instance['title_link'] ) { if ( 0 !== $instance['cat'] ) { $ret .= '' . $title . ''; } elseif ( isset( $instance['title_link_url'] ) && $instance['title_link_url'] ) { $ret .= '' . $title . ''; } else { $ret .= '' . $title . ''; } } else { $ret .= $title; } $ret .= $after_title; } return $ret; } /** * Get the URL of the blog page or home page if no explicit blog page is defined. * * @return string The URL of the blog page * * @since 4.8 */ private function blog_page_url() { $blog_page = get_option( 'page_for_posts' ); if ( $blog_page ) { $url = get_permalink( $blog_page ); } else { $url = home_url(); } return $url; } /** * Calculate the HTML of the load more button based on the widget settings * * @param array $instance Array which contains the various settings. * * @return string The HTML for the load more area * * @since 4.9 */ public function loadMoreHTML( $instance ) { if ( ! $instance['enable_loadmore'] ) { return ''; } $ret = '
'; $context = 0; if ( is_singular() ) { $context = get_the_ID(); } add_action( 'wp_footer', __NAMESPACE__ . '\embed_loadmore_scripts' ); // We rely on the widget number to be properly set. // but need a slight different handling for proper widgets. if ( is_int( $this->number ) ) { // it is a proper widget, add the prefix. $id = 'widget-' . $this->number; } else { $id = str_replace( WIDGET_BASE_ID . '-', '', $this->number ); } $number = $instance['num']; $start = $instance['offset'] + $number; $loading = $instance['loading_text']; $ret .= ''; $ret .= '
'; return $ret; } /** * Calculate the HTML of the footer based on the widget settings * * @param array $instance Array which contains the various settings. * @return string The HTML for the footer area * * @since 4.6 */ public function footerHTML( $instance ) { $ret = ''; $url = ''; $text = ''; if ( isset( $instance['footer_link'] ) ) { $url = $instance['footer_link']; } if ( isset( $instance['footer_link_text'] ) ) { $text = $instance['footer_link_text']; } // if url is set, but no text, just use the url as text. if ( empty( $text ) && ! empty( $url ) ) { $text = $url; } // if no url is set but just text, assume the url should be to the relevant archive page // category archive for categories filter and home page or blog page when "all categories" // is used. if ( ! empty( $text ) && empty( $url ) ) { if ( isset( $instance['cat'] ) && ( 0 !== $instance['cat'] ) && ( null !== get_category( $instance['cat'] ) ) ) { $url = get_category_link( $instance['cat'] ); } else { $url = $this->blog_page_url(); } } if ( ! empty( $url ) ) { $ret .= '' . esc_html( $text ) . ''; } return $ret; } /** * Current post item date string based on the format requested in the settings * * @param array $instance Array which contains the various settings. * @param bool $everything_is_link Indicates whether the return string should avoid links. * * @since 4.8 */ public function itemDate( $instance, $everything_is_link ) { $ret = ''; if ( ! isset( $instance['preset_date_format'] ) ) { $preset_date_format = 'other'; } else { $preset_date_format = $instance['preset_date_format']; } $attr = ''; switch ( $preset_date_format ) { case 'sitedateandtime': $date = get_the_time( get_option( 'date_format' ) . ' ' . get_option( 'time_format' ) ); break; case 'localsitedateandtime': $date = get_the_time( get_option( 'date_format' ) . ' ' . get_option( 'time_format' ) ) . ' GMT'; $time = get_post_time( 'U', true ); $attr = ' data-publishtime="' . $time . '" data-format="time"'; add_action( 'wp_footer', __NAMESPACE__ . '\embed_date_scripts' ); break; case 'sitedate': $date = get_the_time( get_option( 'date_format' ) ); break; case 'localsitedate': $date = get_the_time( get_option( 'date_format' ) ) . ' GMT'; $time = get_post_time( 'U', true ); $attr = ' data-publishtime="' . $time . '" data-format="date"'; add_action( 'wp_footer', __NAMESPACE__ . '\embed_date_scripts' ); break; case 'sincepublished': $date = human_time_diff( get_the_time( 'U' ), current_time( 'timestamp' ) ); break; default: if ( isset( $instance['date_format'] ) && strlen( trim( $instance['date_format'] ) ) > 0 ) { $date_format = $instance['date_format']; } else { $date_format = 'j M Y'; } $date = get_the_time( $date_format ); break; } $ret .= ''; if ( isset( $instance['date_link'] ) && $instance['date_link'] && ! $everything_is_link ) { $ret .= ''; } $ret .= $date; if ( isset( $instance['date_link'] ) && $instance['date_link'] && ! $everything_is_link ) { $ret .= ''; } $ret .= ''; return $ret; } /** * Calculate the HTML for showing the thumb of a post item. * Expected to be called from a loop with globals properly set. * * @param array $instance Array which contains the various settings. * @param bool $no_link indicates whether the thumb should be wrapped in a link or a span. * @return string The HTML for the thumb related to the post * * @since 4.6 */ public function itemThumb( $instance, $no_link ) { $ret = ''; if ( ( isset( $instance['default_thunmbnail'] ) && ( $instance['default_thunmbnail'] ) ) || has_post_thumbnail() ) { $class = ''; $disable_css = isset( $instance['disable_css'] ) && $instance['disable_css']; if ( isset( $this->instance['thumb_hover'] ) && ! $disable_css ) { $class = 'class="cat-post-thumbnail cat-post-' . $instance['thumb_hover'] . '"'; } else { $class = 'class="cat-post-thumbnail"'; } $title_args = array( 'echo' => false ); if ( $no_link ) { $ret .= ''; } else { $ret .= ''; } $ret .= $this->the_post_thumbnail( array( $this->instance['thumb_w'], $this->instance['thumb_h'] ) ); if ( $no_link ) { $ret .= ''; } else { $ret .= ''; } } return $ret; } /** * Current post item categories string * * @param array $instance Array which contains the various settings. * @param bool $everything_is_link Indicates whether the return string should avoid links. * * @since 4.8 */ public function itemCategories( $instance, $everything_is_link ) { $ret = ''; $cat_ids = wp_get_post_categories( get_the_ID(), array( 'number' => 0 ) ); foreach ( $cat_ids as $cat_id ) { if ( $everything_is_link ) { $ret .= ' ' . get_cat_name( $cat_id ); } else { $ret .= " " . get_cat_name( $cat_id ) . ''; } } $ret .= ''; return $ret; } /** * Current post item tags string * * @param array $instance Array which contains the various settings. * @param bool $everything_is_link Indicates whether the return string should avoid links. * * @since 4.8 */ public function itemTags( $instance, $everything_is_link ) { $ret = ''; $tag_ids = wp_get_post_tags( get_the_ID(), array( 'number' => 0 ) ); foreach ( $tag_ids as $tag_id ) { if ( $everything_is_link ) { $ret .= ' ' . $tag_id->name; } else { $ret .= " " . $tag_id->name . ''; } } $ret .= ''; return $ret; } /** * Current post item comment number string * * @param array $instance Array which contains the various settings. * @param bool $everything_is_link Indicates whether the return string should avoid links. * * @since 4.8 */ public function itemCommentNum( $instance, $everything_is_link ) { global $post; $ret = ''; if ( $everything_is_link ) { $ret .= '(' . \get_comments_number() . ')'; } else { $link = sprintf( '(%3$d)', esc_url( get_comments_link( $post->ID ) ), esc_attr( sprintf( __( '(%d) comments to this post' ), get_comments_number() ) ), get_comments_number() ); $ret .= $link; } $ret .= ''; return $ret; } /** * Current post item author string * * @param array $instance Array which contains the various settings. * @param bool $everything_is_link Indicates whether the return string should avoid links. * * @since 4.8 */ public function itemAuthor( $instance, $everything_is_link ) { $ret = ''; if ( $everything_is_link ) { $ret .= get_the_author(); } else { $link = get_the_author_posts_link(); $ret .= $link; } $ret .= ''; return $ret; } /** * Current post item excerpt string * * @param array $instance Array which contains the various settings. * @param bool $everything_is_link Indicates whether the return string should avoid links. * * @since 4.8 */ public function itemExcerpt( $instance, $everything_is_link ) { global $post; // use the_excerpt filter to get the "normal" excerpt of the post // then apply our filter to let users customize excerpts in their own way. if ( isset( $instance['excerpt_length'] ) && ( $instance['excerpt_length'] > 0 ) ) { $length = (int) $instance['excerpt_length']; } else { $length = 999; // Use the wordpress default. } if ( ! isset( $instance['excerpt_filters'] ) || $instance['excerpt_filters'] ) { // pre 4.7 widgets has filters on. $excerpt = apply_filters( 'the_excerpt', \get_the_excerpt() ); } else { // if filters off replicate functionality of core generating excerpt. $more_text = '[…]'; if ( isset( $instance['excerpt_more_text'] ) && $instance['excerpt_more_text'] ) { $more_text = ltrim( $instance['excerpt_more_text'] ); } if ( $everything_is_link ) { $excerpt_more_text = ' ' . $more_text . ''; } else { $excerpt_more_text = ' ' . $more_text . ''; } if ( '' === $post->post_excerpt ) { $text = get_the_content( '' ); $text = strip_shortcodes( $text ); $excerpt = \wp_trim_words( $text, $length, $excerpt_more_text ); // adjust html output same way as for the normal excerpt, // just force all functions depending on the_excerpt hook. $excerpt = shortcode_unautop( wpautop( convert_chars( convert_smilies( wptexturize( $excerpt ) ) ) ) ); } else { $text = $post->post_excerpt; $excerpt = \wp_trim_words( $text, $length, $excerpt_more_text ); $excerpt = shortcode_unautop( wpautop( convert_chars( convert_smilies( wptexturize( $excerpt ) ) ) ) ); } } $ret = apply_filters( 'cpw_excerpt', $excerpt, $this ); return $ret; } /** * Current post item title string * * @param array $instance Array which contains the various settings. * @param bool $everything_is_link Indicates whether the return string should avoid links. * * @since 4.8 */ public function itemTitle( $instance, $everything_is_link ) { $ret = ''; if ( $everything_is_link ) { $ret .= '' . get_the_title() . ''; } else { $ret .= '' . get_the_title(); $ret .= ''; } return $ret; } /** * Calculate the HTML for a post item based on the widget settings and post. * Expected to be called in an active loop with all the globals set. * * @param array $instance Array which contains the various settings. * @param null|integer $current_post_id If on singular page specifies the id of * the post, otherwise null. * @return string The HTML for item related to the post * * @since 4.6 */ public function itemHTML( $instance, $current_post_id ) { global $post; $everything_is_link = isset( $instance['everything_is_link'] ) && $instance['everything_is_link']; $wrap = isset( $instance['text_do_not_wrap_thumb'] ) && $instance['text_do_not_wrap_thumb']; $template = ''; if ( isset( $instance['template'] ) ) { $template = $instance['template']; } else { $template = convert_settings_to_template( $instance ); } $ret = '
  • ID ) { $ret .= "class='cat-post-item cat-post-current'"; } else { $ret .= "class='cat-post-item'"; } $ret .= '>'; // close the li opening tag. if ( $everything_is_link ) { $ret .= ''; } // Try to do smart formatting for floating thumb based on its location. if ( $wrap ) { if ( preg_match( '#(\%thumb\%)#', $template ) && ! preg_match( '#(\%thumb\%$)#', $template ) ) { $thumb_flex = explode( '%thumb%', $template ); if ( 1 === count( $thumb_flex ) ) { $template = '
    %thumb%
    ' . $thumb_flex[0] . '
    '; } if ( 2 === count( $thumb_flex ) ) { $template = $thumb_flex[0] . '
    %thumb%
    ' . $thumb_flex[1] . '
    '; } } } // Post details (Template). $widget = $this; $template_res = preg_replace_callback( get_template_regex(), function ( $matches ) use ( $widget, $instance, $everything_is_link ) { switch ( $matches[0] ) { case '%title%': return $widget->itemTitle( $instance, $everything_is_link ); case '%author%': return $widget->itemAuthor( $instance, $everything_is_link ); case '%commentnum%': return $widget->itemCommentNum( $instance, $everything_is_link ); case '%date%': return $widget->itemDate( $instance, $everything_is_link ); case '%thumb%': return $widget->itemThumb( $instance, $everything_is_link ); case '%post_tag%': return $widget->itemTags( $instance, $everything_is_link ); case '%category%': return $widget->itemCategories( $instance, $everything_is_link ); case '%excerpt%': return $widget->itemExcerpt( $instance, $everything_is_link ); default: return $matches[0]; } }, $template ); // Replace empty line with closing and opening DIV. $template_res = trim( $template_res ); $template_res = str_replace( "\n\r", '
    ', $template_res ); // in widget areas. $template_res = str_replace( "\n\n", '
    ', $template_res ); // as shortcode. $template_res = '
    ' . $template_res . '
    '; // replace new lines with spaces. $template_res = str_replace( "\n\r", ' ', $template_res ); // in widget areas. $template_res = str_replace( "\n\n", ' ', $template_res ); // as shortcode. $ret .= $template_res; if ( $everything_is_link ) { $ret .= '
    '; } $ret .= '
  • '; return $ret; } /** * Filter to set the number of words in an excerpt * * @param int $length The number of words as configured by wordpress core or set by previous filters. * @return int The number of words configured for the widget, * or the $length parameter if it is not configured or garbage value. * * @since 4.6 */ public function excerpt_length_filter( $length ) { if ( isset( $this->instance['excerpt_length'] ) && $this->instance['excerpt_length'] > 0 ) { $length = $this->instance['excerpt_length']; } return $length; } /** * Set the proper excerpt filters based on the settings * * @param array $instance widget settings. * @return void * * @since 4.6 */ public function setExcerpFilters( $instance ) { if ( isset( $instance['excerpt'] ) && $instance['excerpt'] ) { // Excerpt length filter. if ( isset( $instance['excerpt_length'] ) && ( (int) $instance['excerpt_length'] ) > 0 ) { add_filter( 'excerpt_length', array( $this, 'excerpt_length_filter' ) ); } if ( isset( $instance['excerpt_more_text'] ) && ( '' !== ltrim( $instance['excerpt_more_text'] ) ) ) { add_filter( 'excerpt_more', array( $this, 'excerpt_more_filter' ) ); } add_filter( 'the_excerpt', array( $this, 'apply_the_excerpt' ) ); } } /** * Remove the excerpt filter * * @param array $instance widget settings. * @return void * * @since 4.6 */ public function removeExcerpFilters( $instance ) { remove_filter( 'excerpt_length', array( $this, 'excerpt_length_filter' ) ); remove_filter( 'excerpt_more', array( $this, 'excerpt_more_filter' ) ); add_filter( 'get_the_excerpt', 'wp_trim_excerpt' ); remove_filter( 'the_excerpt', array( $this, 'apply_the_excerpt' ) ); } /** * The main widget display controller * * Called by the sidebar processing core logic to display the widget. * * @param array $args An array containing the "environment" setting for the widget, * namely, the enclosing tags for the widget and its title. * @param array $instance The settings associate with the widget. * * @since 4.1 */ public function widget( $args, $instance ) { $instance = upgrade_settings( $instance ); extract( $args ); $this->instance = $instance; $current_post_id = ''; if ( is_singular() ) { $current_post_id = get_the_ID(); } $items = $this->get_elements_HTML( $instance, $current_post_id, 0, 0 ); $cat_slug = $items->query_vars['category_name']; $cat_info = get_category_by_slug($cat_slug); if ( ( 'nothing' === $instance['no_match_handling'] ) || ! empty( $items ) ) { //echo $before_widget; // Xss ok. This is how widget actually expected to behave. $thumb = isset( $this->instance['template'] ) && preg_match( '/%thumb%/', $this->instance['template'] ); /*if ( ! ( isset( $instance['is_shortcode'] ) && $instance['is_shortcode'] ) ) { // the internal id is needed only for widgets. echo '