* @copyright 2007-2014 PrestaShop SA * @license http://opensource.org/licenses/osl-3.0.php Open Software License (OSL 3.0) * International Registered Trademark & Property of PrestaShop SA */ class OrderSlipCore extends ObjectModel { /** @var integer */ public $id; /** @var integer */ public $id_customer; /** @var integer */ public $id_order; /** @var float */ public $conversion_rate; /** @var integer */ public $amount; /** @var integer */ public $shipping_cost; /** @var integer */ public $shipping_cost_amount; /** @var integer */ public $partial; /** @var string Object creation date */ public $date_add; /** @var string Object last modification date */ public $date_upd; /** * @see ObjectModel::$definition */ public static $definition = array( 'table' => 'order_slip', 'primary' => 'id_order_slip', 'fields' => array( 'id_customer' => array('type' => self::TYPE_INT, 'validate' => 'isUnsignedId', 'required' => true), 'id_order' => array('type' => self::TYPE_INT, 'validate' => 'isUnsignedId', 'required' => true), 'conversion_rate' => array('type' => self::TYPE_FLOAT, 'validate' => 'isFloat', 'required' => true), 'amount' => array('type' => self::TYPE_FLOAT, 'validate' => 'isFloat'), 'shipping_cost' => array('type' => self::TYPE_INT), 'shipping_cost_amount' => array('type' => self::TYPE_FLOAT, 'validate' => 'isFloat'), 'partial' => array('type' => self::TYPE_INT), 'date_add' => array('type' => self::TYPE_DATE, 'validate' => 'isDate'), 'date_upd' => array('type' => self::TYPE_DATE, 'validate' => 'isDate'), ), ); protected $webserviceParameters = array( 'objectNodeName' => 'order_slip', 'objectsNodeName' => 'order_slips', 'fields' => array( 'id_customer' => array('xlink_resource'=> 'customers'), 'id_order' => array('xlink_resource'=> 'orders'), ), 'associations' => array( 'order_slip_details' => array('resource' => 'order_slip_details', 'setter' => false, 'virtual_entity' => true, 'fields' => array( 'id' => array(), 'id_order_detail' => array('required' => true), 'product_quantity' => array('required' => true), 'amount_tax_excl' => array('required' => true), 'amount_tax_incl' => array('required' => true), )), ), ); public function addSlipDetail($orderDetailList, $productQtyList) { foreach ($orderDetailList as $key => $id_order_detail) { if ($qty = (int)($productQtyList[$key])) { $order_detail = new OrderDetail((int)$id_order_detail); if (Validate::isLoadedObject($order_detail)) Db::getInstance()->insert('order_slip_detail', array( 'id_order_slip' => (int)$this->id, 'id_order_detail' => (int)$id_order_detail, 'product_quantity' => $qty, 'amount_tax_excl' => $order_detail->unit_price_tax_excl * $qty, 'amount_tax_incl' => $order_detail->unit_price_tax_incl * $qty )); } } } public static function getOrdersSlip($customer_id, $order_id = false) { return Db::getInstance()->executeS(' SELECT * FROM `'._DB_PREFIX_.'order_slip` WHERE `id_customer` = '.(int)($customer_id). ($order_id ? ' AND `id_order` = '.(int)($order_id) : '').' ORDER BY `date_add` DESC'); } public static function getOrdersSlipDetail($id_order_slip = false, $id_order_detail = false) { return Db::getInstance(_PS_USE_SQL_SLAVE_)->executeS( ($id_order_detail ? 'SELECT SUM(`product_quantity`) AS `total`' : 'SELECT *'). 'FROM `'._DB_PREFIX_.'order_slip_detail`' .($id_order_slip ? ' WHERE `id_order_slip` = '.(int)($id_order_slip) : '') .($id_order_detail ? ' WHERE `id_order_detail` = '.(int)($id_order_detail) : '')); } public static function getOrdersSlipProducts($orderSlipId, $order) { $cart_rules = $order->getCartRules(true); $productsRet = OrderSlip::getOrdersSlipDetail($orderSlipId); $order_details = $order->getProductsDetail(); $slip_quantity = array(); foreach ($productsRet as $slip_detail) $slip_quantity[$slip_detail['id_order_detail']] = $slip_detail['product_quantity']; $products = array(); foreach ($order_details as $key => $product) if (isset($slip_quantity[$product['id_order_detail']])) { $products[$key] = $product; $products[$key]['product_quantity'] = $slip_quantity[$product['id_order_detail']]; } return $order->getProducts($products); } /** * * Get resume of all refund for one product line * @param $id_order_detail */ public static function getProductSlipResume($id_order_detail) { return Db::getInstance(_PS_USE_SQL_SLAVE_)->getRow(' SELECT SUM(product_quantity) product_quantity, SUM(amount_tax_excl) amount_tax_excl, SUM(amount_tax_incl) amount_tax_incl FROM `'._DB_PREFIX_.'order_slip_detail` WHERE `id_order_detail` = '.(int)$id_order_detail); } /** * * Get refund details for one product line * @param $id_order_detail */ public static function getProductSlipDetail($id_order_detail) { return Db::getInstance(_PS_USE_SQL_SLAVE_)->executeS(' SELECT product_quantity, amount_tax_excl, amount_tax_incl, date_add FROM `'._DB_PREFIX_.'order_slip_detail` osd LEFT JOIN `'._DB_PREFIX_.'order_slip` os ON os.id_order_slip = osd.id_order_slip WHERE osd.`id_order_detail` = '.(int)$id_order_detail); } public function getProducts() { $result = Db::getInstance(_PS_USE_SQL_SLAVE_)->executeS(' SELECT *, osd.product_quantity FROM `'._DB_PREFIX_.'order_slip_detail` osd INNER JOIN `'._DB_PREFIX_.'order_detail` od ON osd.id_order_detail = od.id_order_detail WHERE osd.`id_order_slip` = '.(int)$this->id); $order = new Order($this->id_order); $products = array(); foreach ($result as $row) { $order->setProductPrices($row); $products[] = $row; } return $products; } public static function getSlipsIdByDate($dateFrom, $dateTo) { $result = Db::getInstance(_PS_USE_SQL_SLAVE_)->executeS(' SELECT `id_order_slip` FROM `'._DB_PREFIX_.'order_slip` os LEFT JOIN `'._DB_PREFIX_.'orders` o ON (o.`id_order` = os.`id_order`) WHERE os.`date_add` BETWEEN \''.pSQL($dateFrom).' 00:00:00\' AND \''.pSQL($dateTo).' 23:59:59\' '.Shop::addSqlRestriction(Shop::SHARE_ORDER, 'o').' ORDER BY os.`date_add` ASC'); $slips = array(); foreach ($result as $slip) $slips[] = (int)$slip['id_order_slip']; return $slips; } public static function createOrderSlip($order, $productList, $qtyList, $shipping_cost = false) { $currency = new Currency($order->id_currency); $orderSlip = new OrderSlip(); $orderSlip->id_customer = (int)($order->id_customer); $orderSlip->id_order = (int)($order->id); $orderSlip->shipping_cost = (int)$shipping_cost; if ($orderSlip->shipping_cost) $orderSlip->shipping_cost_amount = $order->total_shipping_tax_incl; $orderSlip->conversion_rate = $currency->conversion_rate; $orderSlip->partial = 0; $orderSlip->amount = $orderSlip->shipping_cost_amount; foreach ($productList as $id_order_detail) { $order_detail = new OrderDetail((int)$id_order_detail); $orderSlip->amount += $order_detail->unit_price_tax_incl * $qtyList[(int)$id_order_detail]; } if (!$orderSlip->add()) return false; $orderSlip->addSlipDetail($productList, $qtyList); return true; } public static function createPartialOrderSlip($order, $amount, $shipping_cost_amount, $order_detail_list) { $currency = new Currency($order->id_currency); $orderSlip = new OrderSlip(); $orderSlip->id_customer = (int)($order->id_customer); $orderSlip->id_order = (int)($order->id); $orderSlip->amount = (float)($amount); $orderSlip->shipping_cost = false; $orderSlip->shipping_cost_amount = (float)($shipping_cost_amount); $orderSlip->conversion_rate = $currency->conversion_rate; $orderSlip->partial = 1; if (!$orderSlip->add()) return false; $orderSlip->addPartialSlipDetail($order_detail_list); return true; } public function addPartialSlipDetail($order_detail_list) { foreach ($order_detail_list as $id_order_detail => $tab) { $order_detail = new OrderDetail($id_order_detail); $order_slip_resume = self::getProductSlipResume($id_order_detail); if ($tab['amount'] + $order_slip_resume['amount_tax_incl'] > $order_detail->total_price_tax_incl) $tab['amount'] = $order_detail->total_price_tax_incl - $order_slip_resume['amount_tax_incl']; if ($tab['amount'] == 0) continue; if ($tab['quantity'] + $order_slip_resume['product_quantity'] > $order_detail->product_quantity) $tab['quantity'] = $order_detail->product_quantity - $order_slip_resume['product_quantity']; $tab['amount_tax_excl'] = $tab['amount_tax_incl'] = $tab['amount']; $id_tax = (int)Db::getInstance()->getValue('SELECT `id_tax` FROM `'._DB_PREFIX_.'order_detail_tax` WHERE `id_order_detail` = '.(int)$id_order_detail); if ($id_tax > 0) { $rate = (float)Db::getInstance()->getValue('SELECT `rate` FROM `'._DB_PREFIX_.'tax` WHERE `id_tax` = '.(int)$id_tax); if ($rate > 0) { $rate = 1 + ($rate / 100); $tab['amount_tax_excl'] = $tab['amount_tax_excl'] / $rate; } } if ($tab['quantity'] > 0 && $tab['quantity'] > $order_detail->product_quantity_refunded) { $order_detail->product_quantity_refunded = $tab['quantity']; $order_detail->save(); } $insertOrderSlip = array( 'id_order_slip' => (int)($this->id), 'id_order_detail' => (int)($id_order_detail), 'product_quantity' => (int)($tab['quantity']), 'amount_tax_excl' => (float)($tab['amount_tax_excl']), 'amount_tax_incl' => (float)($tab['amount_tax_incl']), ); Db::getInstance()->insert('order_slip_detail', $insertOrderSlip); } } public function getEcoTaxTaxesBreakdown() { $ecotax_detail = array(); foreach ($this->getOrdersSlipDetail((int)$this->id) as $order_slip_details) { $row = Db::getInstance()->getRow(' SELECT `ecotax_tax_rate` as `rate`, `ecotax` as `ecotax_tax_excl`, `ecotax` as `ecotax_tax_incl`, `product_quantity` FROM `'._DB_PREFIX_.'order_detail` WHERE `id_order_detail` = '.(int)$order_slip_details['id_order_detail'] ); if (!isset($ecotax_detail[$row['rate']])) $ecotax_detail[$row['rate']] = array('ecotax_tax_incl' => 0, 'ecotax_tax_excl' => 0, 'rate' => $row['rate']); $ecotax_detail[$row['rate']]['ecotax_tax_incl'] += Tools::ps_round(($row['ecotax_tax_excl'] * $order_slip_details['product_quantity']) + ($row['ecotax_tax_excl'] * $order_slip_details['product_quantity'] * $row['rate'] / 100), 2); $ecotax_detail[$row['rate']]['ecotax_tax_excl'] += Tools::ps_round($row['ecotax_tax_excl'] * $order_slip_details['product_quantity'], 2); } return $ecotax_detail; } public function getWsOrderSlipDetails() { $query = 'SELECT id_order_slip as id, id_order_detail, product_quantity, amount_tax_excl, amount_tax_incl FROM `'._DB_PREFIX_.'order_slip_detail` WHERE id_order_slip = '.(int)$this->id; $result = Db::getInstance()->executeS($query); return $result; } public function setWsOrderSlipDetails($values) { if (Db::getInstance()->execute('DELETE from `'._DB_PREFIX_.'order_slip_detail` where id_order_slip = '.(int)$this->id)) { $query = 'INSERT INTO `'._DB_PREFIX_.'order_slip_detail`(`id_order_slip`, `id_order_detail`, `product_quantity`, `amount_tax_excl`, `amount_tax_incl`) VALUES '; foreach ($values as $value) $query .= '('.(int)$this->id.', '.(int)$value['id_order_detail'].', '.(int)$value['product_quantity'].', '. (isset($value['amount_tax_excl']) ? (float)$value['amount_tax_excl'] : 'NULL').', '. (isset($value['amount_tax_incl']) ? (float)$value['amount_tax_incl'] : 'NULL'). '),'; $query = rtrim($query, ','); Db::getInstance()->execute($query); } return true; } }