WooCommerce: dodatkowa lokalizacja zapasów magazynowych
Opublikowany: 2022-12-07Wtyczka WooCommerce pozwala zarządzać zapasami dla każdego produktu, ale masz tylko jedno pole ilości zapasów!
Co zrobić, jeśli masz dwa magazyny i jako administrator sklepu musisz zarządzać zapasami dla każdej lokalizacji ? Poza tym, co zrobić, jeśli pozycja jest niedostępna w lokalizacji 1, ale jest dostępna w lokalizacji 2, a zatem klient musi mieć możliwość jej zakupu?
To niesamowite obejście doda drugi numer wejściowy w ustawieniach produktu, ponownie zdefiniuje ilość i status zapasów na interfejsie, sumując zapas 1 + zapas 2, a na koniec zmniejszy zapas 1, aż spadnie do 0, po czym zmniejszy zapas 2.
To domyślne zachowanie można oczywiście zmienić, np. możliwe jest zdefiniowanie miejsca, w którym zapas jest zmniejszany (o odległość?) za pomocą dodatkowego kodu. Ponadto można napisać dodatkowy kod, aby był kompatybilny z produktami zmiennymi lub niestandardowymi typami produktów, a także aby działał ze zwrotami. Tak czy inaczej, ciesz się!
PHP Snippet: Drugie zarządzanie lokalizacją zapasów
/**
* @snippet Second Stock Location @ WooCommerce Edit Product
* @how-to Get CustomizeWoo.com FREE
* @author Rodolfo Melogli
* @compatible WooCommerce 7
* @donate $9 https://businessbloomer.com/bloomer-armada/
*/
add_action( 'woocommerce_product_options_stock', 'bbloomer_additional_stock_location' );
function bbloomer_additional_stock_location() {
global $product_object;
echo '<div class="show_if_simple show_if_variable">';
woocommerce_wp_text_input(
array(
'id' => '_stock2',
'value' => get_post_meta( $product_object->get_id(), '_stock2', true ),
'label' => '2nd Stock Location',
'data_type' => 'stock',
)
);
echo '</div>';
}
add_action( 'save_post_product', 'bbloomer_save_additional_stock' );
function bbloomer_save_additional_stock( $product_id ) {
global $typenow;
if ( 'product' === $typenow ) {
if ( defined( 'DOING_AUTOSAVE' ) && DOING_AUTOSAVE ) return;
if ( isset( $_POST['_stock2'] ) ) {
update_post_meta( $product_id, '_stock2', $_POST['_stock2'] );
}
}
}
add_filter( 'woocommerce_product_get_stock_quantity' , 'bbloomer_get_overall_stock_quantity', 9999, 2 );
function bbloomer_get_overall_stock_quantity( $value, $product ) {
$value = (int) $value + (int) get_post_meta( $product->get_id(), '_stock2', true );
return $value;
}
add_filter( 'woocommerce_product_get_stock_status' , 'bbloomer_get_overall_stock_status', 9999, 2 );
function bbloomer_get_overall_stock_status( $status, $product ) {
if ( ! $product->managing_stock() ) return $status;
$stock = (int) $product->get_stock_quantity() + (int) get_post_meta( $product->get_id(), '_stock2', true );
$status = $stock && ( $stock > 0 ) ? 'instock' : 'outofstock';
return $status;
}
add_filter( 'woocommerce_payment_complete_reduce_order_stock', 'bbloomer_maybe_reduce_second_stock', 9999, 2 );
function bbloomer_maybe_reduce_second_stock( $reduce, $order_id ) {
$order = wc_get_order( $order_id );
$atleastastock2change = false;
foreach ( $order->get_items() as $item ) {
if ( ! $item->is_type( 'line_item' ) ) {
continue;
}
$product = $item->get_product();
$item_stock_reduced = $item->get_meta( '_reduced_stock', true );
if ( $item_stock_reduced || ! $product || ! $product->managing_stock() ) {
continue;
}
$qty = apply_filters( 'woocommerce_order_item_quantity', $item->get_quantity(), $order, $item );
$stock1 = (int) get_post_meta( $product->get_id(), '_stock', true );
if ( $qty <= $stock1 ) continue;
$atleastastock2change = true;
}
if ( ! $atleastastock2change ) return $reduce;
foreach ( $order->get_items() as $item ) {
if ( ! $item->is_type( 'line_item' ) ) {
continue;
}
$product = $item->get_product();
$item_stock_reduced = $item->get_meta( '_reduced_stock', true );
if ( $item_stock_reduced || ! $product || ! $product->managing_stock() ) {
continue;
}
$item_name = $product->get_formatted_name();
$qty = apply_filters( 'woocommerce_order_item_quantity', $item->get_quantity(), $order, $item );
$stock1 = (int) get_post_meta( $product->get_id(), '_stock', true );
$stock2 = (int) get_post_meta( $product->get_id(), '_stock2', true );
if ( $qty <= $stock1 ) {
wc_update_product_stock( $product, $qty, 'decrease' );
$order->add_order_note( sprintf( 'Reduced stock for item "%s"; Stock 1: "%s" to "%s".', $item_name, $stock1, $stock1 - $qty ) );
} else {
$newstock2 = $stock2 - ( $qty - $stock1 );
wc_update_product_stock( $product, $stock1, 'decrease' );
update_post_meta( $product->get_id(), '_stock2', $newstock2 );
$item->add_meta_data( '_reduced_stock', $qty, true );
$item->save();
$order->add_order_note( sprintf( 'Reduced stock for item "%s"; Stock 1: "%s" to "0" and Stock 2: "%s" to "%s".', $item_name, $stock1, $stock2, $newstock2 ) );
}
}
$order->get_data_store()->set_stock_reduced( $order_id, true );
return false;
}
W prostym angielskim:
- bbloomer_additional_stock_location pokazuje drugie pole ilości zapasów
- bbloomer_save_additional_stock zapisuje niestandardową ilość zapasów
- bbloomer_get_overall_stock_quantity ustawia zapasy produktów na zapasy 1 + zapasy 2
- bbloomer_get_overall_stock_status ustawia stan zapasów produktu na podstawie zapasów 1 + zapasów 2
- bbloomer_maybe_reduce_second_stock zmniejsza stan z magazynu 1, a następnie z magazynu 2 w przypadku, gdy zamówiona ilość jest większa niż stan 1, w przeciwnym razie pozwala WooCommerce na domyślne zmniejszenie zapasów