2013-08-06 15:56:15 +00:00
< ? php
/**
* Attributes Page
*
* The attributes section lets users add custom attributes to assign to products - they can also be used in the layered nav widget .
*
* @ author WooThemes
* @ category Admin
* @ package WooCommerce / Admin
* @ version 2.1 . 0
*/
if ( ! defined ( 'ABSPATH' ) ) exit ; // Exit if accessed directly
if ( ! class_exists ( 'WC_Admin_Attributes' ) ) :
/**
* WC_Admin_Attributes Class
*/
class WC_Admin_Attributes {
/**
* Handles output of the attributes page in admin .
*
* Shows the created attributes and lets you add new ones or edit existing ones .
* The added attributes are stored in the database and can be used for layered navigation .
*/
public function output () {
global $wpdb , $woocommerce ;
// Action to perform: add, edit, delete or none
$action = '' ;
if ( ! empty ( $_POST [ 'add_new_attribute' ] ) ) {
$action = 'add' ;
} elseif ( ! empty ( $_POST [ 'save_attribute' ] ) && ! empty ( $_GET [ 'edit' ] ) ) {
$action = 'edit' ;
} elseif ( ! empty ( $_GET [ 'delete' ] ) ) {
$action = 'delete' ;
}
// Add or edit an attribute
if ( 'add' === $action || 'edit' === $action ) {
// Security check
if ( 'add' === $action ) {
check_admin_referer ( 'woocommerce-add-new_attribute' );
}
if ( 'edit' === $action ) {
$attribute_id = absint ( $_GET [ 'edit' ] );
check_admin_referer ( 'woocommerce-save-attribute_' . $attribute_id );
}
// Grab the submitted data
$attribute_label = ( isset ( $_POST [ 'attribute_label' ] ) ) ? ( string ) stripslashes ( $_POST [ 'attribute_label' ] ) : '' ;
$attribute_name = ( isset ( $_POST [ 'attribute_name' ] ) ) ? woocommerce_sanitize_taxonomy_name ( stripslashes ( ( string ) $_POST [ 'attribute_name' ] ) ) : '' ;
$attribute_type = ( isset ( $_POST [ 'attribute_type' ] ) ) ? ( string ) stripslashes ( $_POST [ 'attribute_type' ] ) : '' ;
$attribute_orderby = ( isset ( $_POST [ 'attribute_orderby' ] ) ) ? ( string ) stripslashes ( $_POST [ 'attribute_orderby' ] ) : '' ;
// Auto-generate the label or slug if only one of both was provided
if ( ! $attribute_label ) {
2013-10-25 10:53:53 +00:00
$attribute_label = ucfirst ( $attribute_name );
2013-10-04 14:10:49 +00:00
}
if ( ! $attribute_name ) {
2013-08-06 15:56:15 +00:00
$attribute_name = woocommerce_sanitize_taxonomy_name ( stripslashes ( $attribute_label ) );
}
// Forbidden attribute names
// http://codex.wordpress.org/Function_Reference/register_taxonomy#Reserved_Terms
$reserved_terms = array (
'attachment' , 'attachment_id' , 'author' , 'author_name' , 'calendar' , 'cat' , 'category' , 'category__and' ,
'category__in' , 'category__not_in' , 'category_name' , 'comments_per_page' , 'comments_popup' , 'cpage' , 'day' ,
'debug' , 'error' , 'exact' , 'feed' , 'hour' , 'link_category' , 'm' , 'minute' , 'monthnum' , 'more' , 'name' ,
'nav_menu' , 'nopaging' , 'offset' , 'order' , 'orderby' , 'p' , 'page' , 'page_id' , 'paged' , 'pagename' , 'pb' , 'perm' ,
'post' , 'post__in' , 'post__not_in' , 'post_format' , 'post_mime_type' , 'post_status' , 'post_tag' , 'post_type' ,
'posts' , 'posts_per_archive_page' , 'posts_per_page' , 'preview' , 'robots' , 's' , 'search' , 'second' , 'sentence' ,
'showposts' , 'static' , 'subpost' , 'subpost_id' , 'tag' , 'tag__and' , 'tag__in' , 'tag__not_in' , 'tag_id' ,
'tag_slug__and' , 'tag_slug__in' , 'taxonomy' , 'tb' , 'term' , 'type' , 'w' , 'withcomments' , 'withoutcomments' , 'year' ,
);
// Error checking
2013-09-24 09:06:37 +00:00
if ( ! $attribute_name || ! $attribute_label || ! $attribute_type ) {
2013-08-06 15:56:15 +00:00
$error = __ ( 'Please, provide an attribute name, slug and type.' , 'woocommerce' );
} elseif ( strlen ( $attribute_name ) >= 28 ) {
$error = sprintf ( __ ( 'Slug “%s” is too long (28 characters max). Shorten it, please.' , 'woocommerce' ), sanitize_title ( $attribute_name ) );
} elseif ( in_array ( $attribute_name , $reserved_terms ) ) {
$error = sprintf ( __ ( 'Slug “%s” is not allowed because it is a reserved term. Change it, please.' , 'woocommerce' ), sanitize_title ( $attribute_name ) );
} else {
2013-09-12 13:41:02 +00:00
$taxonomy_exists = taxonomy_exists ( wc_attribute_taxonomy_name ( $attribute_name ) );
2013-08-06 15:56:15 +00:00
if ( 'add' === $action && $taxonomy_exists ) {
$error = sprintf ( __ ( 'Slug “%s” is already in use. Change it, please.' , 'woocommerce' ), sanitize_title ( $attribute_name ) );
}
if ( 'edit' === $action ) {
$old_attribute_name = $wpdb -> get_var ( " SELECT attribute_name FROM { $wpdb -> prefix } woocommerce_attribute_taxonomies WHERE attribute_id = $attribute_id " );
if ( $old_attribute_name != $attribute_name && woocommerce_sanitize_taxonomy_name ( $old_attribute_name ) != $attribute_name && $taxonomy_exists ) {
$error = sprintf ( __ ( 'Slug “%s” is already in use. Change it, please.' , 'woocommerce' ), sanitize_title ( $attribute_name ) );
}
}
}
// Show the error message if any
if ( ! empty ( $error ) ) {
echo '<div id="woocommerce_errors" class="error fade"><p>' . $error . '</p></div>' ;
} else {
// Add new attribute
if ( 'add' === $action ) {
$attribute = array (
'attribute_label' => $attribute_label ,
'attribute_name' => $attribute_name ,
'attribute_type' => $attribute_type ,
'attribute_orderby' => $attribute_orderby ,
);
$wpdb -> insert ( $wpdb -> prefix . 'woocommerce_attribute_taxonomies' , $attribute );
do_action ( 'woocommerce_attribute_added' , $wpdb -> insert_id , $attribute );
$action_completed = true ;
}
// Edit existing attribute
if ( 'edit' === $action ) {
$attribute = array (
'attribute_label' => $attribute_label ,
'attribute_name' => $attribute_name ,
'attribute_type' => $attribute_type ,
'attribute_orderby' => $attribute_orderby ,
);
$wpdb -> update ( $wpdb -> prefix . 'woocommerce_attribute_taxonomies' , $attribute , array ( 'attribute_id' => $attribute_id ) );
do_action ( 'woocommerce_attribute_updated' , $attribute_id , $attribute , $old_attribute_name );
if ( $old_attribute_name != $attribute_name && ! empty ( $old_attribute_name ) ) {
// Update taxonomies in the wp term taxonomy table
$wpdb -> update (
$wpdb -> term_taxonomy ,
2013-09-12 13:41:02 +00:00
array ( 'taxonomy' => wc_attribute_taxonomy_name ( $attribute_name ) ),
2013-08-06 15:56:15 +00:00
array ( 'taxonomy' => 'pa_' . $old_attribute_name )
);
// Update taxonomy ordering term meta
$wpdb -> update (
$wpdb -> prefix . 'woocommerce_termmeta' ,
array ( 'meta_key' => 'order_pa_' . sanitize_title ( $attribute_name ) ),
array ( 'meta_key' => 'order_pa_' . sanitize_title ( $old_attribute_name ) )
);
// Update product attributes which use this taxonomy
$old_attribute_name_length = strlen ( $old_attribute_name ) + 3 ;
$attribute_name_length = strlen ( $attribute_name ) + 3 ;
$wpdb -> query ( "
UPDATE { $wpdb -> postmeta }
SET meta_value = REPLACE ( meta_value , 's:{$old_attribute_name_length}:\"pa_{$old_attribute_name}\"' , 's:{$attribute_name_length}:\"pa_{$attribute_name}\"' )
WHERE meta_key = '_product_attributes' "
);
// Update variations which use this taxonomy
$wpdb -> update (
$wpdb -> postmeta ,
array ( 'meta_key' => 'attribute_pa_' . sanitize_title ( $attribute_name ) ),
array ( 'meta_key' => 'attribute_pa_' . sanitize_title ( $old_attribute_name ) )
);
}
$action_completed = true ;
}
flush_rewrite_rules ();
}
}
// Delete an attribute
if ( 'delete' === $action ) {
// Security check
$attribute_id = absint ( $_GET [ 'delete' ] );
check_admin_referer ( 'woocommerce-delete-attribute_' . $attribute_id );
$attribute_name = $wpdb -> get_var ( " SELECT attribute_name FROM { $wpdb -> prefix } woocommerce_attribute_taxonomies WHERE attribute_id = $attribute_id " );
if ( $attribute_name && $wpdb -> query ( " DELETE FROM { $wpdb -> prefix } woocommerce_attribute_taxonomies WHERE attribute_id = $attribute_id " ) ) {
2013-09-12 13:41:02 +00:00
$taxonomy = wc_attribute_taxonomy_name ( $attribute_name );
2013-08-06 15:56:15 +00:00
if ( taxonomy_exists ( $taxonomy ) ) {
$terms = get_terms ( $taxonomy , 'orderby=name&hide_empty=0' );
foreach ( $terms as $term ) {
wp_delete_term ( $term -> term_id , $taxonomy );
}
}
do_action ( 'woocommerce_attribute_deleted' , $attribute_id , $attribute_name , $taxonomy );
$action_completed = true ;
}
}
2013-10-04 14:10:49 +00:00
// If an attribute was added, edited or deleted: clear cache
2013-08-06 15:56:15 +00:00
if ( ! empty ( $action_completed ) ) {
delete_transient ( 'wc_attribute_taxonomies' );
}
// Show admin interface
if ( ! empty ( $_GET [ 'edit' ] ) )
$this -> edit_attribute ();
else
$this -> add_attribute ();
}
/**
* Edit Attribute admin panel
*
* Shows the interface for changing an attributes type between select and text
*/
public function edit_attribute () {
global $wpdb ;
$edit = absint ( $_GET [ 'edit' ] );
$attribute_to_edit = $wpdb -> get_row ( " SELECT * FROM " . $wpdb -> prefix . " woocommerce_attribute_taxonomies WHERE attribute_id = ' $edit ' " );
$att_type = $attribute_to_edit -> attribute_type ;
$att_label = $attribute_to_edit -> attribute_label ;
$att_name = $attribute_to_edit -> attribute_name ;
$att_orderby = $attribute_to_edit -> attribute_orderby ;
?>
< div class = " wrap woocommerce " >
< div class = " icon32 icon32-attributes " id = " icon-woocommerce " >< br /></ div >
< h2 >< ? php _e ( 'Edit Attribute' , 'woocommerce' ) ?> </h2>
2013-10-22 15:46:59 +00:00
< form action = " admin.php?page=product_attributes&edit=<?php echo absint( $edit ); ?> " method = " post " >
2013-08-06 15:56:15 +00:00
< table class = " form-table " >
< tbody >
< tr class = " form-field form-required " >
< th scope = " row " valign = " top " >
< label for = " attribute_label " >< ? php _e ( 'Name' , 'woocommerce' ); ?> </label>
</ th >
< td >
< input name = " attribute_label " id = " attribute_label " type = " text " value = " <?php echo esc_attr( $att_label ); ?> " />
< p class = " description " >< ? php _e ( 'Name for the attribute (shown on the front-end).' , 'woocommerce' ); ?> </p>
</ td >
</ tr >
< tr class = " form-field form-required " >
< th scope = " row " valign = " top " >
< label for = " attribute_name " >< ? php _e ( 'Slug' , 'woocommerce' ); ?> </label>
</ th >
< td >
< input name = " attribute_name " id = " attribute_name " type = " text " value = " <?php echo esc_attr( $att_name ); ?> " maxlength = " 28 " />
< p class = " description " >< ? php _e ( 'Unique slug/reference for the attribute; must be shorter than 28 characters.' , 'woocommerce' ); ?> </p>
</ td >
</ tr >
< tr class = " form-field form-required " >
< th scope = " row " valign = " top " >
< label for = " attribute_type " >< ? php _e ( 'Type' , 'woocommerce' ); ?> </label>
</ th >
< td >
< select name = " attribute_type " id = " attribute_type " >
< option value = " select " < ? php selected ( $att_type , 'select' ); ?> ><?php _e( 'Select', 'woocommerce' ) ?></option>
< option value = " text " < ? php selected ( $att_type , 'text' ); ?> ><?php _e( 'Text', 'woocommerce' ) ?></option>
< ? php do_action ( 'woocommerce_admin_attribute_types' ); ?>
</ select >
2013-10-31 06:14:35 +00:00
< p class = " description " >< ? php _e ( 'Determines how you select attributes for products. Under admin panel -> products -> product data -> attributes -> values, <strong>Text</strong> allows manual entry whereas <strong>select</strong> allows pre-configured terms in a drop-down list.' , 'woocommerce' ); ?> </p>
2013-08-06 15:56:15 +00:00
</ td >
</ tr >
< tr class = " form-field form-required " >
< th scope = " row " valign = " top " >
< label for = " attribute_orderby " >< ? php _e ( 'Default sort order' , 'woocommerce' ); ?> </label>
</ th >
< td >
< select name = " attribute_orderby " id = " attribute_orderby " >
< option value = " menu_order " < ? php selected ( $att_orderby , 'menu_order' ); ?> ><?php _e( 'Custom ordering', 'woocommerce' ) ?></option>
< option value = " name " < ? php selected ( $att_orderby , 'name' ); ?> ><?php _e( 'Name', 'woocommerce' ) ?></option>
< option value = " id " < ? php selected ( $att_orderby , 'id' ); ?> ><?php _e( 'Term ID', 'woocommerce' ) ?></option>
</ select >
2013-10-31 06:14:35 +00:00
< p class = " description " >< ? php _e ( 'Determines the sort order of the terms on the frontend shop product pages. If using custom ordering, you can drag and drop the terms in this attribute.' , 'woocommerce' ); ?> </p>
2013-08-06 15:56:15 +00:00
</ td >
</ tr >
</ tbody >
</ table >
< p class = " submit " >< input type = " submit " name = " save_attribute " id = " submit " class = " button-primary " value = " <?php _e( 'Update', 'woocommerce' ); ?> " ></ p >
< ? php wp_nonce_field ( 'woocommerce-save-attribute_' . $edit ); ?>
</ form >
</ div >
< ? php
}
/**
* Add Attribute admin panel
*
* Shows the interface for adding new attributes
*/
public function add_attribute () {
global $woocommerce ;
?>
< div class = " wrap woocommerce " >
< div class = " icon32 icon32-attributes " id = " icon-woocommerce " >< br /></ div >
< h2 >< ? php _e ( 'Attributes' , 'woocommerce' ) ?> </h2>
< br class = " clear " />
< div id = " col-container " >
< div id = " col-right " >
< div class = " col-wrap " >
< table class = " widefat fixed " style = " width:100% " >
< thead >
< tr >
< th scope = " col " >< ? php _e ( 'Name' , 'woocommerce' ) ?> </th>
< th scope = " col " >< ? php _e ( 'Slug' , 'woocommerce' ) ?> </th>
< th scope = " col " >< ? php _e ( 'Type' , 'woocommerce' ) ?> </th>
< th scope = " col " >< ? php _e ( 'Order by' , 'woocommerce' ) ?> </th>
< th scope = " col " colspan = " 2 " >< ? php _e ( 'Terms' , 'woocommerce' ) ?> </th>
</ tr >
</ thead >
< tbody >
< ? php
2013-09-12 13:41:02 +00:00
$attribute_taxonomies = wc_get_attribute_taxonomies ();
2013-08-06 15:56:15 +00:00
if ( $attribute_taxonomies ) :
foreach ( $attribute_taxonomies as $tax ) :
?> <tr>
2013-09-12 13:41:02 +00:00
< td >< a href = " edit-tags.php?taxonomy=<?php echo esc_html(wc_attribute_taxonomy_name( $tax->attribute_name )); ?>&post_type=product " >< ? php echo esc_html ( $tax -> attribute_label ); ?> </a>
2013-08-06 15:56:15 +00:00
2013-10-22 15:46:59 +00:00
< div class = " row-actions " >< span class = " edit " >< a href = " <?php echo esc_url( add_query_arg('edit', $tax->attribute_id , 'admin.php?page=product_attributes') ); ?> " >< ? php _e ( 'Edit' , 'woocommerce' ); ?> </a> | </span><span class="delete"><a class="delete" href="<?php echo esc_url( wp_nonce_url( add_query_arg('delete', $tax->attribute_id, 'admin.php?page=product_attributes'), 'woocommerce-delete-attribute_' . $tax->attribute_id ) ); ?>"><?php _e( 'Delete', 'woocommerce' ); ?></a></span></div>
2013-08-06 15:56:15 +00:00
</ td >
< td >< ? php echo esc_html ( $tax -> attribute_name ); ?> </td>
2013-10-25 10:53:53 +00:00
< td >< ? php echo esc_html ( ucfirst ( $tax -> attribute_type ) ); ?> </td>
2013-08-06 15:56:15 +00:00
< td >< ? php
switch ( $tax -> attribute_orderby ) {
case 'name' :
_e ( 'Name' , 'woocommerce' );
break ;
case 'id' :
_e ( 'Term ID' , 'woocommerce' );
break ;
default :
_e ( 'Custom ordering' , 'woocommerce' );
break ;
}
?> </td>
< td >< ? php
2013-09-12 13:41:02 +00:00
if ( taxonomy_exists ( wc_attribute_taxonomy_name ( $tax -> attribute_name ))) :
2013-08-06 15:56:15 +00:00
$terms_array = array ();
2013-09-12 13:41:02 +00:00
$terms = get_terms ( wc_attribute_taxonomy_name ( $tax -> attribute_name ), 'orderby=name&hide_empty=0' );
2013-08-06 15:56:15 +00:00
if ( $terms ) :
foreach ( $terms as $term ) :
$terms_array [] = $term -> name ;
endforeach ;
echo implode ( ', ' , $terms_array );
else :
echo '<span class="na">–</span>' ;
endif ;
else :
echo '<span class="na">–</span>' ;
endif ;
?> </td>
2013-09-12 13:41:02 +00:00
< td >< a href = " edit-tags.php?taxonomy=<?php echo esc_html(wc_attribute_taxonomy_name( $tax->attribute_name )); ?>&post_type=product " class = " button alignright " >< ? php _e ( 'Configure terms' , 'woocommerce' ); ?> </a></td>
2013-08-06 15:56:15 +00:00
</ tr >< ? php
endforeach ;
else :
?> <tr><td colspan="6"><?php _e( 'No attributes currently exist.', 'woocommerce' ) ?></td></tr><?php
endif ;
?>
</ tbody >
</ table >
</ div >
</ div >
< div id = " col-left " >
< div class = " col-wrap " >
< div class = " form-wrap " >
< h3 >< ? php _e ( 'Add New Attribute' , 'woocommerce' ) ?> </h3>
< p >< ? php _e ( 'Attributes let you define extra product data, such as size or colour. You can use these attributes in the shop sidebar using the "layered nav" widgets. Please note: you cannot rename an attribute later on.' , 'woocommerce' ) ?> </p>
2013-10-22 15:46:59 +00:00
< form action = " admin.php?page=product_attributes " method = " post " >
2013-08-06 15:56:15 +00:00
< div class = " form-field " >
< label for = " attribute_label " >< ? php _e ( 'Name' , 'woocommerce' ); ?> </label>
< input name = " attribute_label " id = " attribute_label " type = " text " value = " " />
< p class = " description " >< ? php _e ( 'Name for the attribute (shown on the front-end).' , 'woocommerce' ); ?> </p>
</ div >
< div class = " form-field " >
< label for = " attribute_name " >< ? php _e ( 'Slug' , 'woocommerce' ); ?> </label>
< input name = " attribute_name " id = " attribute_name " type = " text " value = " " maxlength = " 28 " />
< p class = " description " >< ? php _e ( 'Unique slug/reference for the attribute; must be shorter than 28 characters.' , 'woocommerce' ); ?> </p>
</ div >
< div class = " form-field " >
< label for = " attribute_type " >< ? php _e ( 'Type' , 'woocommerce' ); ?> </label>
< select name = " attribute_type " id = " attribute_type " >
< option value = " select " >< ? php _e ( 'Select' , 'woocommerce' ) ?> </option>
< option value = " text " >< ? php _e ( 'Text' , 'woocommerce' ) ?> </option>
< ? php do_action ( 'woocommerce_admin_attribute_types' ); ?>
</ select >
2013-10-31 06:14:35 +00:00
< p class = " description " >< ? php _e ( 'Determines how you select attributes for products. Under admin panel -> products -> product data -> attributes -> values, <strong>Text</strong> allows manual entry whereas <strong>select</strong> allows pre-configured terms in a drop-down list.' , 'woocommerce' ); ?> </p>
2013-08-06 15:56:15 +00:00
</ div >
< div class = " form-field " >
< label for = " attribute_orderby " >< ? php _e ( 'Default sort order' , 'woocommerce' ); ?> </label>
< select name = " attribute_orderby " id = " attribute_orderby " >
< option value = " menu_order " >< ? php _e ( 'Custom ordering' , 'woocommerce' ) ?> </option>
< option value = " name " >< ? php _e ( 'Name' , 'woocommerce' ) ?> </option>
< option value = " id " >< ? php _e ( 'Term ID' , 'woocommerce' ) ?> </option>
</ select >
2013-10-31 06:14:35 +00:00
< p class = " description " >< ? php _e ( 'Determines the sort order of the terms on the frontend shop product pages. If using custom ordering, you can drag and drop the terms in this attribute.' , 'woocommerce' ); ?> </p>
2013-08-06 15:56:15 +00:00
</ div >
< p class = " submit " >< input type = " submit " name = " add_new_attribute " id = " submit " class = " button " value = " <?php _e( 'Add Attribute', 'woocommerce' ); ?> " ></ p >
< ? php wp_nonce_field ( 'woocommerce-add-new_attribute' ); ?>
</ form >
</ div >
</ div >
</ div >
</ div >
< script type = " text/javascript " >
/* <![CDATA[ */
jQuery ( 'a.delete' ) . click ( function (){
var answer = confirm ( " <?php _e( 'Are you sure you want to delete this attribute?', 'woocommerce' ); ?> " );
if ( answer ) return true ;
return false ;
});
/* ]]> */
</ script >
</ div >
< ? php
}
}
endif ;
return new WC_Admin_Attributes ();