867 lines
33 KiB
PHP
867 lines
33 KiB
PHP
<?php
|
|
/**
|
|
* Post Types Admin
|
|
*
|
|
* @author Automattic
|
|
* @category Admin
|
|
* @package WooCommerce/admin
|
|
* @version 3.3.0
|
|
*/
|
|
|
|
if ( ! defined( 'ABSPATH' ) ) {
|
|
exit;
|
|
}
|
|
|
|
if ( class_exists( 'WC_Admin_Post_Types', false ) ) {
|
|
new WC_Admin_Post_Types();
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* WC_Admin_Post_Types Class.
|
|
*
|
|
* Handles the edit posts views and some functionality on the edit post screen for WC post types.
|
|
*/
|
|
class WC_Admin_Post_Types {
|
|
|
|
/**
|
|
* Constructor.
|
|
*/
|
|
public function __construct() {
|
|
include_once( dirname( __FILE__ ) . '/class-wc-admin-meta-boxes.php' );
|
|
|
|
if ( ! function_exists( 'duplicate_post_plugin_activation' ) ) {
|
|
include_once( 'class-wc-admin-duplicate-product.php' );
|
|
}
|
|
|
|
// Load correct list table classes for current screen.
|
|
add_action( 'current_screen', array( $this, 'setup_screen' ) );
|
|
add_action( 'check_ajax_referer', array( $this, 'setup_screen' ) );
|
|
|
|
// Admin notices.
|
|
add_filter( 'post_updated_messages', array( $this, 'post_updated_messages' ) );
|
|
add_filter( 'bulk_post_updated_messages', array( $this, 'bulk_post_updated_messages' ), 10, 2 );
|
|
|
|
// Disable Auto Save.
|
|
add_action( 'admin_print_scripts', array( $this, 'disable_autosave' ) );
|
|
|
|
// Extra post data and screen elements.
|
|
add_action( 'edit_form_top', array( $this, 'edit_form_top' ) );
|
|
add_filter( 'enter_title_here', array( $this, 'enter_title_here' ), 1, 2 );
|
|
add_action( 'edit_form_after_title', array( $this, 'edit_form_after_title' ) );
|
|
add_filter( 'default_hidden_meta_boxes', array( $this, 'hidden_meta_boxes' ), 10, 2 );
|
|
add_action( 'post_submitbox_misc_actions', array( $this, 'product_data_visibility' ) );
|
|
|
|
// Uploads.
|
|
add_filter( 'upload_dir', array( $this, 'upload_dir' ) );
|
|
add_action( 'media_upload_downloadable_product', array( $this, 'media_upload_downloadable_product' ) );
|
|
|
|
// Hide template for CPT archive.
|
|
add_filter( 'theme_page_templates', array( $this, 'hide_cpt_archive_templates' ), 10, 3 );
|
|
add_action( 'edit_form_top', array( $this, 'show_cpt_archive_notice' ) );
|
|
|
|
// Add a post display state for special WC pages.
|
|
add_filter( 'display_post_states', array( $this, 'add_display_post_states' ), 10, 2 );
|
|
|
|
// Bulk / quick edit.
|
|
add_action( 'bulk_edit_custom_box', array( $this, 'bulk_edit' ), 10, 2 );
|
|
add_action( 'quick_edit_custom_box', array( $this, 'quick_edit' ), 10, 2 );
|
|
add_action( 'save_post', array( $this, 'bulk_and_quick_edit_hook' ), 10, 2 );
|
|
add_action( 'woocommerce_product_bulk_and_quick_edit', array( $this, 'bulk_and_quick_edit_save_post' ), 10, 2 );
|
|
}
|
|
|
|
/**
|
|
* Looks at the current screen and loads the correct list table handler.
|
|
*
|
|
* @since 3.3.0
|
|
*/
|
|
public function setup_screen() {
|
|
$screen_id = false;
|
|
|
|
if ( function_exists( 'get_current_screen' ) ) {
|
|
$screen = get_current_screen();
|
|
$screen_id = isset( $screen, $screen->id ) ? $screen->id : '';
|
|
}
|
|
|
|
if ( ! empty( $_REQUEST['screen'] ) ) { // WPCS: input var ok.
|
|
$screen_id = wc_clean( wp_unslash( $_REQUEST['screen'] ) ); // WPCS: input var ok, sanitization ok.
|
|
}
|
|
|
|
switch ( $screen_id ) {
|
|
case 'edit-shop_order' :
|
|
include_once( 'list-tables/class-wc-admin-list-table-orders.php' );
|
|
break;
|
|
case 'edit-shop_coupon' :
|
|
include_once( 'list-tables/class-wc-admin-list-table-coupons.php' );
|
|
break;
|
|
case 'edit-product' :
|
|
include_once( 'list-tables/class-wc-admin-list-table-products.php' );
|
|
break;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Change messages when a post type is updated.
|
|
*
|
|
* @param array $messages Array of messages.
|
|
* @return array
|
|
*/
|
|
public function post_updated_messages( $messages ) {
|
|
global $post;
|
|
|
|
$messages['product'] = array(
|
|
0 => '', // Unused. Messages start at index 1.
|
|
1 => sprintf( __( 'Product updated. <a href="%s">View Product</a>', 'woocommerce' ), esc_url( get_permalink( $post->ID ) ) ),
|
|
2 => __( 'Custom field updated.', 'woocommerce' ),
|
|
3 => __( 'Custom field deleted.', 'woocommerce' ),
|
|
4 => __( 'Product updated.', 'woocommerce' ),
|
|
5 => __( 'Revision restored.', 'woocommerce' ),
|
|
/* translators: %s: product url */
|
|
6 => sprintf( __( 'Product published. <a href="%s">View Product</a>', 'woocommerce' ), esc_url( get_permalink( $post->ID ) ) ),
|
|
7 => __( 'Product saved.', 'woocommerce' ),
|
|
/* translators: %s: product url */
|
|
8 => sprintf( __( 'Product submitted. <a target="_blank" href="%s">Preview product</a>', 'woocommerce' ), esc_url( add_query_arg( 'preview', 'true', get_permalink( $post->ID ) ) ) ),
|
|
9 => sprintf(
|
|
/* translators: 1: date 2: product url */
|
|
__( 'Product scheduled for: %1$s. <a target="_blank" href="%2$s">Preview product</a>', 'woocommerce' ),
|
|
'<strong>' . date_i18n( __( 'M j, Y @ G:i', 'woocommerce' ), strtotime( $post->post_date ) ), esc_url( get_permalink( $post->ID ) ) . '</strong>'
|
|
),
|
|
/* translators: %s: product url */
|
|
10 => sprintf( __( 'Product draft updated. <a target="_blank" href="%s">Preview product</a>', 'woocommerce' ), esc_url( add_query_arg( 'preview', 'true', get_permalink( $post->ID ) ) ) ),
|
|
);
|
|
|
|
$messages['shop_order'] = array(
|
|
0 => '', // Unused. Messages start at index 1.
|
|
1 => __( 'Order updated.', 'woocommerce' ),
|
|
2 => __( 'Custom field updated.', 'woocommerce' ),
|
|
3 => __( 'Custom field deleted.', 'woocommerce' ),
|
|
4 => __( 'Order updated.', 'woocommerce' ),
|
|
5 => __( 'Revision restored.', 'woocommerce' ),
|
|
6 => __( 'Order updated.', 'woocommerce' ),
|
|
7 => __( 'Order saved.', 'woocommerce' ),
|
|
8 => __( 'Order submitted.', 'woocommerce' ),
|
|
9 => sprintf(
|
|
/* translators: %s: date */
|
|
__( 'Order scheduled for: %s.', 'woocommerce' ),
|
|
'<strong>' . date_i18n( __( 'M j, Y @ G:i', 'woocommerce' ), strtotime( $post->post_date ) ) . '</strong>'
|
|
),
|
|
10 => __( 'Order draft updated.', 'woocommerce' ),
|
|
11 => __( 'Order updated and sent.', 'woocommerce' ),
|
|
);
|
|
|
|
$messages['shop_coupon'] = array(
|
|
0 => '', // Unused. Messages start at index 1.
|
|
1 => __( 'Coupon updated.', 'woocommerce' ),
|
|
2 => __( 'Custom field updated.', 'woocommerce' ),
|
|
3 => __( 'Custom field deleted.', 'woocommerce' ),
|
|
4 => __( 'Coupon updated.', 'woocommerce' ),
|
|
5 => __( 'Revision restored.', 'woocommerce' ),
|
|
6 => __( 'Coupon updated.', 'woocommerce' ),
|
|
7 => __( 'Coupon saved.', 'woocommerce' ),
|
|
8 => __( 'Coupon submitted.', 'woocommerce' ),
|
|
9 => sprintf(
|
|
/* translators: %s: date */
|
|
__( 'Coupon scheduled for: %s.', 'woocommerce' ),
|
|
'<strong>' . date_i18n( __( 'M j, Y @ G:i', 'woocommerce' ), strtotime( $post->post_date ) ) . '</strong>'
|
|
),
|
|
10 => __( 'Coupon draft updated.', 'woocommerce' ),
|
|
);
|
|
|
|
return $messages;
|
|
}
|
|
|
|
/**
|
|
* Specify custom bulk actions messages for different post types.
|
|
*
|
|
* @param array $bulk_messages Array of messages.
|
|
* @param array $bulk_counts Array of how many objects were updated.
|
|
* @return array
|
|
*/
|
|
public function bulk_post_updated_messages( $bulk_messages, $bulk_counts ) {
|
|
$bulk_messages['product'] = array(
|
|
/* translators: %s: product count */
|
|
'updated' => _n( '%s product updated.', '%s products updated.', $bulk_counts['updated'], 'woocommerce' ),
|
|
/* translators: %s: product count */
|
|
'locked' => _n( '%s product not updated, somebody is editing it.', '%s products not updated, somebody is editing them.', $bulk_counts['locked'], 'woocommerce' ),
|
|
/* translators: %s: product count */
|
|
'deleted' => _n( '%s product permanently deleted.', '%s products permanently deleted.', $bulk_counts['deleted'], 'woocommerce' ),
|
|
/* translators: %s: product count */
|
|
'trashed' => _n( '%s product moved to the Trash.', '%s products moved to the Trash.', $bulk_counts['trashed'], 'woocommerce' ),
|
|
/* translators: %s: product count */
|
|
'untrashed' => _n( '%s product restored from the Trash.', '%s products restored from the Trash.', $bulk_counts['untrashed'], 'woocommerce' ),
|
|
);
|
|
|
|
$bulk_messages['shop_order'] = array(
|
|
/* translators: %s: order count */
|
|
'updated' => _n( '%s order updated.', '%s orders updated.', $bulk_counts['updated'], 'woocommerce' ),
|
|
/* translators: %s: order count */
|
|
'locked' => _n( '%s order not updated, somebody is editing it.', '%s orders not updated, somebody is editing them.', $bulk_counts['locked'], 'woocommerce' ),
|
|
/* translators: %s: order count */
|
|
'deleted' => _n( '%s order permanently deleted.', '%s orders permanently deleted.', $bulk_counts['deleted'], 'woocommerce' ),
|
|
/* translators: %s: order count */
|
|
'trashed' => _n( '%s order moved to the Trash.', '%s orders moved to the Trash.', $bulk_counts['trashed'], 'woocommerce' ),
|
|
/* translators: %s: order count */
|
|
'untrashed' => _n( '%s order restored from the Trash.', '%s orders restored from the Trash.', $bulk_counts['untrashed'], 'woocommerce' ),
|
|
);
|
|
|
|
$bulk_messages['shop_coupon'] = array(
|
|
/* translators: %s: coupon count */
|
|
'updated' => _n( '%s coupon updated.', '%s coupons updated.', $bulk_counts['updated'], 'woocommerce' ),
|
|
/* translators: %s: coupon count */
|
|
'locked' => _n( '%s coupon not updated, somebody is editing it.', '%s coupons not updated, somebody is editing them.', $bulk_counts['locked'], 'woocommerce' ),
|
|
/* translators: %s: coupon count */
|
|
'deleted' => _n( '%s coupon permanently deleted.', '%s coupons permanently deleted.', $bulk_counts['deleted'], 'woocommerce' ),
|
|
/* translators: %s: coupon count */
|
|
'trashed' => _n( '%s coupon moved to the Trash.', '%s coupons moved to the Trash.', $bulk_counts['trashed'], 'woocommerce' ),
|
|
/* translators: %s: coupon count */
|
|
'untrashed' => _n( '%s coupon restored from the Trash.', '%s coupons restored from the Trash.', $bulk_counts['untrashed'], 'woocommerce' ),
|
|
);
|
|
|
|
return $bulk_messages;
|
|
}
|
|
|
|
/**
|
|
* Custom bulk edit - form.
|
|
*
|
|
* @param string $column_name Column being shown.
|
|
* @param string $post_type Post type being shown.
|
|
*/
|
|
public function bulk_edit( $column_name, $post_type ) {
|
|
if ( 'price' !== $column_name || 'product' !== $post_type ) {
|
|
return;
|
|
}
|
|
|
|
$shipping_class = get_terms( 'product_shipping_class', array(
|
|
'hide_empty' => false,
|
|
) );
|
|
|
|
include( WC()->plugin_path() . '/includes/admin/views/html-bulk-edit-product.php' );
|
|
}
|
|
|
|
/**
|
|
* Custom quick edit - form.
|
|
*
|
|
* @param string $column_name Column being shown.
|
|
* @param string $post_type Post type being shown.
|
|
*/
|
|
public function quick_edit( $column_name, $post_type ) {
|
|
if ( 'price' !== $column_name || 'product' !== $post_type ) {
|
|
return;
|
|
}
|
|
|
|
$shipping_class = get_terms( 'product_shipping_class', array(
|
|
'hide_empty' => false,
|
|
) );
|
|
|
|
include( WC()->plugin_path() . '/includes/admin/views/html-quick-edit-product.php' );
|
|
}
|
|
|
|
/**
|
|
* Offers a way to hook into save post without causing an infinite loop
|
|
* when quick/bulk saving product info.
|
|
*
|
|
* @since 3.0.0
|
|
* @param int $post_id Post ID being saved.
|
|
* @param object $post Post object being saved.
|
|
*/
|
|
public function bulk_and_quick_edit_hook( $post_id, $post ) {
|
|
remove_action( 'save_post', array( $this, 'bulk_and_quick_edit_hook' ) );
|
|
do_action( 'woocommerce_product_bulk_and_quick_edit', $post_id, $post );
|
|
add_action( 'save_post', array( $this, 'bulk_and_quick_edit_hook' ), 10, 2 );
|
|
}
|
|
|
|
/**
|
|
* Quick and bulk edit saving.
|
|
*
|
|
* @param int $post_id Post ID being saved.
|
|
* @param object $post Post object being saved.
|
|
* @return int
|
|
*/
|
|
public function bulk_and_quick_edit_save_post( $post_id, $post ) {
|
|
// If this is an autosave, our form has not been submitted, so we don't want to do anything.
|
|
if ( defined( 'DOING_AUTOSAVE' ) && DOING_AUTOSAVE ) {
|
|
return $post_id;
|
|
}
|
|
|
|
// Don't save revisions and autosaves.
|
|
if ( wp_is_post_revision( $post_id ) || wp_is_post_autosave( $post_id ) || 'product' !== $post->post_type || ! current_user_can( 'edit_post', $post_id ) ) {
|
|
return $post_id;
|
|
}
|
|
|
|
// Check nonce.
|
|
if ( ! isset( $_REQUEST['woocommerce_quick_edit_nonce'] ) || ! wp_verify_nonce( $_REQUEST['woocommerce_quick_edit_nonce'], 'woocommerce_quick_edit_nonce' ) ) { // WPCS: input var ok, sanitization ok.
|
|
return $post_id;
|
|
}
|
|
|
|
// Get the product and save.
|
|
$product = wc_get_product( $post );
|
|
|
|
if ( ! empty( $_REQUEST['woocommerce_quick_edit'] ) ) { // WPCS: input var ok.
|
|
$this->quick_edit_save( $post_id, $product );
|
|
} else {
|
|
$this->bulk_edit_save( $post_id, $product );
|
|
}
|
|
|
|
return $post_id;
|
|
}
|
|
|
|
/**
|
|
* Quick edit.
|
|
*
|
|
* @param int $post_id Post ID being saved.
|
|
* @param WC_Product $product Product object.
|
|
*/
|
|
private function quick_edit_save( $post_id, $product ) {
|
|
$data_store = $product->get_data_store();
|
|
$old_regular_price = $product->get_regular_price();
|
|
$old_sale_price = $product->get_sale_price();
|
|
$input_to_props = array(
|
|
'_weight' => 'weight',
|
|
'_length' => 'length',
|
|
'_width' => 'width',
|
|
'_height' => 'height',
|
|
'_visibility' => 'catalog_visibility',
|
|
'_tax_class' => 'tax_class',
|
|
'_tax_status' => 'tax_status',
|
|
);
|
|
|
|
foreach ( $input_to_props as $input_var => $prop ) {
|
|
if ( isset( $_REQUEST[ $input_var ] ) ) { // WPCS: input var ok, sanitization ok.
|
|
$product->{"set_{$prop}"}( wc_clean( wp_unslash( $_REQUEST[ $input_var ] ) ) ); // WPCS: input var ok, sanitization ok.
|
|
}
|
|
}
|
|
|
|
if ( isset( $_REQUEST['_sku'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
$sku = $product->get_sku();
|
|
$new_sku = (string) wc_clean( $_REQUEST['_sku'] ); // WPCS: input var ok, sanitization ok.
|
|
|
|
if ( $new_sku !== $sku ) {
|
|
if ( ! empty( $new_sku ) ) {
|
|
$unique_sku = wc_product_has_unique_sku( $post_id, $new_sku );
|
|
if ( $unique_sku ) {
|
|
$product->set_sku( $new_sku );
|
|
}
|
|
} else {
|
|
$product->set_sku( '' );
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( ! empty( $_REQUEST['_shipping_class'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
if ( '_no_shipping_class' === $_REQUEST['_shipping_class'] ) { // WPCS: input var ok, sanitization ok.
|
|
$product->set_shipping_class_id( 0 );
|
|
} else {
|
|
$shipping_class_id = $data_store->get_shipping_class_id_by_slug( wc_clean( $_REQUEST['_shipping_class'] ) ); // WPCS: input var ok, sanitization ok.
|
|
$product->set_shipping_class_id( $shipping_class_id );
|
|
}
|
|
}
|
|
|
|
$product->set_featured( isset( $_REQUEST['_featured'] ) ); // WPCS: input var ok, sanitization ok.
|
|
|
|
if ( $product->is_type( 'simple' ) || $product->is_type( 'external' ) ) {
|
|
|
|
if ( isset( $_REQUEST['_regular_price'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
$new_regular_price = ( '' === $_REQUEST['_regular_price'] ) ? '' : wc_format_decimal( $_REQUEST['_regular_price'] ); // WPCS: input var ok, sanitization ok.
|
|
$product->set_regular_price( $new_regular_price );
|
|
} else {
|
|
$new_regular_price = null;
|
|
}
|
|
if ( isset( $_REQUEST['_sale_price'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
$new_sale_price = ( '' === $_REQUEST['_sale_price'] ) ? '' : wc_format_decimal( $_REQUEST['_sale_price'] ); // WPCS: input var ok, sanitization ok.
|
|
$product->set_sale_price( $new_sale_price );
|
|
} else {
|
|
$new_sale_price = null;
|
|
}
|
|
|
|
// Handle price - remove dates and set to lowest.
|
|
$price_changed = false;
|
|
|
|
if ( ! is_null( $new_regular_price ) && $new_regular_price !== $old_regular_price ) {
|
|
$price_changed = true;
|
|
} elseif ( ! is_null( $new_sale_price ) && $new_sale_price !== $old_sale_price ) {
|
|
$price_changed = true;
|
|
}
|
|
|
|
if ( $price_changed ) {
|
|
$product->set_date_on_sale_to( '' );
|
|
$product->set_date_on_sale_from( '' );
|
|
}
|
|
}
|
|
|
|
// Handle Stock Data.
|
|
$manage_stock = ! empty( $_REQUEST['_manage_stock'] ) && 'grouped' !== $product->get_type() ? 'yes' : 'no'; // WPCS: input var ok, sanitization ok.
|
|
$backorders = ! empty( $_REQUEST['_backorders'] ) ? wc_clean( $_REQUEST['_backorders'] ) : 'no'; // WPCS: input var ok, sanitization ok.
|
|
$stock_status = ! empty( $_REQUEST['_stock_status'] ) ? wc_clean( $_REQUEST['_stock_status'] ) : 'instock'; // WPCS: input var ok, sanitization ok.
|
|
$stock_amount = 'yes' === $manage_stock && ! empty( $_REQUEST['_stock'] ) ? wc_stock_amount( $_REQUEST['_stock'] ) : ''; // WPCS: input var ok, sanitization ok.
|
|
|
|
$product->set_manage_stock( $manage_stock );
|
|
$product->set_backorders( $backorders );
|
|
|
|
if ( 'yes' === get_option( 'woocommerce_manage_stock' ) ) {
|
|
$product->set_stock_quantity( $stock_amount );
|
|
}
|
|
|
|
// Apply product type constraints to stock status.
|
|
if ( $product->is_type( 'external' ) ) {
|
|
// External products are always in stock.
|
|
$product->set_stock_status( 'instock' );
|
|
} elseif ( $product->is_type( 'variable' ) && ! $product->get_manage_stock() ) {
|
|
// Stock status is determined by children.
|
|
foreach ( $product->get_children() as $child_id ) {
|
|
$child = wc_get_product( $child_id );
|
|
if ( ! $product->get_manage_stock() ) {
|
|
$child->set_stock_status( $stock_status );
|
|
$child->save();
|
|
}
|
|
}
|
|
$product = WC_Product_Variable::sync( $product, false );
|
|
} else {
|
|
$product->set_stock_status( $stock_status );
|
|
}
|
|
|
|
$product->save();
|
|
|
|
do_action( 'woocommerce_product_quick_edit_save', $product );
|
|
}
|
|
|
|
/**
|
|
* Bulk edit.
|
|
*
|
|
* @param int $post_id Post ID being saved.
|
|
* @param WC_Product $product Product object.
|
|
*/
|
|
public function bulk_edit_save( $post_id, $product ) {
|
|
$data_store = $product->get_data_store();
|
|
$old_regular_price = $product->get_regular_price();
|
|
$old_sale_price = $product->get_sale_price();
|
|
$data = wp_unslash( $_REQUEST ); // WPCS: input var ok, CSRF ok.
|
|
|
|
if ( ! empty( $_REQUEST['change_weight'] ) && isset( $_REQUEST['_weight'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
$product->set_weight( wc_clean( wp_unslash( $_REQUEST['_weight'] ) ) ); // WPCS: input var ok, sanitization ok.
|
|
}
|
|
|
|
if ( ! empty( $_REQUEST['change_dimensions'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
if ( isset( $_REQUEST['_length'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
$product->set_length( wc_clean( wp_unslash( $_REQUEST['_length'] ) ) ); // WPCS: input var ok, sanitization ok.
|
|
}
|
|
if ( isset( $_REQUEST['_width'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
$product->set_width( wc_clean( wp_unslash( $_REQUEST['_width'] ) ) ); // WPCS: input var ok, sanitization ok.
|
|
}
|
|
if ( isset( $_REQUEST['_height'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
$product->set_height( wc_clean( wp_unslash( $_REQUEST['_height'] ) ) ); // WPCS: input var ok, sanitization ok.
|
|
}
|
|
}
|
|
|
|
if ( ! empty( $_REQUEST['_tax_status'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
$product->set_tax_status( wc_clean( $_REQUEST['_tax_status'] ) ); // WPCS: input var ok, sanitization ok.
|
|
}
|
|
|
|
if ( ! empty( $_REQUEST['_tax_class'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
$tax_class = wc_clean( wp_unslash( $_REQUEST['_tax_class'] ) ); // WPCS: input var ok, sanitization ok.
|
|
if ( 'standard' === $tax_class ) {
|
|
$tax_class = '';
|
|
}
|
|
$product->set_tax_class( $tax_class );
|
|
}
|
|
|
|
if ( ! empty( $_REQUEST['_shipping_class'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
if ( '_no_shipping_class' === $_REQUEST['_shipping_class'] ) { // WPCS: input var ok, sanitization ok.
|
|
$product->set_shipping_class_id( 0 );
|
|
} else {
|
|
$shipping_class_id = $data_store->get_shipping_class_id_by_slug( wc_clean( $_REQUEST['_shipping_class'] ) ); // WPCS: input var ok, sanitization ok.
|
|
$product->set_shipping_class_id( $shipping_class_id );
|
|
}
|
|
}
|
|
|
|
if ( ! empty( $_REQUEST['_visibility'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
$product->set_catalog_visibility( wc_clean( $_REQUEST['_visibility'] ) ); // WPCS: input var ok, sanitization ok.
|
|
}
|
|
|
|
if ( ! empty( $_REQUEST['_featured'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
$product->set_featured( wp_unslash( $_REQUEST['_featured'] ) ); // WPCS: input var ok, sanitization ok.
|
|
}
|
|
|
|
if ( ! empty( $_REQUEST['_sold_individually'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
if ( 'yes' === $_REQUEST['_sold_individually'] ) { // WPCS: input var ok, sanitization ok.
|
|
$product->set_sold_individually( 'yes' );
|
|
} else {
|
|
$product->set_sold_individually( '' );
|
|
}
|
|
}
|
|
|
|
// Handle price - remove dates and set to lowest.
|
|
$change_price_product_types = apply_filters( 'woocommerce_bulk_edit_save_price_product_types', array( 'simple', 'external' ) );
|
|
$can_product_type_change_price = false;
|
|
foreach ( $change_price_product_types as $product_type ) {
|
|
if ( $product->is_type( $product_type ) ) {
|
|
$can_product_type_change_price = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ( $can_product_type_change_price ) {
|
|
$price_changed = false;
|
|
|
|
if ( ! empty( $_REQUEST['change_regular_price'] ) && isset( $_REQUEST['_regular_price'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
$change_regular_price = absint( $_REQUEST['change_regular_price'] ); // WPCS: input var ok, sanitization ok.
|
|
$regular_price = wc_clean( wp_unslash( $_REQUEST['_regular_price'] ) ); // WPCS: input var ok, sanitization ok.
|
|
|
|
switch ( $change_regular_price ) {
|
|
case 1:
|
|
$new_price = $regular_price;
|
|
break;
|
|
case 2:
|
|
if ( strstr( $regular_price, '%' ) ) {
|
|
$percent = str_replace( '%', '', $regular_price ) / 100;
|
|
$new_price = $old_regular_price + ( round( $old_regular_price * $percent, wc_get_price_decimals() ) );
|
|
} else {
|
|
$new_price = $old_regular_price + $regular_price;
|
|
}
|
|
break;
|
|
case 3:
|
|
if ( strstr( $regular_price, '%' ) ) {
|
|
$percent = str_replace( '%', '', $regular_price ) / 100;
|
|
$new_price = max( 0, $old_regular_price - ( round( $old_regular_price * $percent, wc_get_price_decimals() ) ) );
|
|
} else {
|
|
$new_price = max( 0, $old_regular_price - $regular_price );
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if ( isset( $new_price ) && $new_price !== $old_regular_price ) {
|
|
$price_changed = true;
|
|
$new_price = round( $new_price, wc_get_price_decimals() );
|
|
$product->set_regular_price( $new_price );
|
|
}
|
|
}
|
|
|
|
if ( ! empty( $_REQUEST['change_sale_price'] ) && isset( $_REQUEST['_sale_price'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
$change_sale_price = absint( $_REQUEST['change_sale_price'] ); // WPCS: input var ok, sanitization ok.
|
|
$sale_price = esc_attr( wp_unslash( $_REQUEST['_sale_price'] ) ); // WPCS: input var ok, sanitization ok.
|
|
|
|
switch ( $change_sale_price ) {
|
|
case 1:
|
|
$new_price = $sale_price;
|
|
break;
|
|
case 2:
|
|
if ( strstr( $sale_price, '%' ) ) {
|
|
$percent = str_replace( '%', '', $sale_price ) / 100;
|
|
$new_price = $old_sale_price + ( $old_sale_price * $percent );
|
|
} else {
|
|
$new_price = $old_sale_price + $sale_price;
|
|
}
|
|
break;
|
|
case 3:
|
|
if ( strstr( $sale_price, '%' ) ) {
|
|
$percent = str_replace( '%', '', $sale_price ) / 100;
|
|
$new_price = max( 0, $old_sale_price - ( $old_sale_price * $percent ) );
|
|
} else {
|
|
$new_price = max( 0, $old_sale_price - $sale_price );
|
|
}
|
|
break;
|
|
case 4:
|
|
if ( strstr( $sale_price, '%' ) ) {
|
|
$percent = str_replace( '%', '', $sale_price ) / 100;
|
|
$new_price = max( 0, $product->regular_price - ( $product->regular_price * $percent ) );
|
|
} else {
|
|
$new_price = max( 0, $product->regular_price - $sale_price );
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if ( isset( $new_price ) && $new_price !== $old_sale_price ) {
|
|
$price_changed = true;
|
|
$new_price = ! empty( $new_price ) || '0' === $new_price ? round( $new_price, wc_get_price_decimals() ) : '';
|
|
$product->set_sale_price( $new_price );
|
|
}
|
|
}
|
|
|
|
if ( $price_changed ) {
|
|
$product->set_date_on_sale_to( '' );
|
|
$product->set_date_on_sale_from( '' );
|
|
|
|
if ( $product->get_regular_price() < $product->get_sale_price() ) {
|
|
$product->set_sale_price( '' );
|
|
}
|
|
}
|
|
}
|
|
|
|
// Handle Stock Data.
|
|
$was_managing_stock = $product->get_manage_stock() ? 'yes' : 'no';
|
|
$stock_status = $product->get_stock_status();
|
|
$backorders = $product->get_backorders();
|
|
$backorders = ! empty( $_REQUEST['_backorders'] ) ? wc_clean( $_REQUEST['_backorders'] ) : $backorders; // WPCS: input var ok, sanitization ok.
|
|
$stock_status = ! empty( $_REQUEST['_stock_status'] ) ? wc_clean( $_REQUEST['_stock_status'] ) : $stock_status; // WPCS: input var ok, sanitization ok.
|
|
|
|
if ( ! empty( $_REQUEST['_manage_stock'] ) ) { // WPCS: input var ok, sanitization ok.
|
|
$manage_stock = 'yes' === wc_clean( $_REQUEST['_manage_stock'] ) && 'grouped' !== $product->get_type() ? 'yes' : 'no'; // WPCS: input var ok, sanitization ok.
|
|
} else {
|
|
$manage_stock = $was_managing_stock;
|
|
}
|
|
|
|
$stock_amount = 'yes' === $manage_stock && ! empty( $_REQUEST['change_stock'] ) && isset( $_REQUEST['_stock'] ) ? wc_stock_amount( $_REQUEST['_stock'] ) : $product->get_stock_quantity(); // WPCS: input var ok, sanitization ok.
|
|
|
|
$product->set_manage_stock( $manage_stock );
|
|
$product->set_backorders( $backorders );
|
|
|
|
if ( 'yes' === get_option( 'woocommerce_manage_stock' ) ) {
|
|
$product->set_stock_quantity( $stock_amount );
|
|
}
|
|
|
|
// Apply product type constraints to stock status.
|
|
if ( $product->is_type( 'external' ) ) {
|
|
// External products are always in stock.
|
|
$product->set_stock_status( 'instock' );
|
|
} elseif ( $product->is_type( 'variable' ) && ! $product->get_manage_stock() ) {
|
|
// Stock status is determined by children.
|
|
foreach ( $product->get_children() as $child_id ) {
|
|
$child = wc_get_product( $child_id );
|
|
if ( ! $product->get_manage_stock() ) {
|
|
$child->set_stock_status( $stock_status );
|
|
$child->save();
|
|
}
|
|
}
|
|
$product = WC_Product_Variable::sync( $product, false );
|
|
} else {
|
|
$product->set_stock_status( $stock_status );
|
|
}
|
|
|
|
$product->save();
|
|
|
|
do_action( 'woocommerce_product_bulk_edit_save', $product );
|
|
}
|
|
|
|
/**
|
|
* Disable the auto-save functionality for Orders.
|
|
*/
|
|
public function disable_autosave() {
|
|
global $post;
|
|
|
|
if ( $post && in_array( get_post_type( $post->ID ), wc_get_order_types( 'order-meta-boxes' ), true ) ) {
|
|
wp_dequeue_script( 'autosave' );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Output extra data on post forms.
|
|
*
|
|
* @param WP_Post $post Current post object.
|
|
*/
|
|
public function edit_form_top( $post ) {
|
|
echo '<input type="hidden" id="original_post_title" name="original_post_title" value="' . esc_attr( $post->post_title ) . '" />';
|
|
}
|
|
|
|
/**
|
|
* Change title boxes in admin.
|
|
*
|
|
* @param string $text Text to shown.
|
|
* @param WP_Post $post Current post object.
|
|
* @return string
|
|
*/
|
|
public function enter_title_here( $text, $post ) {
|
|
switch ( $post->post_type ) {
|
|
case 'product':
|
|
$text = esc_html__( 'Product name', 'woocommerce' );
|
|
break;
|
|
case 'shop_coupon':
|
|
$text = esc_html__( 'Coupon code', 'woocommerce' );
|
|
break;
|
|
}
|
|
return $text;
|
|
}
|
|
|
|
/**
|
|
* Print coupon description textarea field.
|
|
*
|
|
* @param WP_Post $post Current post object.
|
|
*/
|
|
public function edit_form_after_title( $post ) {
|
|
if ( 'shop_coupon' === $post->post_type ) {
|
|
?>
|
|
<textarea id="woocommerce-coupon-description" name="excerpt" cols="5" rows="2" placeholder="<?php esc_attr_e( 'Description (optional)', 'woocommerce' ); ?>"><?php echo $post->post_excerpt; // WPCS: XSS ok. ?></textarea>
|
|
<?php
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Hidden default Meta-Boxes.
|
|
*
|
|
* @param array $hidden Hidden boxes.
|
|
* @param object $screen Current screen.
|
|
* @return array
|
|
*/
|
|
public function hidden_meta_boxes( $hidden, $screen ) {
|
|
if ( 'product' === $screen->post_type && 'post' === $screen->base ) {
|
|
$hidden = array_merge( $hidden, array( 'postcustom' ) );
|
|
}
|
|
|
|
return $hidden;
|
|
}
|
|
|
|
/**
|
|
* Output product visibility options.
|
|
*/
|
|
public function product_data_visibility() {
|
|
global $post, $thepostid, $product_object;
|
|
|
|
if ( 'product' !== $post->post_type ) {
|
|
return;
|
|
}
|
|
|
|
$thepostid = $post->ID;
|
|
$product_object = $thepostid ? wc_get_product( $thepostid ) : new WC_Product();
|
|
$current_visibility = $product_object->get_catalog_visibility();
|
|
$current_featured = wc_bool_to_string( $product_object->get_featured() );
|
|
$visibility_options = wc_get_product_visibility_options();
|
|
?>
|
|
<div class="misc-pub-section" id="catalog-visibility">
|
|
<?php esc_html_e( 'Catalog visibility:', 'woocommerce' ); ?> <strong id="catalog-visibility-display"><?php
|
|
|
|
echo isset( $visibility_options[ $current_visibility ] ) ? esc_html( $visibility_options[ $current_visibility ] ) : esc_html( $current_visibility );
|
|
|
|
if ( 'yes' === $current_featured ) {
|
|
echo ', ' . esc_html__( 'Featured', 'woocommerce' );
|
|
}
|
|
?></strong>
|
|
|
|
<a href="#catalog-visibility" class="edit-catalog-visibility hide-if-no-js"><?php esc_html_e( 'Edit', 'woocommerce' ); ?></a>
|
|
|
|
<div id="catalog-visibility-select" class="hide-if-js">
|
|
|
|
<input type="hidden" name="current_visibility" id="current_visibility" value="<?php echo esc_attr( $current_visibility ); ?>" />
|
|
<input type="hidden" name="current_featured" id="current_featured" value="<?php echo esc_attr( $current_featured ); ?>" />
|
|
|
|
<?php
|
|
echo '<p>' . esc_html__( 'This setting determines which shop pages products will be listed on.', 'woocommerce' ) . '</p>';
|
|
|
|
foreach ( $visibility_options as $name => $label ) {
|
|
echo '<input type="radio" name="_visibility" id="_visibility_' . esc_attr( $name ) . '" value="' . esc_attr( $name ) . '" ' . checked( $current_visibility, $name, false ) . ' data-label="' . esc_attr( $label ) . '" /> <label for="_visibility_' . esc_attr( $name ) . '" class="selectit">' . esc_html( $label ) . '</label><br />';
|
|
}
|
|
|
|
echo '<br /><input type="checkbox" name="_featured" id="_featured" ' . checked( $current_featured, 'yes', false ) . ' /> <label for="_featured">' . esc_html_e( 'This is a featured product', 'woocommerce' ) . '</label><br />';
|
|
?>
|
|
<p>
|
|
<a href="#catalog-visibility" class="save-post-visibility hide-if-no-js button"><?php esc_html_e( 'OK', 'woocommerce' ); ?></a>
|
|
<a href="#catalog-visibility" class="cancel-post-visibility hide-if-no-js"><?php esc_html_e( 'Cancel', 'woocommerce' ); ?></a>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
<?php
|
|
}
|
|
|
|
/**
|
|
* Change upload dir for downloadable files.
|
|
*
|
|
* @param array $pathdata Array of paths.
|
|
* @return array
|
|
*/
|
|
public function upload_dir( $pathdata ) {
|
|
if ( isset( $_POST['type'] ) && 'downloadable_product' === $_POST['type'] ) { // WPCS: CSRF ok, input var ok.
|
|
|
|
if ( empty( $pathdata['subdir'] ) ) {
|
|
$pathdata['path'] = $pathdata['path'] . '/woocommerce_uploads';
|
|
$pathdata['url'] = $pathdata['url'] . '/woocommerce_uploads';
|
|
$pathdata['subdir'] = '/woocommerce_uploads';
|
|
} else {
|
|
$new_subdir = '/woocommerce_uploads' . $uploads['subdir'];
|
|
|
|
$uploads['path'] = str_replace( $uploads['subdir'], $new_subdir, $uploads['path'] );
|
|
$uploads['url'] = str_replace( $uploads['subdir'], $new_subdir, $uploads['url'] );
|
|
$uploads['subdir'] = str_replace( $uploads['subdir'], $new_subdir, $uploads['subdir'] );
|
|
}
|
|
}
|
|
return $pathdata;
|
|
}
|
|
|
|
/**
|
|
* Run a filter when uploading a downloadable product.
|
|
*/
|
|
public function woocommerce_media_upload_downloadable_product() {
|
|
do_action( 'media_upload_file' );
|
|
}
|
|
|
|
/**
|
|
* Grant downloadable file access to any newly added files on any existing.
|
|
* orders for this product that have previously been granted downloadable file access.
|
|
*
|
|
* @param int $product_id product identifier.
|
|
* @param int $variation_id optional product variation identifier.
|
|
* @param array $downloadable_files newly set files.
|
|
* @deprecated and moved to post-data class.
|
|
*/
|
|
public function process_product_file_download_paths( $product_id, $variation_id, $downloadable_files ) {
|
|
WC_Post_Data::process_product_file_download_paths( $product_id, $variation_id, $downloadable_files );
|
|
}
|
|
|
|
/**
|
|
* When editing the shop page, we should hide templates.
|
|
*
|
|
* @param array $page_templates Templates array.
|
|
* @param string $class Classname.
|
|
* @param WP_Post $post The current post object.
|
|
* @return array
|
|
*/
|
|
public function hide_cpt_archive_templates( $page_templates, $theme, $post ) {
|
|
$shop_page_id = wc_get_page_id( 'shop' );
|
|
|
|
if ( $post && absint( $shop_page_id ) === absint( $post->ID ) ) {
|
|
$page_templates = array();
|
|
}
|
|
|
|
return $page_templates;
|
|
}
|
|
|
|
/**
|
|
* Show a notice above the CPT archive.
|
|
*
|
|
* @param WP_Post $post The current post object.
|
|
*/
|
|
public function show_cpt_archive_notice( $post ) {
|
|
$shop_page_id = wc_get_page_id( 'shop' );
|
|
|
|
if ( $post && absint( $shop_page_id ) === absint( $post->ID ) ) {
|
|
echo '<div class="notice notice-info">';
|
|
echo '<p>' . sprintf( wp_kses_post( __( 'This is the WooCommerce shop page. The shop page is a special archive that lists your products. <a href="%s">You can read more about this here</a>.', 'woocommerce' ), 'https://docs.woocommerce.com/document/woocommerce-pages/#section-4' ) ) . '</p>';
|
|
echo '</div>';
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Add a post display state for special WC pages in the page list table.
|
|
*
|
|
* @param array $post_states An array of post display states.
|
|
* @param WP_Post $post The current post object.
|
|
*/
|
|
public function add_display_post_states( $post_states, $post ) {
|
|
if ( wc_get_page_id( 'shop' ) === $post->ID ) {
|
|
$post_states['wc_page_for_shop'] = __( 'Shop Page', 'woocommerce' );
|
|
}
|
|
|
|
if ( wc_get_page_id( 'cart' ) === $post->ID ) {
|
|
$post_states['wc_page_for_cart'] = __( 'Cart Page', 'woocommerce' );
|
|
}
|
|
|
|
if ( wc_get_page_id( 'checkout' ) === $post->ID ) {
|
|
$post_states['wc_page_for_checkout'] = __( 'Checkout Page', 'woocommerce' );
|
|
}
|
|
|
|
if ( wc_get_page_id( 'myaccount' ) === $post->ID ) {
|
|
$post_states['wc_page_for_myaccount'] = __( 'My Account Page', 'woocommerce' );
|
|
}
|
|
|
|
if ( wc_get_page_id( 'terms' ) === $post->ID ) {
|
|
$post_states['wc_page_for_terms'] = __( 'Terms and Conditions Page', 'woocommerce' );
|
|
}
|
|
|
|
return $post_states;
|
|
}
|
|
}
|
|
|
|
new WC_Admin_Post_Types();
|