woocommerce/classes/product.class.php

560 lines
15 KiB
PHP

<?php
/**
* Product Class
*
* The WooCommerce product class handles individual product data.
*
* @class woocommerce_product
* @package WooCommerce
* @category Class
* @author WooThemes
*/
class woocommerce_product {
var $id;
var $exists;
var $data;
var $sku;
var $attributes;
var $post;
var $stock;
var $children;
var $visibility;
var $product_type;
var $price;
var $sale_price_dates_to;
var $sale_price_dates_from;
/**
* Loads all product data from custom fields
*
* @param int $id ID of the product to load
*/
function woocommerce_product( $id ) {
$this->id = $id;
$product_custom_fields = get_post_custom( $this->id );
if (isset($product_custom_fields['SKU'][0]) && !empty($product_custom_fields['SKU'][0])) $this->sku = $product_custom_fields['SKU'][0]; else $this->sku = $this->id;
if (isset($product_custom_fields['product_data'][0])) $this->data = maybe_unserialize( $product_custom_fields['product_data'][0] ); else $this->data = '';
if (isset($product_custom_fields['product_attributes'][0])) $this->attributes = maybe_unserialize( $product_custom_fields['product_attributes'][0] ); else $this->attributes = array();
if (isset($product_custom_fields['price'][0])) $this->price = $product_custom_fields['price'][0]; else $this->price = 0;
if (isset($product_custom_fields['visibility'][0])) $this->visibility = $product_custom_fields['visibility'][0]; else $this->visibility = 'hidden';
if (isset($product_custom_fields['stock'][0])) $this->stock = $product_custom_fields['stock'][0]; else $this->stock = 0;
// Again just in case, to fix WP bug
$this->data = maybe_unserialize( $this->data );
$this->attributes = maybe_unserialize( $this->attributes );
$terms = wp_get_object_terms( $id, 'product_type' );
if (!is_wp_error($terms) && $terms) :
$term = current($terms);
$this->product_type = $term->slug;
else :
$this->product_type = 'simple';
endif;
$this->get_children();
if ($this->data) :
$this->exists = true;
else :
$this->exists = false;
endif;
}
/** Returns the product's children */
function get_children() {
if (!is_array($this->children)) :
$this->children = array();
if ($this->is_type('variable')) $child_post_type = 'product_variation'; else $child_post_type = 'product';
if ( $children_products =& get_children( 'post_parent='.$this->id.'&post_type='.$child_post_type.'&orderby=menu_order&order=ASC' ) ) :
if ($children_products) foreach ($children_products as $child) :
if ($this->is_type('variable')) :
$child->product = &new woocommerce_product_variation( $child->ID );
else :
$child->product = &new woocommerce_product( $child->ID );
endif;
endforeach;
$this->children = (array) $children_products;
endif;
endif;
return $this->children;
}
/**
* Reduce stock level of the product
*
* @param int $by Amount to reduce by
*/
function reduce_stock( $by = 1 ) {
if ($this->managing_stock()) :
$reduce_to = $this->stock - $by;
update_post_meta($this->id, 'stock', $reduce_to);
return $reduce_to;
endif;
}
/**
* Increase stock level of the product
*
* @param int $by Amount to increase by
*/
function increase_stock( $by = 1 ) {
if ($this->managing_stock()) :
$increase_to = $this->stock + $by;
update_post_meta($this->id, 'stock', $increase_to);
return $increase_to;
endif;
}
/**
* Checks the product type
*
* @param string $type Type to check against
*/
function is_type( $type ) {
if (is_array($type) && in_array($this->product_type, $type)) return true;
elseif ($this->product_type==$type) return true;
return false;
}
/** Returns whether or not the product has any child product */
function has_child () {
return sizeof($this->children) ? true : false;
}
/** Returns whether or not the product post exists */
function exists() {
if ($this->exists) return true;
return false;
}
/** Returns whether or not the product is taxable */
function is_taxable() {
if (isset($this->data['tax_status']) && $this->data['tax_status']=='taxable') return true;
return false;
}
/** Returns whether or not the product shipping is taxable */
function is_shipping_taxable() {
if (isset($this->data['tax_status']) && ($this->data['tax_status']=='taxable' || $this->data['tax_status']=='shipping')) return true;
return false;
}
/** Get the product's post data */
function get_post_data() {
if (empty($this->post)) :
$this->post = get_post( $this->id );
endif;
return $this->post;
}
/** Get the title of the post */
function get_title () {
$this->get_post_data();
return apply_filters('woocommerce_product_title', $this->post->post_title, $this);
}
/** Get the add to url */
function add_to_cart_url() {
if ($this->is_type('variable')) :
$url = add_query_arg('add-to-cart', 'variation');
$url = add_query_arg('product', $this->id, $url);
elseif ( $this->has_child() ) :
$url = add_query_arg('add-to-cart', 'group');
$url = add_query_arg('product', $this->id, $url);
else :
$url = add_query_arg('add-to-cart', $this->id);
endif;
$url = woocommerce::nonce_url( 'add_to_cart', $url );
return $url;
}
/** Returns whether or not the product is stock managed */
function managing_stock() {
if (get_option('woocommerce_manage_stock')=='yes') :
if (isset($this->data['manage_stock']) && $this->data['manage_stock']=='yes') return true;
endif;
return false;
}
/** Returns whether or not the product is in stock */
function is_in_stock() {
if ($this->managing_stock()) :
if (!$this->backorders_allowed()) :
if ($this->stock==0 || $this->stock<0) :
return false;
else :
if ($this->data['stock_status']=='instock') return true;
return false;
endif;
else :
if ($this->data['stock_status']=='instock') return true;
return false;
endif;
endif;
return true;
}
/** Returns whether or not the product can be backordered */
function backorders_allowed() {
if ($this->data['backorders']=='yes' || $this->data['backorders']=='notify') return true;
return false;
}
/** Returns whether or not the product needs to notify the customer on backorder */
function backorders_require_notification() {
if ($this->data['backorders']=='notify') return true;
return false;
}
/** Returns whether or not the product has enough stock for the order */
function has_enough_stock( $quantity ) {
if ($this->backorders_allowed()) return true;
if ($this->stock >= $quantity) :
return true;
endif;
return false;
}
/** Returns the availability of the product */
function get_availability() {
$availability = "";
$class = "";
if (!$this->managing_stock()) :
if ($this->is_in_stock()) :
//$availability = __('In stock', 'woothemes'); /* Lets not bother showing stock if its not managed and is available */
else :
$availability = __('Out of stock', 'woothemes');
$class = 'out-of-stock';
endif;
else :
if ($this->is_in_stock()) :
if ($this->stock > 0) :
$availability = __('In stock', 'woothemes');
if ($this->backorders_allowed()) :
if ($this->backorders_require_notification()) :
$availability .= ' &ndash; '.$this->stock.' ';
$availability .= __('available', 'woothemes');
$availability .= __(' (backorders allowed)', 'woothemes');
endif;
else :
$availability .= ' &ndash; '.$this->stock.' ';
$availability .= __('available', 'woothemes');
endif;
else :
if ($this->backorders_allowed()) :
if ($this->backorders_require_notification()) :
$availability = __('Available on backorder', 'woothemes');
else :
$availability = __('In stock', 'woothemes');
endif;
else :
$availability = __('Out of stock', 'woothemes');
$class = 'out-of-stock';
endif;
endif;
else :
if ($this->backorders_allowed()) :
$availability = __('Available on backorder', 'woothemes');
else :
$availability = __('Out of stock', 'woothemes');
$class = 'out-of-stock';
endif;
endif;
endif;
return array( 'availability' => $availability, 'class' => $class);
}
/** Returns whether or not the product is featured */
function is_featured() {
if (get_post_meta($this->id, 'featured', true)=='yes') return true;
return false;
}
/** Returns whether or not the product is visible */
function is_visible() {
if ($this->visibility=='hidden') return false;
if ($this->visibility=='visible') return true;
if ($this->visibility=='search' && is_search()) return true;
if ($this->visibility=='search' && !is_search()) return false;
if ($this->visibility=='catalog' && is_search()) return false;
if ($this->visibility=='catalog' && !is_search()) return true;
}
/** Returns whether or not the product is on sale */
function is_on_sale() {
if ( $this->has_child() ) :
$onsale = false;
foreach ($this->children as $child) :
if ( isset($child->product->data['sale_price']) && $child->product->data['sale_price']==$child->product->price ) :
return true;
endif;
endforeach;
else :
if ( isset($this->data['sale_price']) && $this->data['sale_price']==$this->price ) :
return true;
endif;
endif;
return false;
}
/** Returns the product's weight */
function get_weight() {
if ($this->data['weight']) return $this->data['weight'];
}
/** Returns the product's price */
function get_price() {
return $this->price;
/*if (!$price) $price = $this->price;
if (get_option('woocommerce_prices_include_tax')=='yes' && $this->is_taxable() && woocommerce_customer::is_customer_outside_base()) :
$_tax = &new woocommerce_tax();
$price = $price * 100;
$base_rate = $_tax->get_shop_base_rate( $this->data['tax_class'] );
$rate = $_tax->get_rate( $this->data['tax_class'] );
$base_tax_amount = round($_tax->calc_tax( $price, $base_rate, true ));
$tax_amount = round($_tax->calc_tax( ($price-$base_tax_amount), $rate, false ));
return ($price - $base_tax_amount + $tax_amount) / 100;
endif;
return $price;*/
}
/** Returns the price (excluding tax) */
function get_price_excluding_tax() {
$price = $this->price;
if (get_option('woocommerce_prices_include_tax')=='yes') :
if ( $rate = $this->get_tax_base_rate() ) :
if ( $rate>0 ) :
$_tax = &new woocommerce_tax();
$tax_amount = $_tax->calc_tax( $price, $rate, true );
$price = $price - $tax_amount;
endif;
endif;
endif;
return $price;
}
/** Returns the base tax rate */
function get_tax_base_rate() {
if ( $this->is_taxable() && get_option('woocommerce_calc_taxes')=='yes') :
$_tax = &new woocommerce_tax();
$rate = $_tax->get_shop_base_rate( $this->data['tax_class'] );
return $rate;
endif;
}
/** Returns the price in html format */
function get_price_html() {
$price = '';
if ( $this->has_child() ) :
$min_price = '';
$max_price = '';
foreach ($this->children as $child) :
$child_price = $child->product->get_price();
if ($child_price<$min_price || $min_price == '') $min_price = $child_price;
if ($child_price>$max_price || $max_price == '') $max_price = $child_price;
endforeach;
$price .= '<span class="from">' . __('From: ', 'woothemes') . '</span>' . woocommerce_price($min_price);
elseif ($this->is_type('variable')) :
$price .= '<span class="from">' . __('From: ', 'woothemes') . '</span>' . woocommerce_price($this->get_price());
else :
if ($this->price) :
if ($this->is_on_sale() && isset($this->data['regular_price'])) :
$price .= '<del>'.woocommerce_price( $this->data['regular_price'] ).'</del> <ins>'.woocommerce_price($this->get_price()).'</ins>';
else :
$price .= woocommerce_price($this->get_price());
endif;
endif;
endif;
return $price;
}
/** Returns the upsell product ids */
function get_upsells() {
if (isset($this->data['upsell_ids'])) return (array) $this->data['upsell_ids']; else return array();
}
/** Returns the crosssell product ids */
function get_cross_sells() {
if (isset($this->data['crosssell_ids'])) return (array) $this->data['crosssell_ids']; else return array();
}
/** Returns the product categories */
function get_categories( $sep = ', ', $before = '', $after = '' ) {
return get_the_term_list($this->id, 'product_cat', $before, $sep, $after);
}
/** Returns the product tags */
function get_tags( $sep = ', ', $before = '', $after = '' ) {
return get_the_term_list($this->id, 'product_tag', $before, $sep, $after);
}
/** Get and return related products */
function get_related( $limit = 5 ) {
global $wpdb, $all_post_ids;
// Related products are found from category and tag
$tags_array = array(0);
$cats_array = array(0);
$tags = '';
$cats = '';
// Get tags
$terms = wp_get_post_terms($this->id, 'product_tag');
foreach ($terms as $term) {
$tags_array[] = $term->term_id;
}
$tags = implode(',', $tags_array);
$terms = wp_get_post_terms($this->id, 'product_cat');
foreach ($terms as $term) {
$cats_array[] = $term->term_id;
}
$cats = implode(',', $cats_array);
$q = "
SELECT p.ID
FROM $wpdb->term_taxonomy AS tt, $wpdb->term_relationships AS tr, $wpdb->posts AS p, $wpdb->postmeta AS pm
WHERE
p.ID != $this->id
AND p.post_status = 'publish'
AND p.post_date_gmt < NOW()
AND p.post_type = 'product'
AND pm.meta_key = 'visibility'
AND pm.meta_value IN ('visible', 'catalog')
AND pm.post_id = p.ID
AND
(
(
tt.taxonomy ='product_cat'
AND tt.term_taxonomy_id = tr.term_taxonomy_id
AND tr.object_id = p.ID
AND tt.term_id IN ($cats)
)
OR
(
tt.taxonomy ='product_tag'
AND tt.term_taxonomy_id = tr.term_taxonomy_id
AND tr.object_id = p.ID
AND tt.term_id IN ($tags)
)
)
GROUP BY tr.object_id
ORDER BY RAND()
LIMIT $limit;";
$related = $wpdb->get_col($q);
return $related;
}
/** Returns product attributes */
function get_attributes() {
return $this->attributes;
}
/** Returns whether or not the product has any attributes set */
function has_attributes() {
if (isset($this->attributes) && sizeof($this->attributes)>0) :
foreach ($this->attributes as $attribute) :
if ($attribute['visible'] == 'yes') return true;
endforeach;
endif;
return false;
}
/** Lists a table of attributes for the product page */
function list_attributes() {
$attributes = $this->get_attributes();
if ($attributes && sizeof($attributes)>0) :
echo '<table cellspacing="0" class="shop_attributes">';
$alt = 1;
foreach ($attributes as $attribute) :
if ($attribute['visible'] == 'no') continue;
$alt = $alt*-1;
echo '<tr class="';
if ($alt==1) echo 'alt';
echo '"><th>'.wptexturize($attribute['name']).'</th><td>';
if (is_array($attribute['value'])) $attribute['value'] = implode(', ', $attribute['value']);
echo wpautop(wptexturize($attribute['value']));
echo '</td></tr>';
endforeach;
echo '</table>';
endif;
}
}