| Server IP : 213.186.33.4 / Your IP : 216.73.216.193 Web Server : Apache System : Linux webm006.cluster103.gra.hosting.ovh.net 5.15.206-ovh-vps-grsec-zfs-classid #1 SMP Fri May 15 02:41:25 UTC 2026 x86_64 User : awebpaca ( 35430) PHP Version : 8.5.0 Disable Function : _dyuweyrj4,_dyuweyrj4r,dl MySQL : OFF | cURL : ON | WGET : ON | Perl : ON | Python : ON | Sudo : OFF | Pkexec : OFF Directory : /home/awebpaca/boutiques/app/code/core/Mage/Sales/Model/ |
Upload File : |
<?php
/**
* Magento
*
* NOTICE OF LICENSE
*
* This source file is subject to the Open Software License (OSL 3.0)
* that is bundled with this package in the file LICENSE.txt.
* It is also available through the world-wide-web at this URL:
* http://opensource.org/licenses/osl-3.0.php
* If you did not receive a copy of the license and are unable to
* obtain it through the world-wide-web, please send an email
* to license@magentocommerce.com so we can send you a copy immediately.
*
* DISCLAIMER
*
* Do not edit or add to this file if you wish to upgrade Magento to newer
* versions in the future. If you wish to customize Magento for your
* needs please refer to http://www.magentocommerce.com for more information.
*
* @category Mage
* @package Mage_Sales
* @copyright Copyright (c) 2010 Magento Inc. (http://www.magentocommerce.com)
* @license http://opensource.org/licenses/osl-3.0.php Open Software License (OSL 3.0)
*/
/**
* Quote model
*
* Supported events:
* sales_quote_load_after
* sales_quote_save_before
* sales_quote_save_after
* sales_quote_delete_before
* sales_quote_delete_after
*
* @author Magento Core Team <core@magentocommerce.com>
*/
class Mage_Sales_Model_Quote extends Mage_Core_Model_Abstract
{
protected $_eventPrefix = 'sales_quote';
protected $_eventObject = 'quote';
/**
* Quote customer model object
*
* @var Mage_Customer_Model_Customer
*/
protected $_customer;
/**
* Quote addresses collection
*
* @var Mage_Eav_Model_Entity_Collection_Abstract
*/
protected $_addresses = null;
/**
* Quote items collection
*
* @var Mage_Eav_Model_Entity_Collection_Abstract
*/
protected $_items = null;
/**
* Quote payments
*
* @var Mage_Eav_Model_Entity_Collection_Abstract
*/
protected $_payments = null;
/**
* Init resource model
*/
protected function _construct()
{
$this->_init('sales/quote');
}
/**
* Get quote store identifier
*
* @return int
*/
public function getStoreId()
{
if (!$this->hasStoreId()) {
return Mage::app()->getStore()->getId();
}
return $this->_getData('store_id');
}
/**
* Get quote store model object
*
* @return Mage_Core_Model_Store
*/
public function getStore()
{
return Mage::app()->getStore($this->getStoreId());
}
/**
* Declare quote store model
*
* @param Mage_Core_Model_Store $store
* @return Mage_Sales_Model_Quote
*/
public function setStore(Mage_Core_Model_Store $store)
{
$this->setStoreId($store->getId());
return $this;
}
/**
* Get all available store ids for quote
*
* @return array
*/
public function getSharedStoreIds()
{
$ids = $this->_getData('shared_store_ids');
if (is_null($ids) || !is_array($ids)) {
if ($website = $this->getWebsite()) {
return $website->getStoreIds();
}
return $this->getStore()->getWebsite()->getStoreIds();
}
return $ids;
}
/**
* Prepare data before save
*
* @return Mage_Sales_Model_Quote
*/
protected function _beforeSave()
{
/**
* Currency logic
*
* global - currency which is set for default in backend
* base - currency which is set for current website. all attributes that
* have 'base_' prefix saved in this currency
* store - all the time it was currency of website and all attributes
* with 'base_' were saved in this currency. From now on it is
* deprecated and will be duplication of base currency code.
* quote/order - currency which was selected by customer or configured by
* admin for current store. currency in which customer sees
* price thought all checkout.
*
* Rates:
* store_to_base & store_to_quote/store_to_order - are deprecated
* base_to_global & base_to_quote/base_to_order - must be used instead
*/
$globalCurrencyCode = Mage::app()->getBaseCurrencyCode();
$baseCurrency = $this->getStore()->getBaseCurrency();
if ($this->hasForcedCurrency()){
$quoteCurrency = $this->getForcedCurrency();
} else {
$quoteCurrency = $this->getStore()->getCurrentCurrency();
}
$this->setGlobalCurrencyCode($globalCurrencyCode);
$this->setBaseCurrencyCode($baseCurrency->getCode());
$this->setStoreCurrencyCode($baseCurrency->getCode());
$this->setQuoteCurrencyCode($quoteCurrency->getCode());
//deprecated, read above
$this->setStoreToBaseRate($baseCurrency->getRate($globalCurrencyCode));
$this->setStoreToQuoteRate($baseCurrency->getRate($quoteCurrency));
$this->setBaseToGlobalRate($baseCurrency->getRate($globalCurrencyCode));
$this->setBaseToQuoteRate($baseCurrency->getRate($quoteCurrency));
if (!$this->hasChangedFlag() || $this->getChangedFlag() == true) {
$this->setIsChanged(1);
} else {
$this->setIsChanged(0);
}
if ($this->_customer) {
$this->setCustomerId($this->_customer->getId());
}
parent::_beforeSave();
}
/**
* Save related items
*
* @return Mage_Sales_Model_Quote
*/
protected function _afterSave()
{
parent::_afterSave();
if (null !== $this->_addresses) {
$this->getAddressesCollection()->save();
}
if (null !== $this->_items) {
$this->getItemsCollection()->save();
}
if (null !== $this->_payments) {
$this->getPaymentsCollection()->save();
}
return $this;
}
/**
* Loading quote data by customer
*
* @return Mage_Sales_Model_Quote
*/
public function loadByCustomer($customer)
{
if ($customer instanceof Mage_Customer_Model_Customer) {
$customerId = $customer->getId();
}
else {
$customerId = (int) $customer;
}
$this->_getResource()->loadByCustomerId($this, $customerId);
$this->_afterLoad();
return $this;
}
/**
* Loading only active quote
*
* @param int $quoteId
* @return Mage_Sales_Model_Quote
*/
public function loadActive($quoteId)
{
$this->_getResource()->loadActive($this, $quoteId);
$this->_afterLoad();
return $this;
}
/**
* Assign customer model object data to quote
*
* @param Mage_Customer_Model_Customer $customer
* @return Mage_Sales_Model_Quote
*/
public function assignCustomer(Mage_Customer_Model_Customer $customer)
{
if ($customer->getId()) {
$this->setCustomer($customer);
$defaultBillingAddress = $customer->getDefaultBillingAddress();
if ($defaultBillingAddress && $defaultBillingAddress->getId()) {
$billingAddress = Mage::getModel('sales/quote_address')
->importCustomerAddress($defaultBillingAddress);
$this->setBillingAddress($billingAddress);
}
$defaultShippingAddress= $customer->getDefaultShippingAddress();
if ($defaultShippingAddress && $defaultShippingAddress->getId()) {
$shippingAddress = Mage::getModel('sales/quote_address')
->importCustomerAddress($defaultShippingAddress);
}
else {
$shippingAddress = Mage::getModel('sales/quote_address');
}
$this->setShippingAddress($shippingAddress);
}
return $this;
}
/**
* Define customer object
*
* @param Mage_Customer_Model_Customer $customer
* @return Mage_Sales_Model_Quote
*/
public function setCustomer(Mage_Customer_Model_Customer $customer)
{
$this->_customer = $customer;
$this->setCustomerId($customer->getId());
Mage::helper('core')->copyFieldset('customer_account', 'to_quote', $customer, $this);
return $this;
}
/**
* Retrieve customer model object
*
* @return Mage_Customer_Model_Customer
*/
public function getCustomer()
{
if (is_null($this->_customer)) {
$this->_customer = Mage::getModel('customer/customer');
if ($customerId = $this->getCustomerId()) {
$this->_customer->load($customerId);
if (!$this->_customer->getId()) {
$this->_customer->setCustomerId(null);
}
}
}
return $this->_customer;
}
public function getCustomerTaxClassId()
{
/*
* tax class can vary at any time. so instead of using the value from session, we need to retrieve from db everytime
* to get the correct tax class
*/
//if (!$this->getData('customer_group_id') && !$this->getData('customer_tax_class_id')) {
$classId = Mage::getModel('customer/group')->getTaxClassId($this->getCustomerGroupId());
$this->setCustomerTaxClassId($classId);
//}
return $this->getData('customer_tax_class_id');
}
/**
* Retrieve quote address collection
*
* @return Mage_Eav_Model_Entity_Collection_Abstract
*/
public function getAddressesCollection()
{
if (is_null($this->_addresses)) {
$this->_addresses = Mage::getModel('sales/quote_address')->getCollection()
->setQuoteFilter($this->getId());
if ($this->getId()) {
foreach ($this->_addresses as $address) {
$address->setQuote($this);
}
}
}
return $this->_addresses;
}
/**
* Retrieve quote address by type
*
* @param string $type
* @return Mage_Sales_Model_Quote_Address
*/
protected function _getAddressByType($type)
{
foreach ($this->getAddressesCollection() as $address) {
if ($address->getAddressType() == $type && !$address->isDeleted()) {
return $address;
}
}
$address = Mage::getModel('sales/quote_address')->setAddressType($type);
$this->addAddress($address);
return $address;
}
/**
* Retrieve quote billing address
*
* @return Mage_Sales_Model_Quote_Address
*/
public function getBillingAddress()
{
return $this->_getAddressByType(Mage_Sales_Model_Quote_Address::TYPE_BILLING);
}
/**
* retrieve quote shipping address
*
* @return Mage_Sales_Model_Quote_Address
*/
public function getShippingAddress()
{
return $this->_getAddressByType(Mage_Sales_Model_Quote_Address::TYPE_SHIPPING);
}
public function getAllShippingAddresses()
{
$addresses = array();
foreach ($this->getAddressesCollection() as $address) {
if ($address->getAddressType()==Mage_Sales_Model_Quote_Address::TYPE_SHIPPING
&& !$address->isDeleted()) {
$addresses[] = $address;
}
}
return $addresses;
}
public function getAllAddresses()
{
$addresses = array();
foreach ($this->getAddressesCollection() as $address) {
if (!$address->isDeleted()) {
$addresses[] = $address;
}
}
return $addresses;
}
/**
*
* @param int $addressId
* @return Mage_Sales_Model_Quote_Address
*/
public function getAddressById($addressId)
{
foreach ($this->getAddressesCollection() as $address) {
if ($address->getId()==$addressId) {
return $address;
}
}
return false;
}
public function getAddressByCustomerAddressId($addressId)
{
foreach ($this->getAddressesCollection() as $address) {
if (!$address->isDeleted() && $address->getCustomerAddressId()==$addressId) {
return $address;
}
}
return false;
}
public function getShippingAddressByCustomerAddressId($addressId)
{
foreach ($this->getAddressesCollection() as $address) {
if (!$address->isDeleted() && $address->getAddressType()==Mage_Sales_Model_Quote_Address::TYPE_SHIPPING
&& $address->getCustomerAddressId()==$addressId) {
return $address;
}
}
return false;
}
public function removeAddress($addressId)
{
foreach ($this->getAddressesCollection() as $address) {
if ($address->getId()==$addressId) {
$address->isDeleted(true);
break;
}
}
return $this;
}
public function removeAllAddresses()
{
foreach ($this->getAddressesCollection() as $address) {
$address->isDeleted(true);
}
return $this;
}
public function addAddress(Mage_Sales_Model_Quote_Address $address)
{
$address->setQuote($this);
if (!$address->getId()) {
$this->getAddressesCollection()->addItem($address);
}
return $this;
}
/**
* Enter description here...
*
* @param Mage_Sales_Model_Quote_Address $address
* @return Mage_Sales_Model_Quote
*/
public function setBillingAddress(Mage_Sales_Model_Quote_Address $address)
{
$old = $this->getBillingAddress();
if (!empty($old)) {
$old->addData($address->getData());
} else {
$this->addAddress($address->setAddressType(Mage_Sales_Model_Quote_Address::TYPE_BILLING));
}
return $this;
}
/**
* Enter description here...
*
* @param Mage_Sales_Model_Quote_Address $address
* @return Mage_Sales_Model_Quote
*/
public function setShippingAddress(Mage_Sales_Model_Quote_Address $address)
{
if ($this->getIsMultiShipping()) {
$this->addAddress($address->setAddressType(Mage_Sales_Model_Quote_Address::TYPE_SHIPPING));
}
else {
$old = $this->getShippingAddress();
if (!empty($old)) {
$old->addData($address->getData());
} else {
$this->addAddress($address->setAddressType(Mage_Sales_Model_Quote_Address::TYPE_SHIPPING));
}
}
return $this;
}
public function addShippingAddress(Mage_Sales_Model_Quote_Address $address)
{
$this->setShippingAddress($address);
return $this;
}
/**
* Retrieve quote items collection
*
* @param bool $loaded
* @return Mage_Eav_Model_Entity_Collection_Abstract
*/
public function getItemsCollection($useCache = true)
{
if (is_null($this->_items)) {
$this->_items = Mage::getModel('sales/quote_item')->getCollection();
$this->_items->setQuote($this);
}
return $this->_items;
}
/**
* Retrieve quote items array
*
* @return array
*/
public function getAllItems()
{
$items = array();
foreach ($this->getItemsCollection() as $item) {
if (!$item->isDeleted()) {
$items[] = $item;
}
}
return $items;
}
/**
* Get array of all items what can be display directly
*
* @return array
*/
public function getAllVisibleItems()
{
$items = array();
foreach ($this->getItemsCollection() as $item) {
if (!$item->isDeleted() && !$item->getParentItemId()) {
$items[] = $item;
}
}
return $items;
}
/**
* Checking items availability
*
* @return bool
*/
public function hasItems()
{
return sizeof($this->getAllItems())>0;
}
/**
* Checking availability of items with decimal qty
*
* @return bool
*/
public function hasItemsWithDecimalQty()
{
foreach ($this->getAllItems() as $item) {
if ($item->getProduct()->getStockItem()
&& $item->getProduct()->getStockItem()->getIsQtyDecimal()) {
return true;
}
}
return false;
}
/**
* Retrieve item model object by item identifier
*
* @param int $itemId
* @return Mage_Sales_Model_Quote_Item
*/
public function getItemById($itemId)
{
return $this->getItemsCollection()->getItemById($itemId);
}
/**
* Remove quote item by item identifier
*
* @param int $itemId
* @return Mage_Sales_Model_Quote
*/
public function removeItem($itemId)
{
if ($item = $this->getItemById($itemId)) {
/**
* If we remove item from quote - we can't use multishipping mode
*/
$this->setIsMultiShipping(false);
$item->isDeleted(true);
if ($item->getHasChildren()) {
foreach ($item->getChildren() as $child) {
$child->isDeleted(true);
}
}
Mage::dispatchEvent('sales_quote_remove_item', array('quote_item' => $item));
}
return $this;
}
/**
* Adding new item to quote
*
* @param Mage_Sales_Model_Quote_Item $item
* @return Mage_Sales_Model_Quote
*/
public function addItem(Mage_Sales_Model_Quote_Item $item)
{
/**
* Temporary workaround for purchase process: it is too dangerous to purchase more than one nominal item
* or a mixture of nominal and non-nominal items, although technically possible.
*
* The problem is that currently it is implemented as sequential submission of nominal items and order, by one click.
* It makes logically impossible to make the process of the purchase failsafe.
* Proper solution is to submit items one by one with customer confirmation each time.
*/
if ($item->isNominal() && $this->hasItems() || $this->hasNominalItems()) {
Mage::throwException(Mage::helper('sales')->__('Nominal item can be purchased standalone only. To proceed please remove other items from the quote.'));
}
$item->setQuote($this);
if (!$item->getId()) {
$this->getItemsCollection()->addItem($item);
Mage::dispatchEvent('sales_quote_add_item', array('quote_item' => $item));
}
return $this;
}
/**
* Add product to quote
*
* return error message if product type instance can't prepare product
*
* @param mixed $product
* @return Mage_Sales_Model_Quote_Item || string
*/
public function addProduct(Mage_Catalog_Model_Product $product, $request=null)
{
if ($request === null) {
$request = 1;
}
if (is_numeric($request)) {
$request = new Varien_Object(array('qty'=>$request));
}
if (!($request instanceof Varien_Object)) {
Mage::throwException(Mage::helper('sales')->__('Invalid request for adding product to quote.'));
}
$cartCandidates = $product->getTypeInstance(true)
->prepareForCart($request, $product);
/**
* Error message
*/
if (is_string($cartCandidates)) {
return $cartCandidates;
}
/**
* If prepare process return one object
*/
if (!is_array($cartCandidates)) {
$cartCandidates = array($cartCandidates);
}
$parentItem = null;
$errors = array();
$items = array();
foreach ($cartCandidates as $candidate) {
$item = $this->_addCatalogProduct($candidate, $candidate->getCartQty());
$items[] = $item;
/**
* As parent item we should always use the item of first added product
*/
if (!$parentItem) {
$parentItem = $item;
}
if ($parentItem && $candidate->getParentProductId()) {
$item->setParentItem($parentItem);
}
/**
* We specify qty after we know about parent (for stock)
*/
$item->addQty($candidate->getCartQty());
// collect errors instead of throwing first one
if ($item->getHasError()) {
$errors[] = $item->getMessage();
}
}
if (!empty($errors)) {
Mage::throwException(implode("\n", $errors));
}
Mage::dispatchEvent('sales_quote_product_add_after', array('items' => $items));
return $item;
}
/**
* Adding catalog product object data to quote
*
* @param Mage_Catalog_Model_Product $product
* @return Mage_Sales_Model_Quote_Item
*/
protected function _addCatalogProduct(Mage_Catalog_Model_Product $product, $qty = 1)
{
$item = $this->getItemByProduct($product);
if (!$item) {
$item = Mage::getModel('sales/quote_item');
$item->setQuote($this);
if (Mage::app()->getStore()->isAdmin()) {
$item->setStoreId($this->getStore()->getId());
}
else {
$item->setStoreId(Mage::app()->getStore()->getId());
}
}
/**
* We can't modify existing child items
*/
if ($item->getId() && $product->getParentProductId()) {
return $item;
}
$item->setOptions($product->getCustomOptions())
->setProduct($product);
$this->addItem($item);
return $item;
}
/**
* Retrieve quote item by product id
*
* @param int $productId
* @return Mage_Sales_Model_Quote_Item || false
*/
public function getItemByProduct($product)
{
foreach ($this->getAllItems() as $item) {
if ($item->representProduct($product)) {
return $item;
}
}
return false;
}
public function getItemsSummaryQty()
{
$qty = $this->getData('all_items_qty');
if (is_null($qty)) {
$qty = 0;
foreach ($this->getAllItems() as $item) {
if ($item->getParentItem()) {
continue;
}
if (($children = $item->getChildren()) && $item->isShipSeparately()) {
foreach ($children as $child) {
$qty+= $child->getQty()*$item->getQty();
}
} else {
$qty+= $item->getQty();
}
}
$this->setData('all_items_qty', $qty);
}
return $qty;
}
public function getItemVirtualQty()
{
$qty = $this->getData('virtual_items_qty');
if (is_null($qty)) {
$qty = 0;
foreach ($this->getAllItems() as $item) {
if ($item->getParentItem()) {
continue;
}
if (($children = $item->getChildren()) && $item->isShipSeparately()) {
foreach ($children as $child) {
if ($child->getProduct()->getIsVirtual()) {
$qty+= $child->getQty();
}
}
} else {
if ($item->getProduct()->getIsVirtual()) {
$qty+= $item->getQty();
}
}
}
$this->setData('virtual_items_qty', $qty);
}
return $qty;
}
/*********************** PAYMENTS ***************************/
public function getPaymentsCollection()
{
if (is_null($this->_payments)) {
$this->_payments = Mage::getModel('sales/quote_payment')->getCollection()
->setQuoteFilter($this->getId());
if ($this->getId()) {
foreach ($this->_payments as $payment) {
$payment->setQuote($this);
}
}
}
return $this->_payments;
}
/**
* @return Mage_Sales_Model_Quote_Payment
*/
public function getPayment()
{
foreach ($this->getPaymentsCollection() as $payment) {
if (!$payment->isDeleted()) {
return $payment;
}
}
$payment = Mage::getModel('sales/quote_payment');
$this->addPayment($payment);
return $payment;
}
public function getPaymentById($paymentId)
{
foreach ($this->getPaymentsCollection() as $payment) {
if ($payment->getId()==$paymentId) {
return $payment;
}
}
return false;
}
public function addPayment(Mage_Sales_Model_Quote_Payment $payment)
{
$payment->setQuote($this);
if (!$payment->getId()) {
$this->getPaymentsCollection()->addItem($payment);
}
return $this;
}
public function setPayment(Mage_Sales_Model_Quote_Payment $payment)
{
if (!$this->getIsMultiPayment() && ($old = $this->getPayment())) {
$payment->setId($old->getId());
}
$this->addPayment($payment);
return $payment;
}
public function removePayment()
{
$this->getPayment()->isDeleted(true);
return $this;
}
/**
* Collect totals
*
* @return Mage_Sales_Model_Quote
*/
public function collectTotals()
{
/**
* Protect double totals collection
*/
if ($this->getTotalsCollectedFlag()) {
return $this;
}
Mage::dispatchEvent(
$this->_eventPrefix . '_collect_totals_before',
array(
$this->_eventObject=>$this
)
);
$this->setSubtotal(0);
$this->setBaseSubtotal(0);
$this->setSubtotalWithDiscount(0);
$this->setBaseSubtotalWithDiscount(0);
$this->setGrandTotal(0);
$this->setBaseGrandTotal(0);
foreach ($this->getAllAddresses() as $address) {
$address->setSubtotal(0);
$address->setBaseSubtotal(0);
$address->setGrandTotal(0);
$address->setBaseGrandTotal(0);
$address->collectTotals();
$this->setSubtotal((float) $this->getSubtotal()+$address->getSubtotal());
$this->setBaseSubtotal((float) $this->getBaseSubtotal()+$address->getBaseSubtotal());
$this->setSubtotalWithDiscount((float) $this->getSubtotalWithDiscount()+$address->getSubtotalWithDiscount());
$this->setBaseSubtotalWithDiscount((float) $this->getBaseSubtotalWithDiscount()+$address->getBaseSubtotalWithDiscount());
$this->setGrandTotal((float) $this->getGrandTotal()+$address->getGrandTotal());
$this->setBaseGrandTotal((float) $this->getBaseGrandTotal()+$address->getBaseGrandTotal());
}
Mage::helper('sales')->checkQuoteAmount($this, $this->getGrandTotal());
Mage::helper('sales')->checkQuoteAmount($this, $this->getBaseGrandTotal());
$this->setItemsCount(0);
$this->setItemsQty(0);
$this->setVirtualItemsQty(0);
foreach ($this->getAllVisibleItems() as $item) {
if ($item->getParentItem()) {
continue;
}
if (($children = $item->getChildren()) && $item->isShipSeparately()) {
foreach ($children as $child) {
if ($child->getProduct()->getIsVirtual()) {
$this->setVirtualItemsQty($this->getVirtualItemsQty() + $child->getQty()*$item->getQty());
}
}
}
if ($item->getProduct()->getIsVirtual()) {
$this->setVirtualItemsQty($this->getVirtualItemsQty() + $item->getQty());
}
$this->setItemsCount($this->getItemsCount()+1);
$this->setItemsQty((float) $this->getItemsQty()+$item->getQty());
}
$this->setData('trigger_recollect', 0);
$this->_validateCouponCode();
Mage::dispatchEvent(
$this->_eventPrefix . '_collect_totals_after',
array(
$this->_eventObject=>$this
)
);
$this->setTotalsCollectedFlag(true);
return $this;
}
/**
* Get all quote totals (sorted by priority)
* Method process quote states isVirtual and isMultiShipping
*
* @return array
*/
public function getTotals()
{
/**
* If quote is virtual we are using totals of billing address because
* all items assigned to it
*/
if ($this->isVirtual()) {
return $this->getBillingAddress()->getTotals();
}
$shippingAddress = $this->getShippingAddress();
$totals = $shippingAddress->getTotals();
// Going through all quote addresses and merge their totals
foreach ($this->getAddressesCollection() as $address) {
if ($address->isDeleted() || $address === $shippingAddress) {
continue;
}
foreach ($address->getTotals() as $code => $total) {
if (isset($totals[$code])) {
$totals[$code]->merge($total);
} else {
$totals[$code] = $total;
}
}
}
$sortedTotals = array();
foreach ($this->getBillingAddress()->getTotalModels() as $total) {
/* @var $total Mage_Sales_Model_Quote_Address_Total_Abstract */
if (isset($totals[$total->getCode()])) {
$sortedTotals[$total->getCode()] = $totals[$total->getCode()];
}
}
return $sortedTotals;
}
public function addMessage($message, $index='error')
{
$messages = $this->getData('messages');
if (is_null($messages)) {
$messages = array();
}
if (isset($messages[$index])) {
return $this;
}
if (is_string($message)) {
$message = Mage::getSingleton('core/message')->error($message);
}
$messages[$index] = $message;
$this->setData('messages', $messages);
return $this;
}
public function getMessages()
{
$messages = $this->getData('messages');
if (is_null($messages)) {
$messages = array();
$this->setData('messages', $messages);
}
return $messages;
}
/**
* Generate new increment order id and associate it with current quote
*
* @return Mage_Sales_Model_Quote
*/
public function reserveOrderId()
{
if (!$this->getReservedOrderId()) {
$this->setReservedOrderId($this->_getResource()->getReservedOrderId($this));
} else {
//checking if reserved order id was already used for some order
//if yes reserving new one if not using old one
if ($this->_getResource()->isOrderIncrementIdUsed($this->getReservedOrderId())) {
$this->setReservedOrderId($this->_getResource()->getReservedOrderId($this));
}
}
return $this;
}
public function validateMinimumAmount($multishipping = false)
{
$storeId = $this->getStoreId();
$minOrderActive = Mage::getStoreConfigFlag('sales/minimum_order/active', $storeId);
$minOrderMulti = Mage::getStoreConfigFlag('sales/minimum_order/multi_address', $storeId);
if (!$minOrderActive) {
return true;
}
if ($multishipping && !$minOrderMulti) {
$baseTotal = 0;
foreach ($this->getAllAddresses() as $address) {
/* @var $address Mage_Sales_Model_Quote_Address */
$baseTotal += $address->getBaseSubtotalWithDiscount();
}
if ($baseTotal < Mage::getStoreConfig('sales/minimum_order/amount', $storeId)) {
return false;
}
}
else {
foreach ($this->getAllAddresses() as $address) {
/* @var $address Mage_Sales_Model_Quote_Address */
if (!$address->validateMinimumAmount()) {
return false;
}
}
}
return true;
}
/**
* Check quote for virtual product only
*
* @return bool
*/
public function isVirtual()
{
$isVirtual = true;
$countItems = 0;
foreach ($this->getItemsCollection() as $_item) {
/* @var $_item Mage_Sales_Model_Quote_Item */
if ($_item->isDeleted() || $_item->getParentItemId()) {
continue;
}
$countItems ++;
if (!$_item->getProduct()->getIsVirtual()) {
$isVirtual = false;
}
}
return $countItems == 0 ? false : $isVirtual;
}
/**
* Check quote for virtual product only
*
* @return bool
*/
public function getIsVirtual()
{
return intval($this->isVirtual());
}
/**
* Has a virtual products on quote
*
* @return bool
*/
public function hasVirtualItems()
{
$hasVirtual = false;
foreach ($this->getItemsCollection() as $_item) {
if ($_item->getParentItemId()) {
continue;
}
if ($_item->getProduct()->isVirtual()) {
$hasVirtual = true;
}
}
return $hasVirtual;
}
/**
* Merge quotes
*
* @param Mage_Sales_Model_Quote $quote
* @return Mage_Sales_Model_Quote
*/
public function merge(Mage_Sales_Model_Quote $quote)
{
Mage::dispatchEvent(
$this->_eventPrefix . '_merge_before',
array(
$this->_eventObject=>$this,
'source'=>$quote
)
);
foreach ($quote->getAllVisibleItems() as $item) {
$found = false;
foreach ($this->getAllItems() as $quoteItem) {
if ($quoteItem->compare($item)) {
$quoteItem->setQty($quoteItem->getQty() + $item->getQty());
$found = true;
break;
}
}
if (!$found) {
$newItem = clone $item;
$this->addItem($newItem);
if ($item->getHasChildren()) {
foreach ($item->getChildren() as $child) {
$newChild = clone $child;
$newChild->setParentItem($newItem);
$this->addItem($newChild);
}
}
}
}
/**
* Init shipping and billing address if quote is new
*/
if (!$this->getId()) {
$this->getShippingAddress();
$this->getBillingAddress();
}
if ($quote->getCouponCode()) {
$this->setCouponCode($quote->getCouponCode());
}
Mage::dispatchEvent(
$this->_eventPrefix . '_merge_after',
array(
$this->_eventObject=>$this,
'source'=>$quote
)
);
return $this;
}
/**
* Whether there are recurring items
*
* @return bool
*/
public function hasRecurringItems()
{
foreach ($this->getAllVisibleItems() as $item) {
if ($item->getProduct() && $item->getProduct()->isRecurring()) {
return true;
}
}
return false;
}
/**
* Getter whether quote has nominal items
* Can bypass treating virtual items as nominal
*
* @param bool $countVirtual
* @return bool
*/
public function hasNominalItems($countVirtual = true)
{
foreach ($this->getAllVisibleItems() as $item) {
if ($item->isNominal()) {
if ((!$countVirtual) && $item->getProduct()->isVirtual()) {
continue;
}
return true;
}
}
return false;
}
/**
* Whether quote has nominal items only
*
* @return bool
*/
public function isNominal()
{
foreach ($this->getAllVisibleItems() as $item) {
if (!$item->isNominal()) {
return false;
}
}
return true;
}
/**
* Create recurring payment profiles basing on the current items
*
* @return array
*/
public function prepareRecurringPaymentProfiles()
{
if (!$this->getTotalsCollectedFlag()) {
// Whoops! Make sure nominal totals must be calculated here.
throw new Exception('Quote totals must be collected before this operation.');
}
$result = array();
foreach ($this->getAllVisibleItems() as $item) {
$product = $item->getProduct();
if (is_object($product) && ($product->isRecurring())
&& $profile = Mage::getModel('sales/recurring_profile')->importProduct($product)) {
$profile->importQuote($this);
$profile->importQuoteItem($item);
$result[] = $profile;
}
}
return $result;
}
protected function _validateCouponCode()
{
$code = $this->_getData('coupon_code');
if ($code) {
$addressHasCoupon = false;
$addresses = $this->getAllAddresses();
if (count($addresses)>0) {
foreach ($addresses as $address) {
if ($address->hasCouponCode()) {
$addressHasCoupon = true;
}
}
if (!$addressHasCoupon) {
$this->setCouponCode('');
}
}
}
return $this;
}
/**
* Trigger collect totals after loading, if required
*
* @return Mage_Sales_Model_Quote
*/
protected function _afterLoad()
{
// collect totals and save me, if required
if (1 == $this->getData('trigger_recollect')) {
$this->collectTotals()->save();
}
return parent::_afterLoad();
}
/**
* @deprecated after 1.4 beta1 - one page checkout responsibility
*/
const CHECKOUT_METHOD_REGISTER = 'register';
const CHECKOUT_METHOD_GUEST = 'guest';
const CHECKOUT_METHOD_LOGIN_IN = 'login_in';
/**
* Return quote checkout method code
*
* @deprecated after 1.4 beta1 it is checkout module responsibility
* @param boolean $originalMethod if true return defined method from begining
* @return string
*/
public function getCheckoutMethod($originalMethod = false)
{
if ($this->getCustomerId() && !$originalMethod) {
return self::CHECKOUT_METHOD_LOGIN_IN;
}
return $this->_getData('checkout_method');
}
/**
* Check is allow Guest Checkout
*
* @deprecated after 1.4 beta1 it is checkout module responsibility
* @return bool
*/
public function isAllowedGuestCheckout()
{
return Mage::helper('checkout')->isAllowedGuestCheckout($this, $this->getStoreId());
}
}