Использование поля номера ACF в качестве цены для пользовательского типа продукта WooCommerce

Я видел в Интернете, что вы можете добавить все пользовательские типы сообщений в корзину WooCommerce, если в CPT есть поле цены. Единственная проблема заключается в том, что вы должны сообщить WooCommerce, какое поле CPT содержит цену.

После слов вы можете легко создать URL-адрес для добавления в корзину, например: https://yourdomain.com/?add-to-cart=XXX (XXX — это идентификатор публикации пользовательского типа публикации)

Почему это удобно?
У меня есть онлайн-меню (просто для того, чтобы сообщить моему гостю, что мы подаем), но из-за коронавируса мы вынуждены закрыть наши двери, поэтому я хочу, чтобы эти блюда заказывали онлайн.

Код должен быть примерно таким, но CPT не добавляется в корзину:

add_filter('woocommerce_get_price', 'yl_get_dish_price', 20,2);
function yl_get_dish_price($price,$post) {
    if ($post->post->post_type === 'dish') {
        $price = get_field('price', $post->ID);
    }
    return $price;
}

ОБНОВЛЯТЬ

Я взял этот ответ отсюда https://stackoverflow.com/a/60320662/10291365

class YL_Dish_Product extends WC_Product  {

    protected $post_type = 'dish';

    public function get_type() {
        return 'dish';
    }

    public function __construct( $product = 0 ) {
        $this->supports[]   = 'ajax_add_to_cart';

        parent::__construct( $product );


    }
    // maybe overwrite other functions from WC_Product

}

class YL_Data_Store_CPT extends WC_Product_Data_Store_CPT {

    public function read( &$product ) { // this is required
        $product->set_defaults();
        $post_object = get_post( $product->get_id() );

        if ( ! $product->get_id() || ! $post_object || 'dish' !== $post_object->post_type ) {

            throw new Exception( __( 'Invalid product.', 'woocommerce' ) );
        }

        $product->set_props(
            array(
                'name'              => $post_object->post_title,
                'slug'              => $post_object->post_name,
                'date_created'      => 0 < $post_object->post_date_gmt ? wc_string_to_timestamp( $post_object->post_date_gmt ) : null,
                'date_modified'     => 0 < $post_object->post_modified_gmt ? wc_string_to_timestamp( $post_object->post_modified_gmt ) : null,
                'status'            => $post_object->post_status,
                'description'       => $post_object->post_content,
                'short_description' => $post_object->post_excerpt,
                'parent_id'         => $post_object->post_parent,
                'menu_order'        => $post_object->menu_order,
                'reviews_allowed'   => 'open' === $post_object->comment_status,
            )
        );

        $this->read_attributes( $product );
        $this->read_downloads( $product );
        $this->read_visibility( $product );
        $this->read_product_data( $product );
        $this->read_extra_data( $product );
        $product->set_object_read( true );
    }

    // maybe overwrite other functions from WC_Product_Data_Store_CPT

}


class YL_WC_Order_Item_Product extends WC_Order_Item_Product {
    public function set_product_id( $value ) {
        if ( $value > 0 && 'dish' !== get_post_type( absint( $value ) ) ) {
            $this->error( 'order_item_product_invalid_product_id', __( 'Invalid product ID', 'woocommerce' ) );
        }
        $this->set_prop( 'product_id', absint( $value ) );
    }

}




function YL_woocommerce_data_stores( $stores ) {
    // the search is made for product-$post_type so note the required 'product-' in key name
    $stores['product-dish'] = 'YL_Data_Store_CPT';
    return $stores;
}
add_filter( 'woocommerce_data_stores', 'YL_woocommerce_data_stores' , 11, 1 );


function YL_woo_product_class( $class_name ,  $product_type ,  $product_id ) {
    if ($product_type == 'dish')
        $class_name = 'YL_Dish_Product';
    return $class_name; 
}
add_filter('woocommerce_product_class','YL_woo_product_class',25,3 );



function my_woocommerce_product_get_price( $price, $product ) {

    if ($product->get_type() == 'dish' ) {
        $price = 10;  // or get price how ever you see fit     
    }
    return $price;
}
add_filter('woocommerce_get_price','my_woocommerce_product_get_price',20,2);
add_filter('woocommerce_product_get_price', 'my_woocommerce_product_get_price', 10, 2 );



// required function for allowing posty_type to be added; maybe not the best but it works
function YL_woo_product_type($false,$product_id) { 
    if ($false === false) { // don't know why, but this is how woo does it
        global $post;
        // maybe redo it someday?!
        if (is_object($post) && !empty($post)) { // post is set
            if ($post->post_type == 'dish' && $post->ID == $product_id) 
                return 'dish';
            else {
                $product = get_post( $product_id );
                if (is_object($product) && !is_wp_error($product)) { // post not set but it's a dish
                    if ($product->post_type == 'dish') 
                        return 'dish';
                } // end if 
            }    

        } else if (wp_doing_ajax()) { // has post set (usefull when adding using ajax)
            $product_post = get_post( $product_id );
            if ($product_post->post_type == 'dish') 
                return 'dish';
        } else { 
            $product = get_post( $product_id );
            if (is_object($product) && !is_wp_error($product)) { // post not set but it's a dish
                if ($product->post_type == 'dish') 
                    return 'dish';
            } // end if 

        } // end if  // end if 



    } // end if 
    return false;
}
add_filter('woocommerce_product_type_query','YL_woo_product_type',12,2 );

function YL_woocommerce_checkout_create_order_line_item_object($item, $cart_item_key, $values, $order) {

    $product                    = $values['data'];
    if ($product->get_type() == 'dish') {
        return new YL_WC_Order_Item_Product();
    } // end if 
    return $item ;
}   
add_filter( 'woocommerce_checkout_create_order_line_item_object', 'YL_woocommerce_checkout_create_order_line_item_object', 20, 4 );

function cod_woocommerce_checkout_create_order_line_item($item,$cart_item_key,$values,$order) {
    if ($values['data']->get_type() == 'dish') {
        $item->update_meta_data( '_dish', 'yes' ); // add a way to recognize custom post type in ordered items
        return;
    } // end if 

}
add_action( 'woocommerce_checkout_create_order_line_item', 'cod_woocommerce_checkout_create_order_line_item', 20, 4 );

function YL_woocommerce_get_order_item_classname($classname, $item_type, $id) {
    global $wpdb;
    $is_IA = $wpdb->get_var("SELECT meta_value FROM {$wpdb->prefix}woocommerce_order_itemmeta WHERE order_item_id = {$id} AND meta_key = '_dish'");


    if ('yes' === $is_IA) { // load the new class if the item is our custom post
        $classname = 'YL_WC_Order_Item_Product';
    } // end if 
    return $classname;
}
add_filter( 'woocommerce_get_order_item_classname', 'YL_woocommerce_get_order_item_classname', 20, 3 );

Приведенный выше код добавляет CPT в вашу корзину (ОТЛИЧНО!!), но цена всегда установлена ​​​​на 10,00.

Таким образом, приведенный ниже код не дает правильной цены :(

add_filter('woocommerce_get_price', 'yl_get_dish_price', 20,2);
function yl_get_dish_price($price,$post) {
    if ($post->post->post_type === 'dish') {
        $price = get_field('price', $post->ID);
    }
    return $price;
}

Есть идеи?

Этот stackoverflow.com/questions/40732683/… может ответить на ваш вопрос.

Zoli Szabó 15.12.2020 11:23

Конечно $post->post->post_type правильно? Похоже, там лишний ->post.

CBroe 15.12.2020 11:29

Я думаю, ты прав. Я удалил часть 1 -> post без везения :(

Yorlinq 15.12.2020 12:00
Стоит ли изучать PHP в 2023-2024 годах?
Стоит ли изучать PHP в 2023-2024 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Symfony Station Communiqué - 7 июля 2023 г
Symfony Station Communiqué - 7 июля 2023 г
Это коммюнике первоначально появилось на Symfony Station .
Оживление вашего приложения Laravel: Понимание режима обслуживания
Оживление вашего приложения Laravel: Понимание режима обслуживания
Здравствуйте, разработчики! В сегодняшней статье мы рассмотрим важный аспект управления приложениями, который часто упускается из виду в суете...
Установка и настройка Nginx и PHP на Ubuntu-сервере
Установка и настройка Nginx и PHP на Ubuntu-сервере
В этот раз я сделаю руководство по установке и настройке nginx и php на Ubuntu OS.
Коллекции в Laravel более простым способом
Коллекции в Laravel более простым способом
Привет, читатели, сегодня мы узнаем о коллекциях. В Laravel коллекции - это способ манипулировать массивами и играть с массивами данных. Благодаря...
Как установить PHP на Mac
Как установить PHP на Mac
PHP - это популярный язык программирования, который используется для разработки веб-приложений. Если вы используете Mac и хотите разрабатывать...
2
3
1 487
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

Начиная с WooCommerce 3, хук woocommerce_get_price устарел и устарел… Он заменен следующим составным хуком:

add_filter( 'woocommerce_product_get_price', 'yl_get_dish_price', 20, 2 );
add_filter( 'woocommerce_product_get_regular_price', 'yl_get_dish_price', 20, 2 );
function yl_get_dish_price( $price, $product ) {
    if ( $product->is_type('dish') ) {
        $price = get_field( 'price', $product->get_id() );
    }
    return $price;
}

Он может и должен работать лучше.

Ты потрясешь мой мир!! Используйте вышеприведенную часть с большим кодом моего обновления, и вы получите пикник :) Большое спасибо!

Yorlinq 15.12.2020 14:43

Другие вопросы по теме