woocommerce/includes/class-wc-template-loader.php

407 lines
12 KiB
PHP
Raw Normal View History

2013-06-06 12:27:04 +00:00
<?php
/**
2015-11-03 13:53:50 +00:00
* Template Loader
*
* @class WC_Template
* @package WooCommerce/Classes
* @category Class
2017-11-08 19:57:49 +00:00
* @author Automattic
*/
if ( ! defined( 'ABSPATH' ) ) {
exit;
}
/**
* WC_Template_Loader.
*/
class WC_Template_Loader {
2013-06-06 12:27:04 +00:00
2017-11-09 10:44:29 +00:00
/**
* Store the shop page ID.
*
* @var integer
*/
private static $shop_page_id = 0;
2017-11-13 18:42:28 +00:00
/**
* Store whether we're processing a product inside the_content filter.
*
* @var boolean
*/
private static $in_content_filter = false;
/**
2015-11-03 13:31:20 +00:00
* Hook in methods.
*/
public static function init() {
2017-11-09 10:44:29 +00:00
self::$shop_page_id = wc_get_page_id( 'shop' );
// Supported themes.
if ( current_theme_supports( 'woocommerce' ) ) {
add_filter( 'template_include', array( __CLASS__, 'template_loader' ) );
add_filter( 'comments_template', array( __CLASS__, 'comments_template_loader' ) );
} else {
2017-11-13 18:42:28 +00:00
// Unsupported themes.
add_action( 'template_redirect', array( __CLASS__, 'unsupported_theme_init' ) );
}
}
/**
* Hook in methods to enhance the unsupported theme experience on Shop and Product pages.
*
* @since 3.3.0
*/
public static function unsupported_theme_init() {
2017-11-29 21:46:37 +00:00
// todo limit this to only WC taxes
if ( is_tax() ) {
self::unsupported_theme_tax_archive_init();
} elseif ( self::$shop_page_id || is_product() ) {
add_filter( 'the_content', array( __CLASS__, 'unsupported_theme_content_filter' ), 10 );
add_filter( 'the_title', array( __CLASS__, 'unsupported_theme_title_filter' ), 10, 2 );
add_filter( 'post_thumbnail_html', array( __CLASS__, 'unsupported_theme_single_featured_image_filter' ) );
add_filter( 'woocommerce_product_tabs', array( __CLASS__, 'unsupported_theme_remove_review_tab' ) );
2017-11-09 10:44:29 +00:00
add_filter( 'comments_number', '__return_empty_string' );
if ( is_product() ) {
remove_action( 'woocommerce_before_main_content', 'woocommerce_output_content_wrapper', 10 );
remove_action( 'woocommerce_after_main_content', 'woocommerce_output_content_wrapper_end', 10 );
add_theme_support( 'wc-product-gallery-zoom' );
add_theme_support( 'wc-product-gallery-lightbox' );
add_theme_support( 'wc-product-gallery-slider' );
}
2017-11-09 10:44:29 +00:00
}
}
2017-11-29 22:30:43 +00:00
// need to make this think it's a single page better
2017-11-29 21:46:37 +00:00
private static function unsupported_theme_tax_archive_init() {
2017-11-29 22:30:43 +00:00
global $wp_query, $post, $wp_filters;
2017-11-29 21:46:37 +00:00
$queried_object = get_queried_object();
$tax = $queried_object->taxonomy;
$term_slug = $queried_object->slug;
$dummy_post_properties = array(
'ID' => 0,
'post_status' => 'publish',
'post_author' => 0,
'post_parent' => 0,
'post_type' => 'page',
'post_date' => 0,
'post_date_gmt' => 0,
'post_modified' => 0,
'post_modified_gmt' => 0,
'post_content' => do_shortcode( '[product_category category="' . $term_slug . '"]' ),
'post_title' => $queried_object->name,
2017-11-29 22:30:43 +00:00
'post_excerpt' => '',
2017-11-29 21:46:37 +00:00
'post_content_filtered' => '',
'post_mime_type' => '',
'post_password' => '',
'post_name' => '',
'guid' => '',
'menu_order' => 0,
'pinged' => '',
'to_ping' => '',
'ping_status' => '',
'comment_status' => 'closed',
'comment_count' => 0,
'filter' => 'raw',
);
// Set the $post global.
$post = new WP_Post( (object) $dummy_post_properties );
// Copy the new post global into the main $wp_query.
$wp_query->post = $post;
$wp_query->posts = array( $post );
// Prevent comments form from appearing
$wp_query->post_count = 1;
$wp_query->is_404 = false;
2017-11-29 22:30:43 +00:00
$wp_query->is_page = true;
$wp_query->is_single = true;
2017-11-29 21:46:37 +00:00
$wp_query->is_archive = false;
$wp_query->is_tax = false;
setup_postdata( $post );
remove_all_filters( 'the_content' );
2017-11-29 22:30:43 +00:00
remove_all_filters( 'the_excerpt' );
add_filter( 'template_include', function( $template ){
$possible_templates = array(
'page',
'single',
'singular',
'index',
);
foreach ( $possible_templates as $possible_template ) {
$path = get_query_template( $possible_template );
if ( $path ) {
return $path;
}
}
return $template;
});
2017-11-29 21:46:37 +00:00
}
2017-11-09 10:44:29 +00:00
/**
* Get information about the current shop page view.
*
* @since 3.3.0
* @return array
*/
private static function get_current_shop_view_args() {
return (object) array(
'page' => absint( max( 1, absint( get_query_var( 'paged' ) ) ) ),
2017-11-09 17:34:54 +00:00
'columns' => wc_get_default_products_per_row(),
'rows' => wc_get_default_product_rows_per_page(),
2017-11-09 10:44:29 +00:00
);
2017-11-08 16:46:15 +00:00
}
2013-06-06 12:27:04 +00:00
/**
* Load a template.
*
* Handles template usage so that we can use our own templates instead of the themes.
*
2015-11-03 13:31:20 +00:00
* Templates are in the 'templates' folder. woocommerce looks for theme.
* overrides in /theme/woocommerce/ by default.
2013-06-06 12:27:04 +00:00
*
2015-11-03 13:31:20 +00:00
* For beginners, it also looks for a woocommerce.php template first. If the user adds.
* this to the theme (containing a woocommerce() inside) this will be used for all.
2013-06-06 12:27:04 +00:00
* woocommerce templates.
*
2017-11-09 10:44:29 +00:00
* @param string $template Template to load.
2013-06-06 12:27:04 +00:00
* @return string
*/
public static function template_loader( $template ) {
2017-11-09 10:44:29 +00:00
if ( is_embed() ) {
2016-04-25 09:27:53 +00:00
return $template;
}
2016-12-06 11:52:15 +00:00
if ( $default_file = self::get_template_loader_default_file() ) {
/**
* Filter hook to choose which files to find before WooCommerce does it's own logic.
*
2017-03-15 16:36:53 +00:00
* @since 3.0.0
2016-12-06 11:52:15 +00:00
* @var array
*/
$search_files = self::get_template_loader_files( $default_file );
$template = locate_template( $search_files );
2013-06-06 12:27:04 +00:00
2016-12-06 11:52:15 +00:00
if ( ! $template || WC_TEMPLATE_DEBUG_MODE ) {
$template = WC()->plugin_path() . '/templates/' . $default_file;
}
}
2013-06-06 12:27:04 +00:00
2016-12-06 11:52:15 +00:00
return $template;
}
2013-06-06 12:27:04 +00:00
2016-12-06 11:52:15 +00:00
/**
* Get the default filename for a template.
*
2017-03-15 16:36:53 +00:00
* @since 3.0.0
2016-12-06 11:52:15 +00:00
* @return string
*/
private static function get_template_loader_default_file() {
if ( is_singular( 'product' ) ) {
$default_file = 'single-product.php';
} elseif ( is_product_taxonomy() ) {
$object = get_queried_object();
2013-06-06 12:27:04 +00:00
if ( is_tax( 'product_cat' ) || is_tax( 'product_tag' ) ) {
$default_file = 'taxonomy-' . $object->taxonomy . '.php';
} else {
2016-12-06 11:52:15 +00:00
$default_file = 'archive-product.php';
}
2016-12-06 11:52:15 +00:00
} elseif ( is_post_type_archive( 'product' ) || is_page( wc_get_page_id( 'shop' ) ) ) {
2017-11-09 10:44:29 +00:00
$default_file = current_theme_supports( 'woocommerce' ) ? 'archive-product.php' : '';
2016-12-06 11:52:15 +00:00
} else {
$default_file = '';
}
return $default_file;
}
2016-12-06 11:52:15 +00:00
/**
* Get an array of filenames to search for a given template.
*
2017-03-15 16:36:53 +00:00
* @since 3.0.0
* @param string $default_file The default file name.
2016-12-06 11:52:15 +00:00
* @return string[]
*/
private static function get_template_loader_files( $default_file ) {
$templates = apply_filters( 'woocommerce_template_loader_files', array(), $default_file );
$templates[] = 'woocommerce.php';
2016-12-06 11:52:15 +00:00
if ( is_page_template() ) {
$templates[] = get_page_template_slug();
}
if ( is_singular( 'product' ) ) {
$object = get_queried_object();
$name_decoded = urldecode( $object->post_name );
if ( $name_decoded !== $object->post_name ) {
$templates[] = "single-product-{$name_decoded}.php";
}
$templates[] = "single-product-{$object->post_name}.php";
}
2016-12-06 11:52:15 +00:00
if ( is_product_taxonomy() ) {
2017-11-15 09:41:26 +00:00
$object = get_queried_object();
$templates[] = 'taxonomy-' . $object->taxonomy . '-' . $object->slug . '.php';
$templates[] = WC()->template_path() . 'taxonomy-' . $object->taxonomy . '-' . $object->slug . '.php';
$templates[] = 'taxonomy-' . $object->taxonomy . '.php';
$templates[] = WC()->template_path() . 'taxonomy-' . $object->taxonomy . '.php';
2013-06-06 12:27:04 +00:00
}
$templates[] = $default_file;
$templates[] = WC()->template_path() . $default_file;
2013-06-06 12:27:04 +00:00
return array_unique( $templates );
2013-06-06 12:27:04 +00:00
}
/**
* Load comments template.
2013-06-06 12:27:04 +00:00
*
2017-11-08 19:57:49 +00:00
* @param string $template template to load.
2013-06-06 12:27:04 +00:00
* @return string
*/
public static function comments_template_loader( $template ) {
2017-11-09 10:44:29 +00:00
if ( get_post_type() !== 'product' ) {
2013-06-06 12:27:04 +00:00
return $template;
}
$check_dirs = array(
trailingslashit( get_stylesheet_directory() ) . WC()->template_path(),
trailingslashit( get_template_directory() ) . WC()->template_path(),
trailingslashit( get_stylesheet_directory() ),
trailingslashit( get_template_directory() ),
trailingslashit( WC()->plugin_path() ) . 'templates/',
);
if ( WC_TEMPLATE_DEBUG_MODE ) {
$check_dirs = array( array_pop( $check_dirs ) );
}
2013-06-06 12:27:04 +00:00
foreach ( $check_dirs as $dir ) {
if ( file_exists( trailingslashit( $dir ) . 'single-product-reviews.php' ) ) {
return trailingslashit( $dir ) . 'single-product-reviews.php';
}
}
2013-06-06 12:27:04 +00:00
}
/**
* Filter the title and insert WooCommerce content on the shop page.
*
* For non-WC themes, this will setup the main shop page to be shortcode based to improve default appearance.
*
* @since 3.3.0
* @param string $title Existing title.
* @param int $id ID of the post being filtered.
* @return string
*/
public static function unsupported_theme_title_filter( $title, $id ) {
if ( ! current_theme_supports( 'woocommerce' ) && is_page( self::$shop_page_id ) && $id === self::$shop_page_id ) {
$args = self::get_current_shop_view_args();
$title_suffix = array();
if ( $args->page > 1 ) {
$title_suffix[] = sprintf( esc_html__( 'Page %d', 'woocommerce' ), $args->page );
}
if ( $title_suffix ) {
$title = $title . ' &ndash; ' . implode( ', ', $title_suffix );
}
}
return $title;
}
/**
* Filter the content and insert WooCommerce content on the shop page.
*
* For non-WC themes, this will setup the main shop page to be shortcode based to improve default appearance.
*
* @since 3.3.0
* @param string $content Existing post content.
* @return string
*/
public static function unsupported_theme_content_filter( $content ) {
global $wp_query;
if ( current_theme_supports( 'woocommerce' ) || ! is_main_query() ) {
return $content;
}
self::$in_content_filter = true;
// Remove the filter we're in to avoid nested calls.
remove_filter( 'the_content', array( __CLASS__, 'the_content_filter' ) );
// Unsupported theme shop page.
if ( is_page( self::$shop_page_id ) ) {
$args = self::get_current_shop_view_args();
$shortcode = new WC_Shortcode_Products(
array_merge(
wc()->query->get_catalog_ordering_args(),
array(
'page' => $args->page,
'columns' => $args->columns,
'rows' => $args->rows,
'orderby' => '',
'order' => '',
'paginate' => true,
'cache' => false,
)
),
'products' );
// Allow queries to run e.g. layered nav.
add_action( 'pre_get_posts', array( wc()->query, 'product_query' ) );
$content = $content . $shortcode->get_content();
// Remove actions and self to avoid nested calls.
remove_action( 'pre_get_posts', array( wc()->query, 'product_query' ) );
2017-11-13 18:42:28 +00:00
// Unsupported theme product page.
} elseif ( is_product() ) {
$content = do_shortcode( '[product_page id="' . get_the_ID() . '" show_title=0]' );
}
self::$in_content_filter = false;
return $content;
}
/**
* Prevent the main featured image on product pages because there will be another featured image
* in the gallery.
*
* @since 3.3.0
* @param string $html Img element HTML.
* @return string
*/
public static function unsupported_theme_single_featured_image_filter( $html ) {
if ( self::$in_content_filter || ! is_product() || ! is_main_query() ) {
return $html;
}
return '';
}
/**
* Remove the Review tab and just use the regular comment form.
*
* @param array $tabs Tab info.
* @return array
*/
public static function unsupported_theme_remove_review_tab( $tabs ) {
unset( $tabs['reviews'] );
return $tabs;
}
}
add_action( 'init', array( 'WC_Template_Loader', 'init' ) );