Woocommerce: Which hook to replace deprecated “woocommerce_add_order_item_meta”

Need to add custom meta to order items. Googled it and most articles says to use “woocommerce_add_order_item_meta” hook. This hook is deprecated in the newest version 2.3.7. Someone, please tell me which hook to use instead.

http://docs.woothemes.com/wc-apidocs/function-woocommerce_add_order_item_meta.html

Related posts

Leave a Reply

8 comments

  1. 2017/2018 THE RIGHT WAY (Using new CRUD setters and Getters methods)

    Related: Replace woocommerce_add_order_item_meta hook in Woocommerce 3.4

    Since woocommerce 3 that has improved many things making drastic changes, the action hook woocommerce_add_order_item_meta still work perfectly even in woocommerce version 3.3+.

    This hook is enabled by WC_Checkout class methods and related functions in the checkout process and not in WC_Order Class where cart data is not anymore available.

    Now as Woocommmerce 3 has introduced new CRUD setters and getters methods, the similar replacement hook to be used is woocommerce_checkout_create_order_line_item that has similar useful arguments as cart data.

    The woocommerce_new_order_item is really NOT convenient as cart data is not accessible.

    Let see how to work with woocommerce_checkout_create_order_line_item. It has 4 available arguments:

    • $item is an instance of WC_Order_Item_Product new introduced Class
    • $cart_item_key is the cart item unique hash key
    • $values is the cart item
    • $order an instance of the WC_Order object (This is a very useful additional argument in some specific cases)

    In this hook we will replace the old working functions wc_add_order_item_meta() by the new WC_Data update_meta_data() method to be used with $item argument.

    Example:

    ## --- New way --- ##
    add_action( 'woocommerce_checkout_create_order_line_item', 'custom_checkout_create_order_line_item', 20, 4 );
    function custom_checkout_create_order_line_item( $item, $cart_item_key, $values, $order ) {
        // Get a product custom field value
        $custom_field_value = get_post_meta( $item->get_product_id(), '_meta_key', true );
        // Update order item meta
        if ( ! empty( $custom_field_value ) ){
            $item->update_meta_data( 'meta_key1', $custom_field_value );
        }
        // … … Or … …
    
        // Get cart item custom data and update order item meta
        if( isset( $values['custom_data'] ) ) {
            $item->update_meta_data( 'meta_key2', $values['custom_data'] );
        }
    }
    

    Finally we can do the same with old way using woocommerce_add_order_item_meta hook as it has nearly the same useful arguments:

    ## --- Old way --- ##
    add_action( 'woocommerce_add_order_item_meta', 'custom_add_order_item_meta', 20, 3 );
    function custom_add_order_item_meta( $item_id, $values, $cart_item_key ) {
        // Get a product custom field value
        $custom_field_value = get_post_meta( $values['data']->get_id(), '_meta_key', true );
        // Update order item meta
        if ( ! empty( $custom_field_value ) ){
            wc_add_order_item_meta( $item_id, 'meta_key1', $custom_field_value );
        }
        // … … Or … …
    
        // Get cart item custom data and update order item meta
        if( isset( $values['custom_data'] ) ) {
            wc_add_order_item_meta( $item_id, 'meta_key2', $values['custom_data'] );
        }
    }
    

    Conclusion: woocommerce_checkout_create_order_line_item is the right replacement hook to be used with WooCommerce 3+ and that new CRUD setters and getters methods.

  2. If you look at wc-deprecated-functions.php you will see

    /**
     * @deprecated
     */
    function woocommerce_add_order_item_meta( $item_id, $meta_key, $meta_value, $unique = false ) {
        return wc_add_order_item_meta( $item_id, $meta_key, $meta_value, $unique );
    }
    

    Basically, the function was renamed to wc_add_order_item_meta(), so if you need the function then use that. The action hook was not renamed and remains in class-wc-checkout.php as:

    // Allow plugins to add order item meta
    do_action( 'woocommerce_add_order_item_meta', $item_id, $values, $cart_item_key );
    
  3. It seems that the hook is now also deprecated as of version 3.0.4.
    I’m getting this notification:

    The The "woocommerce_add_order_item_meta" hook uses out of date data structures and function is deprecated since version 3.0.4. Replace with woocommerce_new_order_item.
    

    I have replaced the action name ‘woocommerce_add_order_item_meta’ with ‘woocommerce_new_order_item’ in an add_action statement in an offending plugin, and the deprecation notification disappears, The problem is that some parameters now appear inside a legacy_values array. I use the plugin YITH WooCommerce Product Add Ons, and the product meta data that should be attached to an order is not picked up by the plugin and therefore not stored with the order. So until this is fixed in the plugin you have to live with the deprecation notification.

  4. I know this has been answered and there is an accepted reply already. I just wanted to give another way to handle this without actually getting a deprecated message (see reference);

    add_action('woocommerce_new_order_item', 'saveMetaData', 10, 3); // or use just 2 instead of 3; if you don't need order id
    
    /**
     * Add meta to order item
     *
     * @param int $itemId
     * @param WC_Order_Item_Product|WC_Order_Item_Shipping $item
     * @param int @orderId
    */
    function saveMetaData($itemId, $item, $orderId)
    {
        if (!isItemValid($item))
        {
            return;
        }
    
        wc_add_order_item_meta($itemId, 'my_custom_data', $item->legacy_values['my_custom_data']);
    }
    
    /**
     * @param WC_Order_Item_Product|WC_Order_Item_Shipping $item
     *
     * @return bool
    */
    function isItemValid($item)
    {
        return (
            $item instanceof WC_Order_Item_Product &&
            isset($item->legacy_values) &&
            isset($item->legacy_values['my_custom_data']) &&
            !empty($item->legacy_values['my_custom_data'])
        );
    }
    
  5. I wanted to add on to Ilgıt Yıldırım’s answer: in my case, my custom values did not exist in the item->legacy_values array. To fix this, I used the woocommerce_checkout_create_order_line_item hook to add the custom values to the item prior to calling the woocommerce_new_order_item hook. Here is an example of that:

    add_action( ‘woocommerce_checkout_create_order_line_item’, ‘save_values_in_item’, PHP_INT_MAX, 4 );

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

                    $item->myCustomValues = $values;
    

    }

    //THEN call the new hook:
    add_action( ‘woocommerce_new_order_item’, ‘add_product_input_fields_to_order_item_meta_wc3’, PHP_INT_MAX, 3 );

    function add_product_input_fields_to_order_item_meta_wc3( $item_id, $item, $order_id ) {

                if ( isset( $item->myCustomValues ) ) 
                {
                      //iterate through array and place desired values into the meta data using the wc_add_order_item_meta function
                }
    

    }