我必须在购物车中添加一些自定义金额的订单项。 商品保存,价格= 0,我的模块计算价格并将订单项添加到购物车/订单,但我不明白如何以编程方式设置价格。
我读过有关使用规则的内容,但我需要我的模块能够设置/更改价格,而无需调用规则。
我尝试使用实体包装器,我尝试更改使用commerce_product_line_item_new()创建的订单项,但什么都没有,当订单项进入购物车时始终具有原始产品价格(在我的情况下为0)。
如何以编程方式更改订单项价格?
到目前为止我的代码看起来像:
// For debugging, this function is called by hook_menu()
function mymodule_test($product_id)
{
global $user;
$user = user_load($user->uid);
$order = commerce_cart_order_load($user->uid);
$order_wrapper = entity_metadata_wrapper('commerce_order', $order);
$product = commerce_product_load($product_id);
$line_item = commerce_product_line_item_new(
$product,
1,
0,
array(
),
'cover'
);
$line_item_wrapper = entity_metadata_wrapper("commerce_line_item", $line_item);
$line_item_wrapper->commerce_unit_price->data = commerce_price_component_add(
$line_item_wrapper->commerce_unit_price->value(),
'base_price',
array(
'amount' => 1234,
'currency_code' => 'EUR',
'data' => array(),
),
TRUE
);
$insert_line_item = commerce_cart_product_add($user->uid, $line_item_wrapper->value(), FALSE);
return 'done';
}
奇怪的是,我尝试调整在commerce / modules / line_item / commerce_line_item.rules.inc中找到的commerce_line_item_unit_price_amount()代码,但是这个测试:
<?php
global $user;
$product = commerce_product_load(4); // my commerce product for test
$line_item = commerce_product_line_item_new(
$product,
1,
0,
array(
),
'cover' // I do have this line_items type
);
// manually set amount and component name
$amount = 1234;
$component_name = 'base_price'; // tryed with discount, nothing change
$wrapper = entity_metadata_wrapper('commerce_line_item', $line_item);
$unit_price = commerce_price_wrapper_value($wrapper, 'commerce_unit_price', TRUE);
// Calculate the updated amount and create a price array representing the
// difference between it and the current amount.
$current_amount = $unit_price['amount'];
$updated_amount = commerce_round(COMMERCE_ROUND_HALF_UP, $amount);
$difference = array(
'amount' => $updated_amount - $current_amount,
'currency_code' => $unit_price['currency_code'],
'data' => array(),
);
// Set the amount of the unit price and add the difference as a component.
$wrapper->commerce_unit_price->amount = $updated_amount;
$wrapper->commerce_unit_price->data = commerce_price_component_add(
$wrapper->commerce_unit_price->value(),
$component_name,
$difference,
TRUE
);
$insert_line_item = commerce_cart_product_add($user->uid, $line_item, FALSE);
?>
仍然失败,line_item进入购物车但是带有参考产品的原始价格。
有什么想法吗?
答案 0 :(得分:19)
对于那些不想使用规则并希望直接改变价格的人。这是我的解决方案:
// Alter the price in list and single product page.
function my_module_commerce_product_calculate_sell_price_line_item_alter($line_item){
$price = 100; //1 dollar
$line_item->commerce_unit_price[LANGUAGE_NONE]['0']['amount'] = $price;
}
// Alter the price in cart & order.
function my_module_commerce_cart_line_item_refresh($line_item, $order_wrapper){
$price = 100; //1 dollar
$line_item->commerce_unit_price[LANGUAGE_NONE]['0']['amount'] = $price;
// Alter the base_price component.
$line_item->commerce_unit_price[LANGUAGE_NONE]['0']['data']['components']['0']['price']['amount'] = $price;
}
答案 1 :(得分:6)
如果您要忽略以前保存到订单项的所有值,并重新计算新金额中的总金额,那么您要查找的函数是commerce_line_item_rebase_unit_price。
设置新的金额值,然后在那里运行您的订单项,保存订单项和订单:
$line_item_wrapper->commerce_unit_price->amount = 13;
commerce_line_item_rebase_unit_price($line_item_wrapper->value());
commerce_line_item_save($line_item_wrapper->value());
答案 2 :(得分:4)
我今天整天都在努力解决这个问题,最终找到了改变订单项价格的正确途径。问题是,即使您成功将订单项价格更改为自定义值,在下一页上刷新购物车也会重置订单项以匹配原始产品价格。有关详细信息,请查看commerce_cart_order_refresh()
函数。每次在页面上加载订单/购物车时都会执行此功能,并且无法绕过它。
事实证明,更改订单项价格的正确方法是使用规则或实施hook_commerce_cart_line_item_refresh()
功能。无论哪种方式,Drupal Commerce都需要能够在每次加载购物车/订单时应用更改逻辑。
我最终在订单项中创建了一个自定义字段,其中存储了我想要的自定义价格值。然后,我使用定价规则在刷新购物车时将自定义价格值复制到产品价格值。
以下博文非常有助于解决这个问题。它显示了如何向订单项类型添加自定义字段以及如何设置定价规则以将自定义金额复制到单价。
http://commerceguys.com/blog/using-custom-line-items-provide-donation-feature-drupal-commerce
答案 3 :(得分:1)
最近我必须在Commerce中实施捐赠表单,但Commerce Express Checkout模块不处理自定义订单项。既然是捐款而且所有人(谁试图搞砸房子?),我认为将捐赠金额作为Express Checkout模块提供的URL中的第3个参数传递是合适的。以下是我如何破解模块:
我在路由器中添加了一个新条目:
$items['commerce-express-checkout/%/%/%'] = array(
'title' => 'Express Checkout w/ extra argument',
// 'page callback' => 'commerce_express_checkout_create_order',
'page callback' => 'commerce_express_checkout_create_order_extra',
'page arguments' => array(1, 2, 3),
'access arguments' => array('access checkout'),
'type' => MENU_CALLBACK,
);
我复制并调整了默认回调并将'_extra'添加到其中。请注意,“data”属性似乎是一个静态变量存储,适用于此类情况并保持行项目的生命周期。
function commerce_express_checkout_create_order_extra($product_id, $token, $amount) {
if (drupal_hmac_base64($product_id, drupal_get_private_key().drupal_get_hash_salt()) == $token && is_numeric($amount)) {
global $user;
$product = commerce_product_load($product_id);
$product->commerce_price['und'][0]['amount'] = (int)$amount;
$order = ($user->uid) ? commerce_order_new($user->uid, 'checkout_checkout') : commerce_cart_order_new();
commerce_order_save($order);
$price = array('amount' => commerce_round(COMMERCE_ROUND_HALF_UP, $amount), 'currency_code' => commerce_default_currency());
$line_item = commerce_product_line_item_new($product, 1, $order->order_id);
$line_item->data = array('und' => array('0' => $price));
commerce_line_item_save($line_item);
$order_wrapper = entity_metadata_wrapper('commerce_order', $order);
$order_wrapper->commerce_line_items[] = $line_item;
$order->data['type'] = 'commerce_express_checkout_order';
commerce_order_save($order);
drupal_goto('checkout/' . $order->order_id);
return "";
}
return "";
}
由于学习曲线并且不知道要使用什么功能,这部分最终变得最棘手:
/**
* Implements hook_commerce_cart_line_item_refresh().
*/
function commerce_express_checkout_commerce_cart_line_item_refresh($line_item, $order_wrapper) {
if ($line_item->commerce_product['und'][0]['line_item_label'] == 'DONATE' || $line_item->commerce_product['und'][0]['product_id'] == '11') {
$price = array('amount' => commerce_round(COMMERCE_ROUND_HALF_UP, $line_item->data['und'][0]['amount']), 'currency_code' => commerce_default_currency());
$line_item->commerce_unit_price = array('und' => array('0' => $price));
$line_item_wrapper = entity_metadata_wrapper('commerce_line_item', $line_item);
$line_item_wrapper->commerce_unit_price->data = commerce_price_component_add(
$line_item_wrapper->commerce_unit_price->value(), 'base_price', $price, TRUE
);
}
}
每次修改购物车时,它都会刷新并尝试将购物车中的产品设置为其代码内原型。这对我来说似乎效率很低,但我可能会遗漏一些东西。
答案 4 :(得分:0)
这篇文章指出了我使用hook_commerce_cart_line_item_refersh()
以编程方式更改drupal商业订单项的正确方向。但是,这里的一些答案要么完全错误,要么非常低效和草率。
这将是改变Drupal Commerce中订单项类型的正确工作解决方案:
/*
* implements hook_commerce_cart_line_item_refresh()
*
*/
function MYMODULE_commerce_cart_line_item_refresh($line_item, $order_wrapper){
$line_wrapper = entity_metadata_wrapper('commerce_line_item', $line_item);
$new_price = 100; //I use a function to calculate the value of $new_price
if(!empty($new_price)){
$line_wrapper->commerce_unit_price->amount->set($new_price);
$line_wrapper->save();
}
}