2017-08-28 14:36:02 +00:00
|
|
|
/*global wc_add_to_cart_variation_params */
|
2013-02-07 10:12:08 +00:00
|
|
|
;(function ( $, window, document, undefined ) {
|
2016-12-15 12:58:41 +00:00
|
|
|
/**
|
|
|
|
* VariationForm class which handles variation forms and attributes.
|
|
|
|
*/
|
|
|
|
var VariationForm = function( $form ) {
|
2018-07-12 01:02:56 +00:00
|
|
|
var self = this;
|
|
|
|
|
|
|
|
self.$form = $form;
|
|
|
|
self.$attributeFields = $form.find( '.variations select' );
|
|
|
|
self.$singleVariation = $form.find( '.single_variation' );
|
|
|
|
self.$singleVariationWrap = $form.find( '.single_variation_wrap' );
|
|
|
|
self.$resetVariations = $form.find( '.reset_variations' );
|
|
|
|
self.$product = $form.closest( '.product' );
|
|
|
|
self.variationData = $form.data( 'product_variations' );
|
|
|
|
self.useAjax = false === self.variationData;
|
|
|
|
self.xhr = false;
|
|
|
|
self.loading = true;
|
2016-12-15 12:58:41 +00:00
|
|
|
|
|
|
|
// Initial state.
|
2018-07-12 01:02:56 +00:00
|
|
|
self.$singleVariationWrap.show();
|
|
|
|
self.$form.off( '.wc-variation-form' );
|
2016-12-15 12:58:41 +00:00
|
|
|
|
|
|
|
// Methods.
|
2021-01-07 22:16:59 +00:00
|
|
|
self.getChosenAttributes = self.getChosenAttributes.bind( self );
|
|
|
|
self.findMatchingVariations = self.findMatchingVariations.bind( self );
|
|
|
|
self.isMatch = self.isMatch.bind( self );
|
|
|
|
self.toggleResetLink = self.toggleResetLink.bind( self );
|
2016-12-15 12:58:41 +00:00
|
|
|
|
|
|
|
// Events.
|
2018-07-12 01:02:56 +00:00
|
|
|
$form.on( 'click.wc-variation-form', '.reset_variations', { variationForm: self }, self.onReset );
|
|
|
|
$form.on( 'reload_product_variations', { variationForm: self }, self.onReload );
|
|
|
|
$form.on( 'hide_variation', { variationForm: self }, self.onHide );
|
|
|
|
$form.on( 'show_variation', { variationForm: self }, self.onShow );
|
|
|
|
$form.on( 'click', '.single_add_to_cart_button', { variationForm: self }, self.onAddToCart );
|
|
|
|
$form.on( 'reset_data', { variationForm: self }, self.onResetDisplayedVariation );
|
|
|
|
$form.on( 'reset_image', { variationForm: self }, self.onResetImage );
|
|
|
|
$form.on( 'change.wc-variation-form', '.variations select', { variationForm: self }, self.onChange );
|
|
|
|
$form.on( 'found_variation.wc-variation-form', { variationForm: self }, self.onFoundVariation );
|
|
|
|
$form.on( 'check_variations.wc-variation-form', { variationForm: self }, self.onFindVariation );
|
|
|
|
$form.on( 'update_variation_values.wc-variation-form', { variationForm: self }, self.onUpdateAttributes );
|
2016-12-15 12:58:41 +00:00
|
|
|
|
2017-08-30 11:47:59 +00:00
|
|
|
// Init after gallery.
|
|
|
|
setTimeout( function() {
|
|
|
|
$form.trigger( 'check_variations' );
|
2020-05-15 04:26:13 +00:00
|
|
|
$form.trigger( 'wc_variation_form', self );
|
2018-07-12 01:02:56 +00:00
|
|
|
self.loading = false;
|
2017-08-30 11:47:59 +00:00
|
|
|
}, 100 );
|
2016-12-15 12:58:41 +00:00
|
|
|
};
|
2012-07-17 14:09:18 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
/**
|
|
|
|
* Reset all fields.
|
|
|
|
*/
|
|
|
|
VariationForm.prototype.onReset = function( event ) {
|
|
|
|
event.preventDefault();
|
2021-01-07 22:08:36 +00:00
|
|
|
event.data.variationForm.$attributeFields.val( '' ).trigger( 'change' );
|
2016-12-15 12:58:41 +00:00
|
|
|
event.data.variationForm.$form.trigger( 'reset_data' );
|
|
|
|
};
|
2015-07-09 14:56:20 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
/**
|
|
|
|
* Reload variation data from the DOM.
|
|
|
|
*/
|
|
|
|
VariationForm.prototype.onReload = function( event ) {
|
|
|
|
var form = event.data.variationForm;
|
|
|
|
form.variationData = form.$form.data( 'product_variations' );
|
|
|
|
form.useAjax = false === form.variationData;
|
|
|
|
form.$form.trigger( 'check_variations' );
|
|
|
|
};
|
2012-08-08 10:53:36 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
/**
|
|
|
|
* When a variation is hidden.
|
|
|
|
*/
|
|
|
|
VariationForm.prototype.onHide = function( event ) {
|
|
|
|
event.preventDefault();
|
2020-05-05 03:49:29 +00:00
|
|
|
event.data.variationForm.$form
|
|
|
|
.find( '.single_add_to_cart_button' )
|
|
|
|
.removeClass( 'wc-variation-is-unavailable' )
|
|
|
|
.addClass( 'disabled wc-variation-selection-needed' );
|
|
|
|
event.data.variationForm.$form
|
|
|
|
.find( '.woocommerce-variation-add-to-cart' )
|
|
|
|
.removeClass( 'woocommerce-variation-add-to-cart-enabled' )
|
|
|
|
.addClass( 'woocommerce-variation-add-to-cart-disabled' );
|
2016-12-15 12:58:41 +00:00
|
|
|
};
|
2015-10-27 19:24:32 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
/**
|
|
|
|
* When a variation is shown.
|
|
|
|
*/
|
|
|
|
VariationForm.prototype.onShow = function( event, variation, purchasable ) {
|
|
|
|
event.preventDefault();
|
|
|
|
if ( purchasable ) {
|
2020-05-05 03:49:29 +00:00
|
|
|
event.data.variationForm.$form
|
|
|
|
.find( '.single_add_to_cart_button' )
|
|
|
|
.removeClass( 'disabled wc-variation-selection-needed wc-variation-is-unavailable' );
|
|
|
|
event.data.variationForm.$form
|
|
|
|
.find( '.woocommerce-variation-add-to-cart' )
|
|
|
|
.removeClass( 'woocommerce-variation-add-to-cart-disabled' )
|
|
|
|
.addClass( 'woocommerce-variation-add-to-cart-enabled' );
|
2016-12-15 12:58:41 +00:00
|
|
|
} else {
|
2020-05-05 03:49:29 +00:00
|
|
|
event.data.variationForm.$form
|
|
|
|
.find( '.single_add_to_cart_button' )
|
|
|
|
.removeClass( 'wc-variation-selection-needed' )
|
|
|
|
.addClass( 'disabled wc-variation-is-unavailable' );
|
|
|
|
event.data.variationForm.$form
|
|
|
|
.find( '.woocommerce-variation-add-to-cart' )
|
|
|
|
.removeClass( 'woocommerce-variation-add-to-cart-enabled' )
|
|
|
|
.addClass( 'woocommerce-variation-add-to-cart-disabled' );
|
2016-12-15 12:58:41 +00:00
|
|
|
}
|
2019-11-22 22:08:31 +00:00
|
|
|
|
|
|
|
// If present, the media element library needs initialized on the variation description.
|
|
|
|
if ( wp.mediaelement ) {
|
|
|
|
event.data.variationForm.$form.find( '.wp-audio-shortcode, .wp-video-shortcode' )
|
|
|
|
.not( '.mejs-container' )
|
|
|
|
.filter(
|
|
|
|
function () {
|
|
|
|
return ! $( this ).parent().hasClass( 'mejs-mediaelement' );
|
|
|
|
}
|
|
|
|
)
|
|
|
|
.mediaelementplayer( wp.mediaelement.settings );
|
|
|
|
}
|
2016-12-15 12:58:41 +00:00
|
|
|
};
|
2014-12-10 21:43:26 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
/**
|
|
|
|
* When the cart button is pressed.
|
|
|
|
*/
|
|
|
|
VariationForm.prototype.onAddToCart = function( event ) {
|
|
|
|
if ( $( this ).is('.disabled') ) {
|
|
|
|
event.preventDefault();
|
2015-10-27 19:59:26 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
if ( $( this ).is('.wc-variation-is-unavailable') ) {
|
|
|
|
window.alert( wc_add_to_cart_variation_params.i18n_unavailable_text );
|
|
|
|
} else if ( $( this ).is('.wc-variation-selection-needed') ) {
|
|
|
|
window.alert( wc_add_to_cart_variation_params.i18n_make_a_selection_text );
|
2015-07-09 14:56:20 +00:00
|
|
|
}
|
2016-12-15 12:58:41 +00:00
|
|
|
}
|
|
|
|
};
|
2012-12-10 13:11:05 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
/**
|
|
|
|
* When displayed variation data is reset.
|
|
|
|
*/
|
|
|
|
VariationForm.prototype.onResetDisplayedVariation = function( event ) {
|
|
|
|
var form = event.data.variationForm;
|
|
|
|
form.$product.find( '.product_meta' ).find( '.sku' ).wc_reset_content();
|
2020-05-05 03:49:29 +00:00
|
|
|
form.$product
|
|
|
|
.find( '.product_weight, .woocommerce-product-attributes-item--weight .woocommerce-product-attributes-item__value' )
|
|
|
|
.wc_reset_content();
|
|
|
|
form.$product
|
|
|
|
.find( '.product_dimensions, .woocommerce-product-attributes-item--dimensions .woocommerce-product-attributes-item__value' )
|
|
|
|
.wc_reset_content();
|
2016-12-15 12:58:41 +00:00
|
|
|
form.$form.trigger( 'reset_image' );
|
|
|
|
form.$singleVariation.slideUp( 200 ).trigger( 'hide_variation' );
|
|
|
|
};
|
2012-12-10 13:11:05 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
/**
|
|
|
|
* When the product image is reset.
|
|
|
|
*/
|
|
|
|
VariationForm.prototype.onResetImage = function( event ) {
|
|
|
|
event.data.variationForm.$form.wc_variations_image_update( false );
|
|
|
|
};
|
2012-08-08 10:53:36 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
/**
|
|
|
|
* Looks for matching variations for current selected attributes.
|
|
|
|
*/
|
2020-05-15 04:27:40 +00:00
|
|
|
VariationForm.prototype.onFindVariation = function( event, chosenAttributes ) {
|
2017-01-06 11:33:06 +00:00
|
|
|
var form = event.data.variationForm,
|
2020-05-15 04:27:40 +00:00
|
|
|
attributes = 'undefined' !== typeof chosenAttributes ? chosenAttributes : form.getChosenAttributes(),
|
2017-01-06 11:33:06 +00:00
|
|
|
currentAttributes = attributes.data;
|
2012-08-08 10:53:36 +00:00
|
|
|
|
2020-05-15 04:28:39 +00:00
|
|
|
if ( attributes.count && attributes.count === attributes.chosenCount ) {
|
2017-01-06 11:33:06 +00:00
|
|
|
if ( form.useAjax ) {
|
|
|
|
if ( form.xhr ) {
|
|
|
|
form.xhr.abort();
|
|
|
|
}
|
|
|
|
form.$form.block( { message: null, overlayCSS: { background: '#fff', opacity: 0.6 } } );
|
|
|
|
currentAttributes.product_id = parseInt( form.$form.data( 'product_id' ), 10 );
|
|
|
|
currentAttributes.custom_data = form.$form.data( 'custom_data' );
|
|
|
|
form.xhr = $.ajax( {
|
2017-08-28 14:36:02 +00:00
|
|
|
url: wc_add_to_cart_variation_params.wc_ajax_url.toString().replace( '%%endpoint%%', 'get_variation' ),
|
2017-01-06 11:33:06 +00:00
|
|
|
type: 'POST',
|
|
|
|
data: currentAttributes,
|
|
|
|
success: function( variation ) {
|
|
|
|
if ( variation ) {
|
|
|
|
form.$form.trigger( 'found_variation', [ variation ] );
|
|
|
|
} else {
|
|
|
|
form.$form.trigger( 'reset_data' );
|
2018-03-01 10:32:26 +00:00
|
|
|
attributes.chosenCount = 0;
|
|
|
|
|
|
|
|
if ( ! form.loading ) {
|
2020-05-05 03:49:29 +00:00
|
|
|
form.$form
|
|
|
|
.find( '.single_variation' )
|
|
|
|
.after(
|
|
|
|
'<p class="wc-no-matching-variations woocommerce-info">' +
|
|
|
|
wc_add_to_cart_variation_params.i18n_no_matching_variations_text +
|
|
|
|
'</p>'
|
|
|
|
);
|
2018-03-01 10:32:26 +00:00
|
|
|
form.$form.find( '.wc-no-matching-variations' ).slideDown( 200 );
|
|
|
|
}
|
2017-01-06 11:33:06 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
complete: function() {
|
|
|
|
form.$form.unblock();
|
|
|
|
}
|
|
|
|
} );
|
2015-10-28 15:43:06 +00:00
|
|
|
} else {
|
2017-01-06 11:33:06 +00:00
|
|
|
form.$form.trigger( 'update_variation_values' );
|
|
|
|
|
|
|
|
var matching_variations = form.findMatchingVariations( form.variationData, currentAttributes ),
|
|
|
|
variation = matching_variations.shift();
|
|
|
|
|
|
|
|
if ( variation ) {
|
|
|
|
form.$form.trigger( 'found_variation', [ variation ] );
|
|
|
|
} else {
|
|
|
|
form.$form.trigger( 'reset_data' );
|
2018-03-01 10:32:26 +00:00
|
|
|
attributes.chosenCount = 0;
|
|
|
|
|
|
|
|
if ( ! form.loading ) {
|
2020-05-05 03:49:29 +00:00
|
|
|
form.$form
|
|
|
|
.find( '.single_variation' )
|
|
|
|
.after(
|
|
|
|
'<p class="wc-no-matching-variations woocommerce-info">' +
|
|
|
|
wc_add_to_cart_variation_params.i18n_no_matching_variations_text +
|
|
|
|
'</p>'
|
|
|
|
);
|
2018-03-01 10:32:26 +00:00
|
|
|
form.$form.find( '.wc-no-matching-variations' ).slideDown( 200 );
|
|
|
|
}
|
2017-01-06 11:33:06 +00:00
|
|
|
}
|
2015-07-09 14:56:20 +00:00
|
|
|
}
|
2016-12-15 12:58:41 +00:00
|
|
|
} else {
|
2017-01-15 08:17:03 +00:00
|
|
|
form.$form.trigger( 'update_variation_values' );
|
2016-12-15 12:58:41 +00:00
|
|
|
form.$form.trigger( 'reset_data' );
|
|
|
|
}
|
2014-10-20 08:59:01 +00:00
|
|
|
|
2017-01-06 11:33:06 +00:00
|
|
|
// Show reset link.
|
|
|
|
form.toggleResetLink( attributes.chosenCount > 0 );
|
2016-12-15 12:58:41 +00:00
|
|
|
};
|
2012-08-21 15:41:18 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
/**
|
|
|
|
* Triggered when a variation has been found which matches all attributes.
|
|
|
|
*/
|
|
|
|
VariationForm.prototype.onFoundVariation = function( event, variation ) {
|
|
|
|
var form = event.data.variationForm,
|
|
|
|
$sku = form.$product.find( '.product_meta' ).find( '.sku' ),
|
2020-05-05 03:49:29 +00:00
|
|
|
$weight = form.$product.find(
|
|
|
|
'.product_weight, .woocommerce-product-attributes-item--weight .woocommerce-product-attributes-item__value'
|
|
|
|
),
|
|
|
|
$dimensions = form.$product.find(
|
|
|
|
'.product_dimensions, .woocommerce-product-attributes-item--dimensions .woocommerce-product-attributes-item__value'
|
|
|
|
),
|
2016-12-15 12:58:41 +00:00
|
|
|
$qty = form.$singleVariationWrap.find( '.quantity' ),
|
|
|
|
purchasable = true,
|
|
|
|
variation_id = '',
|
|
|
|
template = false,
|
|
|
|
$template_html = '';
|
|
|
|
|
|
|
|
if ( variation.sku ) {
|
|
|
|
$sku.wc_set_content( variation.sku );
|
|
|
|
} else {
|
|
|
|
$sku.wc_reset_content();
|
|
|
|
}
|
2012-08-21 15:41:18 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
if ( variation.weight ) {
|
2017-05-29 09:31:31 +00:00
|
|
|
$weight.wc_set_content( variation.weight_html );
|
2016-12-15 12:58:41 +00:00
|
|
|
} else {
|
|
|
|
$weight.wc_reset_content();
|
|
|
|
}
|
2012-08-21 15:41:18 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
if ( variation.dimensions ) {
|
2018-11-06 22:10:20 +00:00
|
|
|
// Decode HTML entities.
|
|
|
|
$dimensions.wc_set_content( $.parseHTML( variation.dimensions_html )[0].data );
|
2016-12-15 12:58:41 +00:00
|
|
|
} else {
|
|
|
|
$dimensions.wc_reset_content();
|
|
|
|
}
|
2012-08-22 06:51:19 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
form.$form.wc_variations_image_update( variation );
|
2018-07-12 01:02:56 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
if ( ! variation.variation_is_visible ) {
|
2018-06-15 15:08:34 +00:00
|
|
|
template = wp_template( 'unavailable-variation-template' );
|
2016-12-15 12:58:41 +00:00
|
|
|
} else {
|
2018-06-15 15:08:34 +00:00
|
|
|
template = wp_template( 'variation-template' );
|
2016-12-15 12:58:41 +00:00
|
|
|
variation_id = variation.variation_id;
|
|
|
|
}
|
2012-07-17 14:09:18 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
$template_html = template( {
|
|
|
|
variation: variation
|
|
|
|
} );
|
|
|
|
$template_html = $template_html.replace( '/*<![CDATA[*/', '' );
|
|
|
|
$template_html = $template_html.replace( '/*]]>*/', '' );
|
2012-08-08 10:53:36 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
form.$singleVariation.html( $template_html );
|
2021-01-07 22:08:36 +00:00
|
|
|
form.$form.find( 'input[name="variation_id"], input.variation_id' ).val( variation.variation_id ).trigger( 'change' );
|
2012-07-17 14:09:18 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
// Hide or show qty input
|
|
|
|
if ( variation.is_sold_individually === 'yes' ) {
|
2021-01-07 22:08:36 +00:00
|
|
|
$qty.find( 'input.qty' ).val( '1' ).attr( 'min', '1' ).attr( 'max', '' ).trigger( 'change' );
|
2016-12-15 12:58:41 +00:00
|
|
|
$qty.hide();
|
|
|
|
} else {
|
2020-06-18 04:42:25 +00:00
|
|
|
|
|
|
|
var $qty_input = $qty.find( 'input.qty' ),
|
|
|
|
qty_val = parseFloat( $qty_input.val() );
|
|
|
|
|
|
|
|
if ( isNaN( qty_val ) ) {
|
|
|
|
qty_val = variation.min_qty;
|
|
|
|
} else {
|
|
|
|
qty_val = qty_val > parseFloat( variation.max_qty ) ? variation.max_qty : qty_val;
|
|
|
|
qty_val = qty_val < parseFloat( variation.min_qty ) ? variation.min_qty : qty_val;
|
|
|
|
}
|
|
|
|
|
2021-01-07 22:08:36 +00:00
|
|
|
$qty_input.attr( 'min', variation.min_qty ).attr( 'max', variation.max_qty ).val( qty_val ).trigger( 'change' );
|
2016-12-15 12:58:41 +00:00
|
|
|
$qty.show();
|
|
|
|
}
|
2012-08-08 10:53:36 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
// Enable or disable the add to cart button
|
|
|
|
if ( ! variation.is_purchasable || ! variation.is_in_stock || ! variation.variation_is_visible ) {
|
|
|
|
purchasable = false;
|
|
|
|
}
|
2012-08-21 15:41:18 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
// Reveal
|
2021-01-07 22:15:15 +00:00
|
|
|
if ( form.$singleVariation.text().trim() ) {
|
2016-12-15 12:58:41 +00:00
|
|
|
form.$singleVariation.slideDown( 200 ).trigger( 'show_variation', [ variation, purchasable ] );
|
|
|
|
} else {
|
|
|
|
form.$singleVariation.show().trigger( 'show_variation', [ variation, purchasable ] );
|
|
|
|
}
|
|
|
|
};
|
2012-08-08 10:53:36 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
/**
|
|
|
|
* Triggered when an attribute field changes.
|
|
|
|
*/
|
|
|
|
VariationForm.prototype.onChange = function( event ) {
|
|
|
|
var form = event.data.variationForm;
|
2012-08-08 10:53:36 +00:00
|
|
|
|
2021-01-07 22:08:36 +00:00
|
|
|
form.$form.find( 'input[name="variation_id"], input.variation_id' ).val( '' ).trigger( 'change' );
|
2016-12-15 12:58:41 +00:00
|
|
|
form.$form.find( '.wc-no-matching-variations' ).remove();
|
2014-02-22 04:08:08 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
if ( form.useAjax ) {
|
2017-01-06 11:33:06 +00:00
|
|
|
form.$form.trigger( 'check_variations' );
|
2016-12-15 12:58:41 +00:00
|
|
|
} else {
|
|
|
|
form.$form.trigger( 'woocommerce_variation_select_change' );
|
|
|
|
form.$form.trigger( 'check_variations' );
|
|
|
|
}
|
2012-08-08 10:53:36 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
// Custom event for when variation selection has been changed
|
|
|
|
form.$form.trigger( 'woocommerce_variation_has_changed' );
|
|
|
|
};
|
2012-08-08 10:53:36 +00:00
|
|
|
|
2016-12-22 22:52:49 +00:00
|
|
|
/**
|
|
|
|
* Escape quotes in a string.
|
|
|
|
* @param {string} string
|
|
|
|
* @return {string}
|
|
|
|
*/
|
|
|
|
VariationForm.prototype.addSlashes = function( string ) {
|
|
|
|
string = string.replace( /'/g, '\\\'' );
|
|
|
|
string = string.replace( /"/g, '\\\"' );
|
|
|
|
return string;
|
2016-12-22 22:56:01 +00:00
|
|
|
};
|
2016-12-22 22:52:49 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
/**
|
|
|
|
* Updates attributes in the DOM to show valid values.
|
|
|
|
*/
|
|
|
|
VariationForm.prototype.onUpdateAttributes = function( event ) {
|
|
|
|
var form = event.data.variationForm,
|
|
|
|
attributes = form.getChosenAttributes(),
|
|
|
|
currentAttributes = attributes.data;
|
2012-08-08 10:53:36 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
if ( form.useAjax ) {
|
|
|
|
return;
|
|
|
|
}
|
2012-07-17 14:09:18 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
// Loop through selects and disable/enable options based on selections.
|
|
|
|
form.$attributeFields.each( function( index, el ) {
|
|
|
|
var current_attr_select = $( el ),
|
|
|
|
current_attr_name = current_attr_select.data( 'attribute_name' ) || current_attr_select.attr( 'name' ),
|
|
|
|
show_option_none = $( el ).data( 'show_option_none' ),
|
|
|
|
option_gt_filter = ':gt(0)',
|
|
|
|
attached_options_count = 0,
|
|
|
|
new_attr_select = $( '<select/>' ),
|
|
|
|
selected_attr_val = current_attr_select.val() || '',
|
|
|
|
selected_attr_val_valid = true;
|
|
|
|
|
|
|
|
// Reference options set at first.
|
|
|
|
if ( ! current_attr_select.data( 'attribute_html' ) ) {
|
|
|
|
var refSelect = current_attr_select.clone();
|
|
|
|
|
2021-01-21 17:22:30 +00:00
|
|
|
refSelect.find( 'option' ).prop( 'disabled attached', false ).prop( 'selected', false );
|
2016-12-15 12:58:41 +00:00
|
|
|
|
2020-05-05 03:49:29 +00:00
|
|
|
// Legacy data attribute.
|
|
|
|
current_attr_select.data(
|
|
|
|
'attribute_options',
|
|
|
|
refSelect.find( 'option' + option_gt_filter ).get()
|
|
|
|
);
|
2016-12-15 12:58:41 +00:00
|
|
|
current_attr_select.data( 'attribute_html', refSelect.html() );
|
|
|
|
}
|
2013-02-01 11:49:29 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
new_attr_select.html( current_attr_select.data( 'attribute_html' ) );
|
2013-02-01 11:49:29 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
// The attribute of this select field should not be taken into account when calculating its matching variations:
|
|
|
|
// The constraints of this attribute are shaped by the values of the other attributes.
|
|
|
|
var checkAttributes = $.extend( true, {}, currentAttributes );
|
2012-08-08 10:53:36 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
checkAttributes[ current_attr_name ] = '';
|
2014-09-26 16:22:43 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
var variations = form.findMatchingVariations( form.variationData, checkAttributes );
|
2012-08-08 10:53:36 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
// Loop through variations.
|
|
|
|
for ( var num in variations ) {
|
|
|
|
if ( typeof( variations[ num ] ) !== 'undefined' ) {
|
|
|
|
var variationAttributes = variations[ num ].attributes;
|
2012-08-08 10:53:36 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
for ( var attr_name in variationAttributes ) {
|
|
|
|
if ( variationAttributes.hasOwnProperty( attr_name ) ) {
|
|
|
|
var attr_val = variationAttributes[ attr_name ],
|
|
|
|
variation_active = '';
|
2012-08-08 10:53:36 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
if ( attr_name === current_attr_name ) {
|
|
|
|
if ( variations[ num ].variation_is_active ) {
|
|
|
|
variation_active = 'enabled';
|
|
|
|
}
|
2013-02-01 11:49:29 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
if ( attr_val ) {
|
2019-01-08 14:08:27 +00:00
|
|
|
// Decode entities.
|
2016-12-15 12:58:41 +00:00
|
|
|
attr_val = $( '<div/>' ).html( attr_val ).text();
|
|
|
|
|
2019-01-08 14:08:27 +00:00
|
|
|
// Attach to matching options by value. This is done to compare
|
|
|
|
// TEXT values rather than any HTML entities.
|
2019-03-15 12:21:51 +00:00
|
|
|
var $option_elements = new_attr_select.find( 'option' );
|
|
|
|
if ( $option_elements.length ) {
|
|
|
|
for (var i = 0, len = $option_elements.length; i < len; i++) {
|
|
|
|
var $option_element = $( $option_elements[i] ),
|
|
|
|
option_value = $option_element.val();
|
|
|
|
|
|
|
|
if ( attr_val === option_value ) {
|
|
|
|
$option_element.addClass( 'attached ' + variation_active );
|
|
|
|
break;
|
|
|
|
}
|
2019-01-08 14:08:27 +00:00
|
|
|
}
|
2019-03-15 12:21:51 +00:00
|
|
|
}
|
2016-12-15 12:58:41 +00:00
|
|
|
} else {
|
|
|
|
// Attach all apart from placeholder.
|
|
|
|
new_attr_select.find( 'option:gt(0)' ).addClass( 'attached ' + variation_active );
|
2014-02-22 04:08:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-03-17 05:19:58 +00:00
|
|
|
}
|
2016-12-15 12:58:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Count available options.
|
|
|
|
attached_options_count = new_attr_select.find( 'option.attached' ).length;
|
2014-02-22 04:08:08 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
// Check if current selection is in attached options.
|
2019-01-08 14:08:27 +00:00
|
|
|
if ( selected_attr_val ) {
|
2016-12-15 12:58:41 +00:00
|
|
|
selected_attr_val_valid = false;
|
2019-01-08 14:08:27 +00:00
|
|
|
|
2019-02-04 12:51:20 +00:00
|
|
|
if ( 0 !== attached_options_count ) {
|
2019-03-15 12:21:51 +00:00
|
|
|
new_attr_select.find( 'option.attached.enabled' ).each( function() {
|
2019-01-08 14:08:27 +00:00
|
|
|
var option_value = $( this ).val();
|
|
|
|
|
|
|
|
if ( selected_attr_val === option_value ) {
|
|
|
|
selected_attr_val_valid = true;
|
|
|
|
return false; // break.
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2016-12-15 12:58:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Detach the placeholder if:
|
|
|
|
// - Valid options exist.
|
|
|
|
// - The current selection is non-empty.
|
|
|
|
// - The current selection is valid.
|
|
|
|
// - Placeholders are not set to be permanently visible.
|
|
|
|
if ( attached_options_count > 0 && selected_attr_val && selected_attr_val_valid && ( 'no' === show_option_none ) ) {
|
|
|
|
new_attr_select.find( 'option:first' ).remove();
|
|
|
|
option_gt_filter = '';
|
|
|
|
}
|
2012-08-21 15:41:18 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
// Detach unattached.
|
|
|
|
new_attr_select.find( 'option' + option_gt_filter + ':not(.attached)' ).remove();
|
2012-08-21 15:41:18 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
// Finally, copy to DOM and set value.
|
|
|
|
current_attr_select.html( new_attr_select.html() );
|
|
|
|
current_attr_select.find( 'option' + option_gt_filter + ':not(.enabled)' ).prop( 'disabled', true );
|
|
|
|
|
2017-01-04 12:19:40 +00:00
|
|
|
// Choose selected value.
|
2016-12-15 12:58:41 +00:00
|
|
|
if ( selected_attr_val ) {
|
|
|
|
// If the previously selected value is no longer available, fall back to the placeholder (it's going to be there).
|
|
|
|
if ( selected_attr_val_valid ) {
|
|
|
|
current_attr_select.val( selected_attr_val );
|
2016-11-29 08:21:00 +00:00
|
|
|
} else {
|
2021-01-07 22:08:36 +00:00
|
|
|
current_attr_select.val( '' ).trigger( 'change' );
|
2016-11-29 08:21:00 +00:00
|
|
|
}
|
2017-01-04 12:19:40 +00:00
|
|
|
} else {
|
|
|
|
current_attr_select.val( '' ); // No change event to prevent infinite loop.
|
2016-12-15 12:58:41 +00:00
|
|
|
}
|
|
|
|
});
|
2016-11-29 08:21:00 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
// Custom event for when variations have been updated.
|
|
|
|
form.$form.trigger( 'woocommerce_update_variation_values' );
|
|
|
|
};
|
2016-11-29 08:21:00 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
/**
|
|
|
|
* Get chosen attributes from form.
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
VariationForm.prototype.getChosenAttributes = function() {
|
|
|
|
var data = {};
|
|
|
|
var count = 0;
|
|
|
|
var chosen = 0;
|
2013-10-10 15:34:44 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
this.$attributeFields.each( function() {
|
|
|
|
var attribute_name = $( this ).data( 'attribute_name' ) || $( this ).attr( 'name' );
|
|
|
|
var value = $( this ).val() || '';
|
2013-02-07 10:12:08 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
if ( value.length > 0 ) {
|
|
|
|
chosen ++;
|
|
|
|
}
|
|
|
|
|
|
|
|
count ++;
|
|
|
|
data[ attribute_name ] = value;
|
|
|
|
});
|
2013-02-07 10:12:08 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
return {
|
|
|
|
'count' : count,
|
|
|
|
'chosenCount': chosen,
|
|
|
|
'data' : data
|
|
|
|
};
|
2015-07-09 14:56:20 +00:00
|
|
|
};
|
2013-02-07 10:12:08 +00:00
|
|
|
|
2015-07-09 14:56:20 +00:00
|
|
|
/**
|
2016-12-15 12:58:41 +00:00
|
|
|
* Find matching variations for attributes.
|
2015-07-09 14:56:20 +00:00
|
|
|
*/
|
2016-12-15 12:58:41 +00:00
|
|
|
VariationForm.prototype.findMatchingVariations = function( variations, attributes ) {
|
|
|
|
var matching = [];
|
|
|
|
for ( var i = 0; i < variations.length; i++ ) {
|
|
|
|
var variation = variations[i];
|
2013-02-07 10:12:08 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
if ( this.isMatch( variation.attributes, attributes ) ) {
|
|
|
|
matching.push( variation );
|
2015-07-09 14:56:20 +00:00
|
|
|
}
|
2016-12-15 12:58:41 +00:00
|
|
|
}
|
|
|
|
return matching;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* See if attributes match.
|
|
|
|
* @return {Boolean}
|
|
|
|
*/
|
|
|
|
VariationForm.prototype.isMatch = function( variation_attributes, attributes ) {
|
|
|
|
var match = true;
|
|
|
|
for ( var attr_name in variation_attributes ) {
|
|
|
|
if ( variation_attributes.hasOwnProperty( attr_name ) ) {
|
|
|
|
var val1 = variation_attributes[ attr_name ];
|
|
|
|
var val2 = attributes[ attr_name ];
|
|
|
|
if ( val1 !== undefined && val2 !== undefined && val1.length !== 0 && val2.length !== 0 && val1 !== val2 ) {
|
|
|
|
match = false;
|
2014-03-17 05:19:58 +00:00
|
|
|
}
|
2015-07-09 14:56:20 +00:00
|
|
|
}
|
|
|
|
}
|
2016-12-15 12:58:41 +00:00
|
|
|
return match;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Show or hide the reset link.
|
|
|
|
*/
|
|
|
|
VariationForm.prototype.toggleResetLink = function( on ) {
|
|
|
|
if ( on ) {
|
|
|
|
if ( this.$resetVariations.css( 'visibility' ) === 'hidden' ) {
|
|
|
|
this.$resetVariations.css( 'visibility', 'visible' ).hide().fadeIn();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
this.$resetVariations.css( 'visibility', 'hidden' );
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function to call wc_variation_form on jquery selector.
|
|
|
|
*/
|
|
|
|
$.fn.wc_variation_form = function() {
|
|
|
|
new VariationForm( this );
|
|
|
|
return this;
|
2014-02-22 04:08:08 +00:00
|
|
|
};
|
2013-02-07 10:12:08 +00:00
|
|
|
|
2015-10-28 13:04:28 +00:00
|
|
|
/**
|
|
|
|
* Stores the default text for an element so it can be reset later
|
|
|
|
*/
|
|
|
|
$.fn.wc_set_content = function( content ) {
|
|
|
|
if ( undefined === this.attr( 'data-o_content' ) ) {
|
|
|
|
this.attr( 'data-o_content', this.text() );
|
|
|
|
}
|
|
|
|
this.text( content );
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Stores the default text for an element so it can be reset later
|
|
|
|
*/
|
|
|
|
$.fn.wc_reset_content = function() {
|
|
|
|
if ( undefined !== this.attr( 'data-o_content' ) ) {
|
|
|
|
this.text( this.attr( 'data-o_content' ) );
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Stores a default attribute for an element so it can be reset later
|
|
|
|
*/
|
|
|
|
$.fn.wc_set_variation_attr = function( attr, value ) {
|
|
|
|
if ( undefined === this.attr( 'data-o_' + attr ) ) {
|
|
|
|
this.attr( 'data-o_' + attr, ( ! this.attr( attr ) ) ? '' : this.attr( attr ) );
|
|
|
|
}
|
WIP - Product CRUD (#12065)
* Created function to get the catalog visibility options
* First methods for WP_Product crud
* Product set methods
* Fixed several erros while setting data
* First methods for WP_Product crud
* Product set methods
* Fixed several erros while setting data
* Hardcode the get_type per product class
* Initial look through getters and setters and abstract data
* Missing var
* Add related product functions and deprecate those in class.
* No need to exclude ID
* Fixed coding standards and improved the docblocks
* Get cached terms from wc_get_related_terms()
* Fixed wrong variable in wc_get_related_terms
* Use count() instead of sizeof()
* Sanitize ids later
* Remove unneeded comments
* wc_get_product_term_ids instead of related wording and use in other places.
get_the_terms is used here and also handles caching, something
wp_get_post_terms does not.
* Clean up the abstract product class a bit, deprecate two functions we have renamed, make update & create work properly, and add some tests for it.
* Bump template version
* Handle PR feedback: Remove duplicate regular_price update, allow changing of post status for products, remove deprecation for get_title since we might still offer it as a function
* Made abstract function useful
* External Product CRUD
* _virtual meta should be 'no', not taxable, in product unit test helper
* Grouped product class
* Tests
* Move children to meta and update test
* Use get_upsell_ids
* Spacing in query
* Moving and refactoring methods
* Availability html
* Tidy/add todos
* Rename method
* Put back review functions (still todo)
* missing $this
* get_price_including_tax/excluding_tax functions
* wc_get_price_to_display
* Price handling
* [Product CRUD] Variable (#12146)
* [Product CRUD] Variable Products
* Handle PR feedback.
* [Product CRUD] Grouped Handling (#12151)
* Handle grouped product saving
* Update routine
* [Product CRUD] Product crud terms (#12149)
* Category and tag id handling
* Replace template functions
* Remove todo
* Handle default name in save function
* Product crud admin save routine (#12174)
* Initial props
* Work on admin saving
* Set/get attributes
* Atom was moaning about this before but no longer.
* Update get_shipping_class
* WC_Product_Attribute
* Use getter in admin panel
* Fix attribute saving
* Spacing
* Fix comment
* wc_implode_text_attributes helper function
* [Product CRUD] Product crud admin use getters (#12196)
* Initial props
* Work on admin saving
* Set/get attributes
* Atom was moaning about this before but no longer.
* Update get_shipping_class
* WC_Product_Attribute
* Use getter in admin panel
* Fix attribute saving
* Move settings into new files
* Refactor panels and use getters
* Use getters for variation panel
* Revert save variation changes for now
* Add todos
* Fix downloads
* REST API CRUD Updates
* Additional API updates/fixes. Added some todos
* Fix final failing tests and implementing setters/getters and attributes functionality.
* Fix comparison for is_on_sale and remove download_type from WC_Product.
* Add a wc_get_products wrapper.
* Remove the download type input from the product data metabox for downloadable products. (#12221)
* [Product CRUD] Variations - setters, getters and admin. (#12228)
* Started on variation changes
* Stock functions
* Variation class
* Bulk change ->id to get_id() to fix variation form display
* Missing status
* Fix add to cart
* Start on stored data save
* save variation
* Save_variations
* Variation edit panel
* Save variations code works.
* Remove stored data code and fix save
* Improve legacy class
* wc_bool_to_string
* prepare_set_attributes
* Use wc_get_products
* More feedback fixes
* Feedback fixes
* Implement CRUD in the legacy REST API
* Handle PR feedback
* [Product CRUD] Getter setter proxy methods (#12236)
* Started on variation changes
* Stock functions
* Variation class
* Bulk change ->id to get_id() to fix variation form display
* Missing status
* Fix add to cart
* Start on stored data save
* save variation
* Save_variations
* Variation edit panel
* Save variations code works.
* Remove stored data code and fix save
* Improve legacy class
* wc_bool_to_string
* prepare_set_attributes
* Use wc_get_products
* More feedback fixes
* get_prop implementation in abstract and data classes
* Implement set_prop
* Change handling
* Array key exists
* set_object_read
* Use get_the_terms() instead of wp_get_post_terms()
wp_get_post_terms() is a wrapper around wp_get_object_terms() which does not
use the object cache, and generates a database query every time it is used.
get_the_terms() however can use data from the object cache if present.
* Allow WP_Query to preload post data, and meta in wc_get_products()
Allow WP_Query to bulk query for post data and meta if more than
just IDs are requested from wc_get_products(). Reduces query count
significantly.
* [Product CRUD] Variable, variation, notices, and stock handling (#12277)
* No longer needed
* Remove old todos
* Use getters in admin list
* Related and upsells update for CRUD
* Fix notice in gallery
* Variable fixes and todos
* Context
* Price sync
* Revert variation attributes change
* Return parent data in view context
* Defer term counting
* wc_find_matching_product_variation
* Stock manage tweaks
* Stock fixes
* Correct id
* correct id
* Better sync
* Data logic setter fix
* feedback
* First methods for WP_Product crud
* Product set methods
* Fixed several erros while setting data
* Hardcode the get_type per product class
* Initial look through getters and setters and abstract data
* Missing var
* Fixed coding standards and improved the docblocks
* Get cached terms from wc_get_related_terms()
* Fixed wrong variable in wc_get_related_terms
* Use count() instead of sizeof()
* Add related product functions and deprecate those in class.
* No need to exclude ID
* Sanitize ids later
* Clean up the abstract product class a bit, deprecate two functions we have renamed, make update & create work properly, and add some tests for it.
* Remove unneeded comments
* wc_get_product_term_ids instead of related wording and use in other places.
get_the_terms is used here and also handles caching, something
wp_get_post_terms does not.
* Handle PR feedback: Remove duplicate regular_price update, allow changing of post status for products, remove deprecation for get_title since we might still offer it as a function
* External Product CRUD
* _virtual meta should be 'no', not taxable, in product unit test helper
* Bump template version
* Made abstract function useful
* Grouped product class
* Tests
* Move children to meta and update test
* Use get_upsell_ids
* Spacing in query
* Moving and refactoring methods
* Availability html
* Tidy/add todos
* Rename method
* Put back review functions (still todo)
* missing $this
* get_price_including_tax/excluding_tax functions
* wc_get_price_to_display
* Price handling
* [Product CRUD] Variable (#12146)
* [Product CRUD] Variable Products
* Handle PR feedback.
* [Product CRUD] Grouped Handling (#12151)
* Handle grouped product saving
* Update routine
* [Product CRUD] Product crud terms (#12149)
* Category and tag id handling
* Replace template functions
* Remove todo
* Handle default name in save function
* Product crud admin save routine (#12174)
* Initial props
* Work on admin saving
* Set/get attributes
* Atom was moaning about this before but no longer.
* Update get_shipping_class
* WC_Product_Attribute
* Use getter in admin panel
* Fix attribute saving
* Spacing
* Fix comment
* wc_implode_text_attributes helper function
* [Product CRUD] Product crud admin use getters (#12196)
* Initial props
* Work on admin saving
* Set/get attributes
* Atom was moaning about this before but no longer.
* Update get_shipping_class
* WC_Product_Attribute
* Use getter in admin panel
* Fix attribute saving
* Move settings into new files
* Refactor panels and use getters
* Use getters for variation panel
* Revert save variation changes for now
* Add todos
* Fix downloads
* REST API CRUD Updates
* Additional API updates/fixes. Added some todos
* Fix final failing tests and implementing setters/getters and attributes functionality.
* Fix comparison for is_on_sale and remove download_type from WC_Product.
* Add a wc_get_products wrapper.
* Remove the download type input from the product data metabox for downloadable products. (#12221)
* [Product CRUD] Variations - setters, getters and admin. (#12228)
* Started on variation changes
* Stock functions
* Variation class
* Bulk change ->id to get_id() to fix variation form display
* Missing status
* Fix add to cart
* Start on stored data save
* save variation
* Save_variations
* Variation edit panel
* Save variations code works.
* Remove stored data code and fix save
* Improve legacy class
* wc_bool_to_string
* prepare_set_attributes
* Use wc_get_products
* More feedback fixes
* Feedback fixes
* Implement CRUD in the legacy REST API
* Handle PR feedback
* [Product CRUD] Getter setter proxy methods (#12236)
* Started on variation changes
* Stock functions
* Variation class
* Bulk change ->id to get_id() to fix variation form display
* Missing status
* Fix add to cart
* Start on stored data save
* save variation
* Save_variations
* Variation edit panel
* Save variations code works.
* Remove stored data code and fix save
* Improve legacy class
* wc_bool_to_string
* prepare_set_attributes
* Use wc_get_products
* More feedback fixes
* get_prop implementation in abstract and data classes
* Implement set_prop
* Change handling
* Array key exists
* set_object_read
* Use get_the_terms() instead of wp_get_post_terms()
wp_get_post_terms() is a wrapper around wp_get_object_terms() which does not
use the object cache, and generates a database query every time it is used.
get_the_terms() however can use data from the object cache if present.
* [Product CRUD] Variable, variation, notices, and stock handling (#12277)
* No longer needed
* Remove old todos
* Use getters in admin list
* Related and upsells update for CRUD
* Fix notice in gallery
* Variable fixes and todos
* Context
* Price sync
* Revert variation attributes change
* Return parent data in view context
* Defer term counting
* wc_find_matching_product_variation
* Stock manage tweaks
* Stock fixes
* Correct id
* correct id
* Better sync
* Data logic setter fix
* feedback
* Prevent notices
* Handle image_id from parent
* Fix error
* Remove _wc_save_product_price
* Remove todo
* Fixed wrong variation URLs
* Fixed undefined $image_id in WC_Product_Variation::get_image_id()
* Allow wc_rest_prepare_date_response() handle timestamps
* Updated get methods on REST API for variations
* Use variations CRUD to save variations metadata
* [Product CRUD] Abstract todos (#12305)
* Get dimensions and weights, with soft deprecation
* Product attributes
* Ratings
* Fix read method
* Downloads
* Feedback
* Revert "[Product CRUD] Abstract todos (#12305)"
This reverts commit 9a6136fcf88fec16f97457b7c8a4388f7587bfa2.
* Remove deprecated get_variation_id()
* New default attributes method
* [Product CRUD] Product Datastore (#12317)
* Fix up tests in the product/* folder.
* Handle data store updates for grouped, variable, external, simple, and general data store updates for products.
* Variations & variable changes.
* Update -functions.php calls to use data store.
* Add an interface for the public product data store methods.
* Finished product factory tests
* Correctly delete in the api, fix up some comments, and implement an interface for the public variable methods.
* Fix up delete in all versions of the api
* Handle feedback
* Match protected decloration to parent
* Product crud abstract todos (#12316)
* Get dimensions and weights, with soft deprecation
* Product attributes
* Ratings
* Fix read method
* Downloads
* Feedback
* Fix up store
* Fixed method returning in write context
* Fix error in variation admin
* Check for parent value - fixes tax class
* Remove old/complete todos
* Allow set tax class as "parent"
* Removed duplicated sync
* Fixed wrong variation URLs
* Fixed undefined $image_id in WC_Product_Variation::get_image_id()
* Allow wc_rest_prepare_date_response() handle timestamps
* Updated get methods on REST API for variations
* Use variations CRUD to save variations metadata
* Remove deprecated get_variation_id()
* New default attributes method
* Fixed method returning in write context
* Allow set tax class as "parent"
* Removed duplicated sync
* Fixed coding standards
* TODO is not accurate.
* Should pass WC_Product instancies to WC_Comments methods (#12327)
* Use new method in abstract order class to prevent headers sent issue in tests
* Fixed variable description in REST API
* Updated how create initial product variation
* Fixed a few fatal errors and warnings in Products CRUD (#12329)
* Fixed a few fatal errors and warnings in Products CRUD
* Fixed sync functions
* Add variations CRUD to legacy API (#12331)
* Apply crud to variable products in legacy API v1
* New REST API do not need fallback for default attributes
* Apply variations CRUD to legacy API v2
* Legacy v2 - save default attributes
* Variations in legacy API v2 do not have descriptions
* Fixed legacy API v2 variations params
* Applied variations CRUD to legacy API v3
* Sync before save in legacy apis
* Punc
* Removed API todos
* Removed test
* Products endpoint tweaks (#12354)
* Var type already normalized on CRUD
* Let Product CRUD handle with validation, sanitization and conditional checks
* Set downloads using WC_Product_Download
* Stop try catch exceptions more than one time
* Handle WC_Data_Exception in legacy API
* Complete remove products when fails on creating
* On creating I mean!
* Already have a method to complete delete products
* Fixed standards using WP CodeSniffer
* get_the_terms() returns false when empty
* get_manage_stock returns boolean
@claudiosanches
* Merge conflict
* Variations API endpoint fixes
* Product CRUD improvements (#12359)
* args is not used any more - remove todo
* Added test for attributes
* wc_get_price_excluding_tax usage
* parent usage
* Fix rating counts
* Test fixes
* Cleanup after tests
* Make sure status transition code runs even during API calls, not just in admin.
* Default visibility
* Fix attribute setting in API
* Use get name instead of get title
* variation id usage
* Improved cross sell templates
* variation_data
* Grouped product sync
* Notices
* Sync is not needed in API
* Delete
* Rename interfaces
* Update counts in data store
2016-11-16 12:38:24 +00:00
|
|
|
if ( false === value ) {
|
|
|
|
this.removeAttr( attr );
|
|
|
|
} else {
|
|
|
|
this.attr( attr, value );
|
|
|
|
}
|
2015-10-28 13:04:28 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reset a default attribute for an element so it can be reset later
|
|
|
|
*/
|
|
|
|
$.fn.wc_reset_variation_attr = function( attr ) {
|
|
|
|
if ( undefined !== this.attr( 'data-o_' + attr ) ) {
|
|
|
|
this.attr( attr, this.attr( 'data-o_' + attr ) );
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-03-03 21:28:38 +00:00
|
|
|
/**
|
|
|
|
* Reset the slide position if the variation has a different image than the current one
|
|
|
|
*/
|
|
|
|
$.fn.wc_maybe_trigger_slide_position_reset = function( variation ) {
|
2017-03-13 16:58:39 +00:00
|
|
|
var $form = $( this ),
|
|
|
|
$product = $form.closest( '.product' ),
|
2017-03-14 10:55:25 +00:00
|
|
|
$product_gallery = $product.find( '.images' ),
|
2017-03-03 21:28:38 +00:00
|
|
|
reset_slide_position = false,
|
2018-06-05 12:22:35 +00:00
|
|
|
new_image_id = ( variation && variation.image_id ) ? variation.image_id : '';
|
2017-03-03 21:28:38 +00:00
|
|
|
|
|
|
|
if ( $form.attr( 'current-image' ) !== new_image_id ) {
|
|
|
|
reset_slide_position = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
$form.attr( 'current-image', new_image_id );
|
|
|
|
|
|
|
|
if ( reset_slide_position ) {
|
2017-06-06 13:16:56 +00:00
|
|
|
$product_gallery.trigger( 'woocommerce_gallery_reset_slide_position' );
|
2017-03-03 21:28:38 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2015-10-28 13:04:28 +00:00
|
|
|
/**
|
|
|
|
* Sets product images for the chosen variation
|
|
|
|
*/
|
|
|
|
$.fn.wc_variations_image_update = function( variation ) {
|
|
|
|
var $form = this,
|
2017-01-06 11:51:09 +00:00
|
|
|
$product = $form.closest( '.product' ),
|
2017-03-14 10:55:25 +00:00
|
|
|
$product_gallery = $product.find( '.images' ),
|
2017-11-03 14:40:06 +00:00
|
|
|
$gallery_nav = $product.find( '.flex-control-nav' ),
|
|
|
|
$gallery_img = $gallery_nav.find( 'li:eq(0) img' ),
|
2020-05-05 03:49:29 +00:00
|
|
|
$product_img_wrap = $product_gallery
|
|
|
|
.find( '.woocommerce-product-gallery__image, .woocommerce-product-gallery__image--placeholder' )
|
|
|
|
.eq( 0 ),
|
2017-03-16 11:56:00 +00:00
|
|
|
$product_img = $product_img_wrap.find( '.wp-post-image' ),
|
|
|
|
$product_link = $product_img_wrap.find( 'a' ).eq( 0 );
|
2015-10-28 13:04:28 +00:00
|
|
|
|
2016-12-15 13:04:50 +00:00
|
|
|
if ( variation && variation.image && variation.image.src && variation.image.src.length > 1 ) {
|
2018-05-31 13:57:01 +00:00
|
|
|
// See if the gallery has an image with the same original src as the image we want to switch to.
|
|
|
|
var galleryHasImage = $gallery_nav.find( 'li img[data-o_src="' + variation.image.gallery_thumbnail_src + '"]' ).length > 0;
|
2018-04-26 16:47:39 +00:00
|
|
|
|
2018-05-31 13:57:01 +00:00
|
|
|
// If the gallery has the image, reset the images. We'll scroll to the correct one.
|
|
|
|
if ( galleryHasImage ) {
|
|
|
|
$form.wc_variations_image_reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
// See if gallery has a matching image we can slide to.
|
|
|
|
var slideToImage = $gallery_nav.find( 'li img[src="' + variation.image.gallery_thumbnail_src + '"]' );
|
|
|
|
|
|
|
|
if ( slideToImage.length > 0 ) {
|
|
|
|
slideToImage.trigger( 'click' );
|
2017-06-06 13:16:56 +00:00
|
|
|
$form.attr( 'current-image', variation.image_id );
|
2018-06-05 12:22:35 +00:00
|
|
|
window.setTimeout( function() {
|
|
|
|
$( window ).trigger( 'resize' );
|
|
|
|
$product_gallery.trigger( 'woocommerce_gallery_init_zoom' );
|
|
|
|
}, 20 );
|
2017-06-06 13:16:56 +00:00
|
|
|
return;
|
|
|
|
}
|
2018-06-05 12:22:35 +00:00
|
|
|
|
|
|
|
$product_img.wc_set_variation_attr( 'src', variation.image.src );
|
|
|
|
$product_img.wc_set_variation_attr( 'height', variation.image.src_h );
|
|
|
|
$product_img.wc_set_variation_attr( 'width', variation.image.src_w );
|
|
|
|
$product_img.wc_set_variation_attr( 'srcset', variation.image.srcset );
|
|
|
|
$product_img.wc_set_variation_attr( 'sizes', variation.image.sizes );
|
|
|
|
$product_img.wc_set_variation_attr( 'title', variation.image.title );
|
2018-08-09 12:23:29 +00:00
|
|
|
$product_img.wc_set_variation_attr( 'data-caption', variation.image.caption );
|
2018-06-05 12:22:35 +00:00
|
|
|
$product_img.wc_set_variation_attr( 'alt', variation.image.alt );
|
|
|
|
$product_img.wc_set_variation_attr( 'data-src', variation.image.full_src );
|
|
|
|
$product_img.wc_set_variation_attr( 'data-large_image', variation.image.full_src );
|
|
|
|
$product_img.wc_set_variation_attr( 'data-large_image_width', variation.image.full_src_w );
|
|
|
|
$product_img.wc_set_variation_attr( 'data-large_image_height', variation.image.full_src_h );
|
|
|
|
$product_img_wrap.wc_set_variation_attr( 'data-thumb', variation.image.src );
|
|
|
|
$gallery_img.wc_set_variation_attr( 'src', variation.image.gallery_thumbnail_src );
|
|
|
|
$product_link.wc_set_variation_attr( 'href', variation.image.full_src );
|
2015-10-28 13:04:28 +00:00
|
|
|
} else {
|
2018-03-09 14:36:14 +00:00
|
|
|
$form.wc_variations_image_reset();
|
2015-10-28 13:04:28 +00:00
|
|
|
}
|
2017-02-16 11:00:09 +00:00
|
|
|
|
|
|
|
window.setTimeout( function() {
|
|
|
|
$( window ).trigger( 'resize' );
|
2017-05-24 16:15:22 +00:00
|
|
|
$form.wc_maybe_trigger_slide_position_reset( variation );
|
|
|
|
$product_gallery.trigger( 'woocommerce_gallery_init_zoom' );
|
|
|
|
}, 20 );
|
2015-10-28 13:04:28 +00:00
|
|
|
};
|
|
|
|
|
2018-03-09 14:36:14 +00:00
|
|
|
/**
|
|
|
|
* Reset main image to defaults.
|
|
|
|
*/
|
|
|
|
$.fn.wc_variations_image_reset = function() {
|
|
|
|
var $form = this,
|
|
|
|
$product = $form.closest( '.product' ),
|
|
|
|
$product_gallery = $product.find( '.images' ),
|
|
|
|
$gallery_nav = $product.find( '.flex-control-nav' ),
|
|
|
|
$gallery_img = $gallery_nav.find( 'li:eq(0) img' ),
|
2020-05-05 03:49:29 +00:00
|
|
|
$product_img_wrap = $product_gallery
|
|
|
|
.find( '.woocommerce-product-gallery__image, .woocommerce-product-gallery__image--placeholder' )
|
|
|
|
.eq( 0 ),
|
2018-03-09 14:36:14 +00:00
|
|
|
$product_img = $product_img_wrap.find( '.wp-post-image' ),
|
|
|
|
$product_link = $product_img_wrap.find( 'a' ).eq( 0 );
|
|
|
|
|
|
|
|
$product_img.wc_reset_variation_attr( 'src' );
|
|
|
|
$product_img.wc_reset_variation_attr( 'width' );
|
|
|
|
$product_img.wc_reset_variation_attr( 'height' );
|
|
|
|
$product_img.wc_reset_variation_attr( 'srcset' );
|
|
|
|
$product_img.wc_reset_variation_attr( 'sizes' );
|
|
|
|
$product_img.wc_reset_variation_attr( 'title' );
|
2018-08-09 12:23:29 +00:00
|
|
|
$product_img.wc_reset_variation_attr( 'data-caption' );
|
2018-03-09 14:36:14 +00:00
|
|
|
$product_img.wc_reset_variation_attr( 'alt' );
|
|
|
|
$product_img.wc_reset_variation_attr( 'data-src' );
|
|
|
|
$product_img.wc_reset_variation_attr( 'data-large_image' );
|
|
|
|
$product_img.wc_reset_variation_attr( 'data-large_image_width' );
|
|
|
|
$product_img.wc_reset_variation_attr( 'data-large_image_height' );
|
|
|
|
$product_img_wrap.wc_reset_variation_attr( 'data-thumb' );
|
|
|
|
$gallery_img.wc_reset_variation_attr( 'src' );
|
|
|
|
$product_link.wc_reset_variation_attr( 'href' );
|
|
|
|
};
|
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
$(function() {
|
2015-07-09 14:56:20 +00:00
|
|
|
if ( typeof wc_add_to_cart_variation_params !== 'undefined' ) {
|
2015-07-09 15:21:17 +00:00
|
|
|
$( '.variations_form' ).each( function() {
|
2016-12-15 12:58:41 +00:00
|
|
|
$( this ).wc_variation_form();
|
2015-07-09 15:21:17 +00:00
|
|
|
});
|
2015-07-10 05:13:30 +00:00
|
|
|
}
|
2014-02-22 04:08:08 +00:00
|
|
|
});
|
2013-02-07 10:12:08 +00:00
|
|
|
|
2016-12-15 12:58:41 +00:00
|
|
|
/**
|
|
|
|
* Matches inline variation objects to chosen attributes
|
|
|
|
* @deprecated 2.6.9
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
|
|
|
var wc_variation_form_matcher = {
|
|
|
|
find_matching_variations: function( product_variations, settings ) {
|
|
|
|
var matching = [];
|
|
|
|
for ( var i = 0; i < product_variations.length; i++ ) {
|
|
|
|
var variation = product_variations[i];
|
|
|
|
|
|
|
|
if ( wc_variation_form_matcher.variations_match( variation.attributes, settings ) ) {
|
|
|
|
matching.push( variation );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return matching;
|
|
|
|
},
|
|
|
|
variations_match: function( attrs1, attrs2 ) {
|
|
|
|
var match = true;
|
|
|
|
for ( var attr_name in attrs1 ) {
|
|
|
|
if ( attrs1.hasOwnProperty( attr_name ) ) {
|
|
|
|
var val1 = attrs1[ attr_name ];
|
|
|
|
var val2 = attrs2[ attr_name ];
|
|
|
|
if ( val1 !== undefined && val2 !== undefined && val1.length !== 0 && val2.length !== 0 && val1 !== val2 ) {
|
|
|
|
match = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return match;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-06-15 17:05:00 +00:00
|
|
|
/**
|
|
|
|
* Avoids using wp.template where possible in order to be CSP compliant.
|
|
|
|
* wp.template uses internally eval().
|
|
|
|
* @param {string} templateId
|
|
|
|
* @return {Function}
|
|
|
|
*/
|
|
|
|
var wp_template = function( templateId ) {
|
|
|
|
var html = document.getElementById( 'tmpl-' + templateId ).textContent;
|
|
|
|
var hard = false;
|
|
|
|
// any <# #> interpolate (evaluate).
|
|
|
|
hard = hard || /<#\s?data\./.test( html );
|
|
|
|
// any data that is NOT data.variation.
|
|
|
|
hard = hard || /{{{?\s?data\.(?!variation\.).+}}}?/.test( html );
|
|
|
|
// any data access deeper than 1 level e.g.
|
|
|
|
// data.variation.object.item
|
|
|
|
// data.variation.object['item']
|
|
|
|
// data.variation.array[0]
|
|
|
|
hard = hard || /{{{?\s?data\.variation\.[\w-]*[^\s}]/.test ( html );
|
|
|
|
if ( hard ) {
|
|
|
|
return wp.template( templateId );
|
|
|
|
}
|
|
|
|
return function template ( data ) {
|
|
|
|
var variation = data.variation || {};
|
|
|
|
return html.replace( /({{{?)\s?data\.variation\.([\w-]*)\s?(}}}?)/g, function( _, open, key, close ) {
|
|
|
|
// Error in the format, ignore.
|
|
|
|
if ( open.length !== close.length ) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
var replacement = variation[ key ] || '';
|
|
|
|
// {{{ }}} => interpolate (unescaped).
|
|
|
|
// {{ }} => interpolate (escaped).
|
|
|
|
// https://codex.wordpress.org/Javascript_Reference/wp.template
|
|
|
|
if ( open.length === 2 ) {
|
|
|
|
return window.escape( replacement );
|
|
|
|
}
|
|
|
|
return replacement;
|
|
|
|
});
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2015-07-14 09:28:49 +00:00
|
|
|
})( jQuery, window, document );
|