source: branches/version-2_12-dev/data/class/helper/SC_Helper_Purchase.php @ 21664

Revision 21664, 50.3 KB checked in by Seasoft, 12 years ago (diff)

#1613 (typo修正・ソース整形・ソースコメントの改善)

  • Property svn:eol-style set to LF
  • Property svn:keywords set to Id
  • Property svn:mime-type set to text/x-httpd-php; charset=UTF-8
RevLine 
[18860]1<?php
2/*
3 * This file is part of EC-CUBE
4 *
[20764]5 * Copyright(c) 2000-2011 LOCKON CO.,LTD. All Rights Reserved.
[18860]6 *
7 * http://www.lockon.co.jp/
8 *
9 * This program is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU General Public License
11 * as published by the Free Software Foundation; either version 2
12 * of the License, or (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
22 */
23
24/**
25 * 商品購入関連のヘルパークラス.
26 *
27 * TODO 購入時強制会員登録機能(#521)の実装を検討
28 * TODO dtb_customer.buy_times, dtb_customer.buy_total の更新
29 *
30 * @package Helper
31 * @author Kentaro Ohkouchi
32 * @version $Id$
33 */
34class SC_Helper_Purchase {
35
[21571]36    var $arrShippingKey = array(
37        'name01', 'name02', 'kana01', 'kana02',
38        'sex', 'zip01', 'zip02', 'pref', 'addr01', 'addr02',
39        'tel01', 'tel02', 'tel03',
40    );
41
[18860]42    /**
43     * 受注を完了する.
[18871]44     *
45     * 下記のフローで受注を完了する.
46     *
47     * 1. トランザクションを開始する
48     * 2. カートの内容を検証する.
49     * 3. 受注一時テーブルから受注データを読み込む
50     * 4. ユーザーがログインしている場合はその他の発送先へ登録する
51     * 5. 受注データを受注テーブルへ登録する
52     * 6. トランザクションをコミットする
53     *
54     * 実行中に, 何らかのエラーが発生した場合, 処理を中止しエラーページへ遷移する
55     *
[21182]56     * 決済モジュールを使用する場合は対応状況を「決済処理中」に設定し,
[18871]57     * 決済完了後「新規受付」に変更すること
58     *
[21182]59     * @param integer $orderStatus 受注処理を完了する際に設定する対応状況
[18871]60     * @return void
[18860]61     */
[18871]62    function completeOrder($orderStatus = ORDER_NEW) {
[20507]63        $objQuery =& SC_Query_Ex::getSingletonInstance();
[20448]64        $objSiteSession = new SC_SiteSession_Ex();
[20444]65        $objCartSession = new SC_CartSession_Ex();
[20490]66        $objCustomer = new SC_Customer_Ex();
[18860]67        $customerId = $objCustomer->getValue('customer_id');
68
69        $objQuery->begin();
[19912]70        if (!$objSiteSession->isPrePage()) {
[20484]71            SC_Utils_Ex::sfDispSiteError(PAGE_ERROR, $objSiteSession);
[19912]72        }
[19860]73
74        $uniqId = $objSiteSession->getUniqId();
75        $this->verifyChangeCart($uniqId, $objCartSession);
76
[18860]77        $orderTemp = $this->getOrderTemp($uniqId);
78
[18871]79        $orderTemp['status'] = $orderStatus;
[20771]80        $cartkey = $objCartSession->getKey();
[21589]81        $order_id = $this->registerOrderComplete($orderTemp, $objCartSession, $cartkey);
[21016]82        $isMultiple = SC_Helper_Purchase::isMultiple();
[21446]83        $shippingTemp =& $this->getShippingTemp($isMultiple);
[21006]84        foreach ($shippingTemp as $shippingId => $val) {
[21589]85            $this->registerShipmentItem($order_id, $shippingId, $val['shipment_item']);
[19862]86        }
[19868]87
[21589]88        $this->registerShipping($order_id, $shippingTemp);
[18860]89        $objQuery->commit();
[21259]90
91        //会員情報の最終購入日、購入合計を更新
[21441]92        if ($customerId > 0) {
[21259]93            SC_Customer_Ex::updateOrderSummary($customerId);
94        }
95
[21589]96        $this->cleanupSession($order_id, $objCartSession, $objCustomer, $cartkey);
[21013]97
[21589]98        GC_Utils_Ex::gfPrintLog('order complete. order_id=' . $order_id);
[18860]99    }
100
101    /**
[20813]102     * 受注をキャンセルする.
103     *
104     * 受注完了後の受注をキャンセルする.
105     * この関数は, 主に決済モジュールにて, 受注をキャンセルする場合に使用する.
106     *
[21182]107     * 対応状況を引数 $orderStatus で指定した値に変更する.
[20813]108     * (デフォルト ORDER_CANCEL)
109     * 引数 $is_delete が true の場合は, 受注データを論理削除する.
110     * 商品の在庫数は, 受注前の在庫数に戻される.
111     *
112     * @param integer $order_id 受注ID
[21182]113     * @param integer $orderStatus 対応状況
[20813]114     * @param boolean $is_delete 受注データを論理削除する場合 true
115     * @return void
116     */
117    function cancelOrder($order_id, $orderStatus = ORDER_CANCEL, $is_delete = false) {
118        $objQuery =& SC_Query_Ex::getSingletonInstance();
119        $in_transaction = $objQuery->inTransaction();
120        if (!$in_transaction) {
121            $objQuery->begin();
122        }
123
124        $arrParams['status'] = $orderStatus;
125        if ($is_delete) {
126            $arrParams['del_flg'] = 1;
127        }
128
129        $this->registerOrder($order_id, $arrParams);
130
131        $arrOrderDetail = $this->getOrderDetail($order_id);
132        foreach ($arrOrderDetail as $arrDetail) {
133            $objQuery->update('dtb_products_class', array(),
[21514]134                              'product_class_id = ?', array($arrDetail['product_class_id']),
[20813]135                              array('stock' => 'stock + ?'), array($arrDetail['quantity']));
136        }
137        if (!$in_transaction) {
138            $objQuery->commit();
139        }
140    }
141
142    /**
143     * 受注をキャンセルし, カートをロールバックして, 受注一時IDを返す.
144     *
145     * 受注完了後の受注をキャンセルし, カートの状態を受注前の状態へ戻す.
146     * この関数は, 主に, 決済モジュールに遷移した後, 購入確認画面へ戻る場合に使用する.
147     *
[21182]148     * 対応状況を引数 $orderStatus で指定した値に変更する.
[20813]149     * (デフォルト ORDER_CANCEL)
150     * 引数 $is_delete が true の場合は, 受注データを論理削除する.
151     * 商品の在庫数, カートの内容は受注前の状態に戻される.
152     *
153     * @param integer $order_id 受注ID
[21182]154     * @param integer $orderStatus 対応状況
[20813]155     * @param boolean $is_delete 受注データを論理削除する場合 true
156     * @return string 受注一時ID
157     */
158    function rollbackOrder($order_id, $orderStatus = ORDER_CANCEL, $is_delete = false) {
159        $objQuery =& SC_Query_Ex::getSingletonInstance();
160        $in_transaction = $objQuery->inTransaction();
161        if (!$in_transaction) {
162            $objQuery->begin();
163        }
164
165        $this->cancelOrder($order_id, $orderStatus, $is_delete);
166        $arrOrderTemp = $this->getOrderTempByOrderId($order_id);
167        $_SESSION = array_merge($_SESSION, unserialize($arrOrderTemp['session']));
168
169        $objSiteSession = new SC_SiteSession_Ex();
170        $objCartSession = new SC_CartSession_Ex();
171        $objCustomer = new SC_Customer_Ex();
172
173        // 新たに受注一時情報を保存する
174        $objSiteSession->unsetUniqId();
175        $uniqid = $objSiteSession->getUniqId();
176        $arrOrderTemp['del_flg'] = 0;
[20835]177        $this->saveOrderTemp($uniqid, $arrOrderTemp, $objCustomer);
[20813]178        $this->verifyChangeCart($uniqid, $objCartSession);
179        $objSiteSession->setRegistFlag();
180
181        if (!$in_transaction) {
182            $objQuery->commit();
183        }
184        return $uniqid;
185    }
186
187    /**
[19860]188     * カートに変化が無いか検証する.
189     *
190     * ユニークIDとセッションのユニークIDを比較し, 異なる場合は
191     * エラー画面を表示する.
192     *
193     * カートが空の場合, 購入ボタン押下後にカートが変更された場合は
194     * カート画面へ遷移する.
195     *
196     * @param string $uniqId ユニークID
197     * @param SC_CartSession $objCartSession
198     * @return void
199     */
200    function verifyChangeCart($uniqId, &$objCartSession) {
[20111]201        $cartKeys = $objCartSession->getKeys();
[19860]202
[20111]203        // カート内が空でないか
204        if (SC_Utils_Ex::isBlank($cartKeys)) {
205            SC_Response_Ex::sendRedirect(CART_URLPATH);
[20167]206            exit;
[20111]207        }
208
[19860]209        foreach ($cartKeys as $cartKey) {
210            // 初回のみカートの内容を保存
[20553]211            $objCartSession->saveCurrentCart($uniqId, $cartKey);
[20111]212
[19860]213            /*
214             * POSTのユニークIDとセッションのユニークIDを比較
215             *(ユニークIDがPOSTされていない場合はスルー)
216             */
[21441]217            if (!SC_SiteSession_Ex::checkUniqId()) {
[20111]218                SC_Utils_Ex::sfDispSiteError(CANCEL_PURCHASE);
[20167]219                exit;
[19860]220            }
221
[20111]222            // 購入ボタンを押してから変化がないか
223            $quantity = $objCartSession->getTotalQuantity($cartKey);
[21441]224            if ($objCartSession->checkChangeCart($cartKey) || !($quantity > 0)) {
[20111]225                SC_Response_Ex::sendRedirect(CART_URLPATH);
[19860]226                exit;
227            }
228        }
229    }
230
231    /**
[18860]232     * 受注一時情報を取得する.
233     *
234     * @param integer $uniqId 受注一時情報ID
235     * @return array 受注一時情報の配列
236     */
237    function getOrderTemp($uniqId) {
[20507]238        $objQuery =& SC_Query_Ex::getSingletonInstance();
[21527]239        return $objQuery->getRow('*', 'dtb_order_temp', 'order_temp_id = ?', array($uniqId));
[18860]240    }
241
242    /**
[20813]243     * 受注IDをキーにして受注一時情報を取得する.
244     *
245     * @param integer $order_id 受注ID
246     * @return array 受注一時情報の配列
247     */
248    function getOrderTempByOrderId($order_id) {
249        $objQuery =& SC_Query_Ex::getSingletonInstance();
[21527]250        return $objQuery->getRow('*', 'dtb_order_temp', 'order_id = ?', array($order_id));
[20813]251    }
252
253    /**
[19860]254     * 受注一時情報を保存する.
255     *
256     * 既存のデータが存在しない場合は新規保存. 存在する場合は更新する.
257     * 既存のデータが存在せず, ユーザーがログインしている場合は,
258     * 会員情報をコピーする.
259     *
260     * @param integer $uniqId 受注一時情報ID
261     * @param array $params 登録する受注情報の配列
262     * @param SC_Customer $objCustomer SC_Customer インスタンス
263     * @return array void
264     */
265    function saveOrderTemp($uniqId, $params, &$objCustomer) {
266        if (SC_Utils_Ex::isBlank($uniqId)) {
267            return;
268        }
[21303]269        $params['device_type_id'] = SC_Display_Ex::detectDevice();
[20507]270        $objQuery =& SC_Query_Ex::getSingletonInstance();
[19860]271        // 存在するカラムのみを対象とする
272        $cols = $objQuery->listTableFields('dtb_order_temp');
273        foreach ($params as $key => $val) {
274            if (in_array($key, $cols)) {
275                $sqlval[$key] = $val;
276            }
277        }
278
279        $sqlval['session'] = serialize($_SESSION);
280        $exists = $this->getOrderTemp($uniqId);
281        if (SC_Utils_Ex::isBlank($exists)) {
282            $this->copyFromCustomer($sqlval, $objCustomer);
283            $sqlval['order_temp_id'] = $uniqId;
[21185]284            $sqlval['create_date'] = 'CURRENT_TIMESTAMP';
[21481]285            $objQuery->insert('dtb_order_temp', $sqlval);
[19860]286        } else {
[21527]287            $objQuery->update('dtb_order_temp', $sqlval, 'order_temp_id = ?', array($uniqId));
[19860]288        }
289    }
290
291    /**
[21006]292     * 配送情報をセッションから取得する.
293     *
294     * @param bool $has_shipment_item 配送商品を保有している配送先のみ返す。
[19861]295     */
[21662]296    function getShippingTemp($has_shipment_item = false) {
[21006]297        if ($has_shipment_item) {
298            $arrReturn = array();
299            foreach ($_SESSION['shipping'] as $key => $arrVal) {
300                if (count($arrVal['shipment_item']) == 0) continue;
301                $arrReturn[$key] = $arrVal;
302            }
303            return $arrReturn;
304        }
305
[19861]306        return $_SESSION['shipping'];
307    }
308
309    /**
[20963]310     * 配送商品をクリア(消去)する
311     *
312     * @param integer $shipping_id 配送先ID
313     * @return void
314     */
315    function clearShipmentItemTemp($shipping_id = null) {
316        if (is_null($shipping_id)) {
[21648]317            foreach (array_keys($_SESSION['shipping']) as $key) {
[21571]318                $this->clearShipmentItemTemp($key);
319            }
[20963]320        } else {
[21648]321            if (!isset($_SESSION['shipping'][$shipping_id])) return;
322            if (!is_array($_SESSION['shipping'][$shipping_id])) return;
[21571]323            unset($_SESSION['shipping'][$shipping_id]['shipment_item']);
[20963]324        }
325    }
326
327    /**
[19868]328     * 配送商品を設定する.
[20167]329     *
330     * @param integer $shipping_id 配送先ID
331     * @param integer $product_class_id 商品規格ID
332     * @param integer $quantity 数量
333     * @return void
[19860]334     */
[20167]335    function setShipmentItemTemp($shipping_id, $product_class_id, $quantity) {
336        // 配列が長くなるので, リファレンスを使用する
[20975]337        $arrItems =& $_SESSION['shipping'][$shipping_id]['shipment_item'][$product_class_id];
[19868]338
[20975]339        $arrItems['shipping_id'] = $shipping_id;
340        $arrItems['product_class_id'] = $product_class_id;
341        $arrItems['quantity'] = $quantity;
[20167]342
[20487]343        $objProduct = new SC_Product_Ex();
[20975]344
345        // カート情報から読みこめば済むと思うが、一旦保留。むしろ、カート情報も含め、セッション情報を縮小すべきかもしれない。
346        /*
347        $objCartSession = new SC_CartSession_Ex();
348        $cartKey = $objCartSession->getKey();
349        // 詳細情報を取得
350        $cartItems = $objCartSession->getCartList($cartKey);
351        */
352
353        if (empty($arrItems['productsClass'])) {
[20167]354            $product =& $objProduct->getDetailAndProductsClass($product_class_id);
[20975]355            $arrItems['productsClass'] = $product;
[19860]356        }
[20975]357        $arrItems['price'] = $arrItems['productsClass']['price02'];
358        $inctax = SC_Helper_DB_Ex::sfCalcIncTax($arrItems['price']);
359        $arrItems['total_inctax'] = $inctax * $arrItems['quantity'];
[19860]360    }
361
362    /**
[19928]363     * 配送先都道府県の配列を返す.
364     */
[21016]365    function getShippingPref($is_multiple) {
[19928]366        $results = array();
[21016]367        foreach (SC_Helper_Purchase_Ex::getShippingTemp($is_multiple) as $val) {
[19928]368            $results[] = $val['shipping_pref'];
369        }
370        return $results;
371    }
372
373    /**
[19868]374     * 複数配送指定の購入かどうか.
375     *
376     * @return boolean 複数配送指定の購入の場合 true
377     */
378    function isMultiple() {
[21571]379        return count(SC_Helper_Purchase_Ex::getShippingTemp(true)) >= 2;
[19868]380    }
381
382    /**
383     * 配送情報をセッションに保存する.
[20167]384     *
[21571]385     * XXX マージする理由が不明(なんとなく便利な気はするけど)。分かる方コメントに残してください。
[20167]386     * @param array $arrSrc 配送情報の連想配列
387     * @param integer $shipping_id 配送先ID
388     * @return void
[19868]389     */
[21006]390    function saveShippingTemp($arrSrc, $shipping_id = 0) {
391        // 配送商品は引き継がない
392        unset($arrSrc['shipment_item']);
393
[21571]394        if (!isset($_SESSION['shipping'][$shipping_id])) {
395            $_SESSION['shipping'][$shipping_id] = array();
[19872]396        }
[21571]397        $_SESSION['shipping'][$shipping_id] = array_merge($_SESSION['shipping'][$shipping_id], $arrSrc);
398        $_SESSION['shipping'][$shipping_id]['shipping_id'] = $shipping_id;
[19868]399    }
400
401    /**
[19861]402     * セッションの配送情報を破棄する.
[21571]403     *
404     * @deprecated 2.12.0 から EC-CUBE 本体では使用していない。
405     * @param integer $shipping_id 配送先ID
406     * @return void
[19861]407     */
408    function unsetShippingTemp() {
[21572]409        SC_Helper_Purchase_Ex::unsetAllShippingTemp(true);
[21571]410    }
411
412    /**
413     * セッションの配送情報を全て破棄する
414     *
415     * @param bool $multiple_temp 複数お届け先の画面戻り処理用の情報も破棄するか
416     * @return void
417     */
[21572]418    static function unsetAllShippingTemp($multiple_temp = false) {
[19861]419        unset($_SESSION['shipping']);
[21571]420        if ($multiple_temp) {
421            unset($_SESSION['multiple_temp']);
422        }
[19861]423    }
424
425    /**
[21571]426     * セッションの配送情報を個別に破棄する
427     *
428     * @param integer $shipping_id 配送先ID
429     * @return void
430     */
[21572]431    static function unsetOneShippingTemp($shipping_id) {
[21571]432        unset($_SESSION['shipping'][$shipping_id]);
433    }
434
435    /**
[19860]436     * 会員情報を受注情報にコピーする.
437     *
438     * ユーザーがログインしていない場合は何もしない.
439     * 会員情報を $dest の order_* へコピーする.
440     * customer_id は強制的にコピーされる.
441     *
442     * @param array $dest コピー先の配列
443     * @param SC_Customer $objCustomer SC_Customer インスタンス
[19861]444     * @param string $prefix コピー先の接頭辞. デフォルト order
[19860]445     * @param array $keys コピー対象のキー
446     * @return void
447     */
[19861]448    function copyFromCustomer(&$dest, &$objCustomer, $prefix = 'order',
[21527]449        $keys = array('name01', 'name02', 'kana01', 'kana02',
450            'sex', 'zip01', 'zip02', 'pref', 'addr01', 'addr02',
451            'tel01', 'tel02', 'tel03', 'job', 'birth', 'email',
452        )
453    ) {
[19860]454        if ($objCustomer->isLoginSuccess(true)) {
455
456            foreach ($keys as $key) {
457                if (in_array($key, $keys)) {
458                    $dest[$prefix . '_' . $key] = $objCustomer->getValue($key);
459                }
460            }
461
[21302]462            if ((SC_Display_Ex::detectDevice() == DEVICE_TYPE_MOBILE)
463                && in_array('email', $keys)
464            ) {
[19860]465                $email_mobile = $objCustomer->getValue('email_mobile');
466                if (empty($email_mobile)) {
467                    $dest[$prefix . '_email'] = $objCustomer->getValue('email');
468                } else {
469                    $dest[$prefix . '_email'] = $email_mobile;
470                }
471            }
472
473            $dest['customer_id'] = $objCustomer->getValue('customer_id');
[21185]474            $dest['update_date'] = 'CURRENT_TIMESTAMP';
[19860]475        }
476    }
477
478    /**
479     * 受注情報を配送情報にコピーする.
480     *
481     * 受注情報($src)を $dest の order_* へコピーする.
482     *
[19920]483     * TODO 汎用的にして SC_Utils へ移動
484     *
[19860]485     * @param array $dest コピー先の配列
486     * @param array $src コピー元の配列
[21571]487     * @param array $arrKey コピー対象のキー
[19860]488     * @param string $prefix コピー先の接頭辞. デフォルト shipping
489     * @param string $src_prefix コピー元の接頭辞. デフォルト order
490     * @return void
491     */
[21571]492    function copyFromOrder(&$dest, $src, $prefix = 'shipping', $src_prefix = 'order', $arrKey = null) {
493        if (is_null($arrKey)) {
494            $arrKey = $this->arrShippingKey;
495        }
[19920]496        if (!SC_Utils_Ex::isBlank($prefix)) {
497            $prefix = $prefix . '_';
498        }
499        if (!SC_Utils_Ex::isBlank($src_prefix)) {
500            $src_prefix = $src_prefix . '_';
501        }
[21571]502        foreach ($arrKey as $key) {
503            if (isset($src[$src_prefix . $key])) {
[19920]504                $dest[$prefix . $key] = $src[$src_prefix . $key];
[19860]505            }
506        }
507    }
508
509    /**
[21571]510     * 配送情報のみ抜き出す。
511     *
512     * @param string $arrSrc 元となる配列
513     * @return void
514     */
515    function extractShipping($arrSrc) {
516        $arrKey = array();
[21589]517        foreach ($this->arrShippingKey as $key) {
[21571]518            $arrKey[] = 'shipping_' . $key;
519        }
520        return SC_Utils_Ex::sfArrayIntersectKeys($arrSrc, $arrKey);
521    }
522
523    /**
[19860]524     * 購入金額に応じた支払方法を取得する.
525     *
526     * @param integer $total 購入金額
[20069]527     * @param integer $deliv_id 配送業者ID
[19860]528     * @return array 購入金額に応じた支払方法の配列
529     */
[20069]530    function getPaymentsByPrice($total, $deliv_id) {
[19860]531
[20069]532        $arrPaymentIds = $this->getPayments($deliv_id);
[20662]533        if (SC_Utils_Ex::isBlank($arrPaymentIds)) {
534            return array();
535        }
536
[20507]537        $objQuery =& SC_Query_Ex::getSingletonInstance();
[19860]538
539        // 削除されていない支払方法を取得
[21564]540        $where = 'del_flg = 0 AND payment_id IN (' . SC_Utils_Ex::repeatStrWithSeparator('?', count($arrPaymentIds)) . ')';
[21514]541        $objQuery->setOrder('rank DESC');
542        $payments = $objQuery->select('payment_id, payment_method, rule, upper_rule, note, payment_image, charge', 'dtb_payment', $where, $arrPaymentIds);
[19860]543        foreach ($payments as $data) {
544            // 下限と上限が設定されている
545            if (strlen($data['rule']) != 0 && strlen($data['upper_rule']) != 0) {
[20662]546                if ($data['rule'] <= $total && $data['upper_rule'] >= $total) {
[19860]547                    $arrPayment[] = $data;
548                }
549            }
550            // 下限のみ設定されている
551            elseif (strlen($data['rule']) != 0) {
[21441]552                if ($data['rule'] <= $total) {
[19860]553                    $arrPayment[] = $data;
554                }
555            }
556            // 上限のみ設定されている
557            elseif (strlen($data['upper_rule']) != 0) {
[21441]558                if ($data['upper_rule'] >= $total) {
[19860]559                    $arrPayment[] = $data;
560                }
561            }
562            // いずれも設定なし
563            else {
564                $arrPayment[] = $data;
565            }
[21527]566        }
[19860]567        return $arrPayment;
568    }
569
570    /**
[21628]571     * 支払方法の詳細を取得する.
572     *
573     * @param integer $payment_id お支払い方法
574     * @return array 支払方法詳細の配列
575     */
576    function getPaymentsByPaymentsId($payment_id) {
[21664]577        $objQuery =& SC_Query_Ex::getSingletonInstance();
[21628]578        $where = 'payment_id = ? AND del_flg = 0';
579        $arrValues = array($payment_id);
580        return $objQuery->getRow('*', 'dtb_payment', $where, $arrValues);
581    }
582
583    /**
[19923]584     * お届け日一覧を取得する.
[19860]585     */
[19923]586    function getDelivDate(&$objCartSess, $productTypeId) {
587        $cartList = $objCartSess->getCartList($productTypeId);
588        $delivDateIds = array();
589        foreach ($cartList as $item) {
590            $delivDateIds[] = $item['productsClass']['deliv_date_id'];
591        }
592        $max_date = max($delivDateIds);
593        //発送目安
[21441]594        switch ($max_date) {
[21526]595            //即日発送
596            case '1':
597                $start_day = 1;
598                break;
599                //1-2日後
600            case '2':
601                $start_day = 3;
602                break;
603                //3-4日後
604            case '3':
605                $start_day = 5;
606                break;
607                //1週間以内
608            case '4':
609                $start_day = 8;
610                break;
611                //2週間以内
612            case '5':
613                $start_day = 15;
614                break;
615                //3週間以内
616            case '6':
617                $start_day = 22;
618                break;
619                //1ヶ月以内
620            case '7':
621                $start_day = 32;
622                break;
623                //2ヶ月以降
624            case '8':
625                $start_day = 62;
626                break;
627                //お取り寄せ(商品入荷後)
628            case '9':
629                $start_day = '';
630                break;
631            default:
632                //お届け日が設定されていない場合
633                $start_day = '';
634                break;
[19923]635        }
636        //お届け可能日のスタート値から、お届け日の配列を取得する
637        $arrDelivDate = $this->getDateArray($start_day, DELIV_DATE_END_MAX);
638        return $arrDelivDate;
[19860]639    }
640
641    /**
[19923]642     * お届け可能日のスタート値から, お届け日の配列を取得する.
643     */
644    function getDateArray($start_day, $end_day) {
645        $masterData = new SC_DB_MasterData();
[21481]646        $arrWDAY = $masterData->getMasterData('mtb_wday');
[19923]647        //お届け可能日のスタート値がセットされていれば
[21441]648        if ($start_day >= 1) {
[19923]649            $now_time = time();
650            $max_day = $start_day + $end_day;
651            // 集計
652            for ($i = $start_day; $i < $max_day; $i++) {
653                // 基本時間から日数を追加していく
654                $tmp_time = $now_time + ($i * 24 * 3600);
[21514]655                list($y, $m, $d, $w) = explode(' ', date('Y m d w', $tmp_time));
656                $val = sprintf('%04d/%02d/%02d(%s)', $y, $m, $d, $arrWDAY[$w]);
[19923]657                $arrDate[$val] = $val;
658            }
659        } else {
660            $arrDate = false;
661        }
662        return $arrDate;
663    }
664
665    /**
[20112]666     * 配送業者IDからお届け時間の配列を取得する.
667     *
668     * @param integer $deliv_id 配送業者ID
669     * @return array お届け時間の配列
[19860]670     */
[20112]671    function getDelivTime($deliv_id) {
[20507]672        $objQuery =& SC_Query_Ex::getSingletonInstance();
[20112]673        $objQuery->setOrder('time_id');
674        $results = $objQuery->select('time_id, deliv_time',
675                                     'dtb_delivtime',
676                                     'deliv_id = ?', array($deliv_id));
677        $arrDelivTime = array();
678        foreach ($results as $val) {
679            $arrDelivTime[$val['time_id']] = $val['deliv_time'];
680        }
681        return $arrDelivTime;
[19860]682    }
683
684    /**
[20069]685     * 商品種別ID から配送業者を取得する.
[20112]686     *
687     * @param integer $product_type_id 商品種別ID
688     * @return array 配送業者の配列
[19861]689     */
[20112]690    function getDeliv($product_type_id) {
[20507]691        $objQuery =& SC_Query_Ex::getSingletonInstance();
[20650]692        $objQuery->setOrder('rank DESC');
693        return $objQuery->select('*', 'dtb_deliv', 'product_type_id = ? AND del_flg = 0',
[20112]694                                 array($product_type_id));
[19861]695    }
696
697    /**
[20112]698     * 配送業者ID から, 有効な支払方法IDを取得する.
[20069]699     *
700     * @param integer $deliv_id 配送業者ID
701     * @return array 有効な支払方法IDの配列
702     */
703    function getPayments($deliv_id) {
[20507]704        $objQuery =& SC_Query_Ex::getSingletonInstance();
[20112]705        $objQuery->setOrder('rank');
706        return $objQuery->getCol('payment_id', 'dtb_payment_options',
707                                 'deliv_id = ?',
[20069]708                                 array($deliv_id), MDB2_FETCHMODE_ORDERED);
709    }
710
711    /**
[20167]712     * 配送情報の登録を行う.
713     *
714     * $arrParam のうち, dtb_shipping テーブルに存在するカラムのみを登録する.
715     *
[20180]716     * TODO UPDATE/INSERT にする
717     *
[20167]718     * @param integer $order_id 受注ID
719     * @param array $arrParams 配送情報の連想配列
[20180]720     * @param boolean $convert_shipping_date yyyy/mm/dd(EEE) 形式の配送日付を変換する場合 true
[20167]721     * @return void
[19861]722     */
[20180]723    function registerShipping($order_id, $arrParams, $convert_shipping_date = true) {
[20507]724        $objQuery =& SC_Query_Ex::getSingletonInstance();
[20167]725        $table = 'dtb_shipping';
[20180]726        $where = 'order_id = ?';
727        $objQuery->delete($table, $where, array($order_id));
[19861]728
[21006]729        foreach ($arrParams as $key => $arrShipping) {
[19861]730
[20167]731            $arrValues = $objQuery->extractOnlyColsOf($table, $arrShipping);
[19861]732
[19926]733            // 配送日付を timestamp に変換
[20180]734            if (!SC_Utils_Ex::isBlank($arrValues['shipping_date'])
735                && $convert_shipping_date) {
[21481]736                $d = mb_strcut($arrValues['shipping_date'], 0, 10);
[21514]737                $arrDate = explode('/', $d);
[19926]738                $ts = mktime(0, 0, 0, $arrDate[1], $arrDate[2], $arrDate[0]);
[21514]739                $arrValues['shipping_date'] = date('Y-m-d', $ts);
[19926]740            }
741
[20697]742            // 非会員購入の場合は shipping_id が存在しない
743            if (!isset($arrValues['shipping_id'])) {
744                $arrValues['shipping_id'] = $key;
745            }
[20167]746            $arrValues['order_id'] = $order_id;
[21185]747            $arrValues['create_date'] = 'CURRENT_TIMESTAMP';
748            $arrValues['update_date'] = 'CURRENT_TIMESTAMP';
[20167]749            $objQuery->insert($table, $arrValues);
[19861]750        }
751    }
752
753    /**
[19862]754     * 配送商品を登録する.
[20167]755     *
756     * @param integer $order_id 受注ID
757     * @param integer $shipping_id 配送先ID
758     * @param array $arrParams 配送商品の配列
759     * @return void
[19862]760     */
[20180]761    function registerShipmentItem($order_id, $shipping_id, $arrParams) {
[20507]762        $objQuery =& SC_Query_Ex::getSingletonInstance();
[20180]763        $table = 'dtb_shipment_item';
764        $where = 'order_id = ? AND shipping_id = ?';
765        $objQuery->delete($table, $where, array($order_id, $shipping_id));
766
[20487]767        $objProduct = new SC_Product_Ex();
[20180]768        foreach ($arrParams as $arrValues) {
769            if (SC_Utils_Ex::isBlank($arrValues['product_class_id'])) {
770                continue;
771            }
772            $d = $objProduct->getDetailAndProductsClass($arrValues['product_class_id']);
773            $name = SC_Utils_Ex::isBlank($arrValues['product_name'])
774                ? $d['name']
775                : $arrValues['product_name'];
776
777            $code = SC_Utils_Ex::isBlank($arrValues['product_code'])
[20655]778                ? $d['product_code']
[20180]779                : $arrValues['product_code'];
780
781            $cname1 = SC_Utils_Ex::isBlank($arrValues['classcategory_name1'])
782                ? $d['classcategory_name1']
783                : $arrValues['classcategory_name1'];
784
785            $cname2 = SC_Utils_Ex::isBlank($arrValues['classcategory_name2'])
786                ? $d['classcategory_name2']
787                : $arrValues['classcategory_name2'];
788
789            $price = SC_Utils_Ex::isBlank($arrValues['price'])
[20975]790                ? $d['price']
[20180]791                : $arrValues['price'];
792
793            $arrValues['order_id'] = $order_id;
794            $arrValues['shipping_id'] = $shipping_id;
795            $arrValues['product_name'] = $name;
796            $arrValues['product_code'] = $code;
797            $arrValues['classcategory_name1'] = $cname1;
798            $arrValues['classcategory_name2'] = $cname2;
799            $arrValues['price'] = $price;
800
801            $arrExtractValues = $objQuery->extractOnlyColsOf($table, $arrValues);
802            $objQuery->insert($table, $arrExtractValues);
[19862]803        }
804    }
805
806    /**
[20180]807     * 受注登録を完了する.
[18860]808     *
[18871]809     * 引数の受注情報を受注テーブル及び受注詳細テーブルに登録する.
[20771]810     * 登録後, 受注一時テーブルに削除フラグを立てる.
[18871]811     *
812     * @param array $orderParams 登録する受注情報の配列
813     * @param SC_CartSession $objCartSession カート情報のインスタンス
814     * @param integer $cartKey 登録を行うカート情報のキー
815     * @param integer 受注ID
[18860]816     */
[20180]817    function registerOrderComplete($orderParams, &$objCartSession, $cartKey) {
[20507]818        $objQuery =& SC_Query_Ex::getSingletonInstance();
[18860]819
820        // 不要な変数を unset
821        $unsets = array('mailmaga_flg', 'deliv_check', 'point_check', 'password',
822                        'reminder', 'reminder_answer', 'mail_flag', 'session');
823        foreach ($unsets as $unset) {
824            unset($orderParams[$unset]);
825        }
826
[21182]827        // 対応状況の指定が無い場合は新規受付
[21441]828        if (SC_Utils_Ex::isBlank($orderParams['status'])) {
[18860]829            $orderParams['status'] = ORDER_NEW;
830        }
831
[21185]832        $orderParams['create_date'] = 'CURRENT_TIMESTAMP';
833        $orderParams['update_date'] = 'CURRENT_TIMESTAMP';
[18860]834
[20180]835        $this->registerOrder($orderParams['order_id'], $orderParams);
[18860]836
837        // 詳細情報を取得
838        $cartItems = $objCartSession->getCartList($cartKey);
839
[20180]840        // 詳細情報を生成
[20487]841        $objProduct = new SC_Product_Ex();
[20180]842        $i = 0;
[18860]843        foreach ($cartItems as $item) {
844            $p =& $item['productsClass'];
[20180]845            $arrDetail[$i]['order_id'] = $orderParams['order_id'];
846            $arrDetail[$i]['product_id'] = $p['product_id'];
847            $arrDetail[$i]['product_class_id'] = $p['product_class_id'];
848            $arrDetail[$i]['product_name'] = $p['name'];
849            $arrDetail[$i]['product_code'] = $p['product_code'];
850            $arrDetail[$i]['classcategory_name1'] = $p['classcategory_name1'];
851            $arrDetail[$i]['classcategory_name2'] = $p['classcategory_name2'];
852            $arrDetail[$i]['point_rate'] = $item['point_rate'];
853            $arrDetail[$i]['price'] = $item['price'];
854            $arrDetail[$i]['quantity'] = $item['quantity'];
[18860]855
856            // 在庫の減少処理
857            if (!$objProduct->reduceStock($p['product_class_id'], $item['quantity'])) {
858                $objQuery->rollback();
[21514]859                SC_Utils_Ex::sfDispSiteError(SOLD_OUT, '', true);
[18860]860            }
[20180]861            $i++;
[18860]862        }
[20180]863        $this->registerOrderDetail($orderParams['order_id'], $arrDetail);
[18860]864
[21481]865        $objQuery->update('dtb_order_temp', array('del_flg' => 1),
[21514]866                          'order_temp_id = ?',
[20448]867                          array(SC_SiteSession_Ex::getUniqId()));
[21420]868
869
870
[18860]871        return $orderParams['order_id'];
872    }
873
874    /**
[20180]875     * 受注情報を登録する.
876     *
877     * 既に受注IDが存在する場合は, 受注情報を更新する.
878     * 引数の受注IDが, 空白又は null の場合は, 新しく受注IDを発行して登録する.
879     *
880     * @param integer $order_id 受注ID
881     * @param array $arrParams 受注情報の連想配列
882     * @return integer 受注ID
883     */
884    function registerOrder($order_id, $arrParams) {
885        $table = 'dtb_order';
886        $where = 'order_id = ?';
[20507]887        $objQuery = SC_Query_Ex::getSingletonInstance();
[20180]888        $arrValues = $objQuery->extractOnlyColsOf($table, $arrParams);
889
[21376]890        $exists = $objQuery->exists($table, $where, array($order_id));
891        if ($exists) {
[20180]892
893            $this->sfUpdateOrderStatus($order_id, $arrValues['status'],
894                                       $arrValues['add_point'],
895                                       $arrValues['use_point'],
896                                       $arrValues);
897            $this->sfUpdateOrderNameCol($order_id);
898
[21185]899            $arrValues['update_date'] = 'CURRENT_TIMESTAMP';
[20180]900            $objQuery->update($table, $arrValues, $where, array($order_id));
901        } else {
902            if (SC_Utils_Ex::isBlank($order_id)) {
903                $order_id = $objQuery->nextVal('dtb_order_order_id');
904            }
[20660]905            /*
[21182]906             * 新規受付の場合は対応状況 null で insert し,
[20660]907             * sfUpdateOrderStatus で ORDER_NEW に変更する.
908             */
909            $status = $arrValues['status'];
910            $arrValues['status'] = null;
[20180]911            $arrValues['order_id'] = $order_id;
912            $arrValues['customer_id'] =
913                    SC_Utils_Ex::isBlank($arrValues['customer_id'])
914                    ? 0 : $arrValues['customer_id'];
[21185]915            $arrValues['create_date'] = 'CURRENT_TIMESTAMP';
916            $arrValues['update_date'] = 'CURRENT_TIMESTAMP';
[20180]917            $objQuery->insert($table, $arrValues);
918
[20660]919            $this->sfUpdateOrderStatus($order_id, $status,
[20180]920                                       $arrValues['add_point'],
921                                       $arrValues['use_point'],
922                                       $arrValues);
923            $this->sfUpdateOrderNameCol($order_id);
924
925        }
926        return $order_id;
927    }
928
929    /**
930     * 受注詳細情報を登録する.
931     *
932     * 既に, 該当の受注が存在する場合は, 受注情報を削除し, 登録する.
933     *
934     * @param integer $order_id 受注ID
935     * @param array $arrParams 受注情報の連想配列
936     * @return void
937     */
938    function registerOrderDetail($order_id, $arrParams) {
939        $table = 'dtb_order_detail';
940        $where = 'order_id = ?';
[20507]941        $objQuery = SC_Query_Ex::getSingletonInstance();
[20180]942
943        $objQuery->delete($table, $where, array($order_id));
944        foreach ($arrParams as $arrDetail) {
945            $arrValues = $objQuery->extractOnlyColsOf($table, $arrDetail);
[20435]946            $arrValues['order_detail_id'] = $objQuery->nextVal('dtb_order_detail_order_detail_id');
[20180]947            $arrValues['order_id'] = $order_id;
948            $objQuery->insert($table, $arrValues);
949        }
950    }
951
952    /**
[20167]953     * 受注情報を取得する.
954     *
955     * @param integer $order_id 受注ID
[21419]956     * @param integer $customer_id 会員ID
[20167]957     * @return array 受注情報の配列
958     */
959    function getOrder($order_id, $customer_id = null) {
[20507]960        $objQuery =& SC_Query_Ex::getSingletonInstance();
[20167]961        $where = 'order_id = ?';
962        $arrValues = array($order_id);
963        if (!SC_Utils_Ex::isBlank($customer_id)) {
964            $where .= ' AND customer_id = ?';
965            $arrValues[] = $customer_id;
966        }
967        return $objQuery->getRow('*', 'dtb_order', $where, $arrValues);
968    }
969
970    /**
971     * 受注詳細を取得する.
972     *
973     * @param integer $order_id 受注ID
[21182]974     * @param boolean $has_order_status 対応状況, 入金日も含める場合 true
[20167]975     * @return array 受注詳細の配列
976     */
[20180]977    function getOrderDetail($order_id, $has_order_status = true) {
[20507]978        $objQuery =& SC_Query_Ex::getSingletonInstance();
[20167]979        $dbFactory  = SC_DB_DBFactory_Ex::getInstance();
980        $col = <<< __EOS__
981            T3.product_id,
982            T3.product_class_id as product_class_id,
983            T3.product_type_id AS product_type_id,
984            T2.product_code,
985            T2.product_name,
986            T2.classcategory_name1 AS classcategory_name1,
987            T2.classcategory_name2 AS classcategory_name2,
988            T2.price,
989            T2.quantity,
990            T2.point_rate,
[20180]991__EOS__;
992        if ($has_order_status) {
993            $col .= 'T1.status AS status, T1.payment_date AS payment_date,';
994
995        }
996        $col .= <<< __EOS__
[21527]997            CASE WHEN
998                EXISTS(
[20959]999                    SELECT * FROM dtb_products
[21527]1000                    WHERE product_id = T3.product_id
1001                        AND del_flg = 0
1002                        AND status = 1
1003                )
1004                THEN '1'
1005                ELSE '0'
1006            END AS enable,
[20167]1007__EOS__;
1008        $col .= $dbFactory->getDownloadableDaysWhereSql('T1') . ' AS effective';
1009        $from = <<< __EOS__
[21527]1010            dtb_order T1
1011            JOIN dtb_order_detail T2
1012                ON T1.order_id = T2.order_id
[21251]1013            LEFT JOIN dtb_products_class T3
[21527]1014                ON T2.product_class_id = T3.product_class_id
[20167]1015__EOS__;
[20435]1016        $objQuery->setOrder('T2.order_detail_id');
[20167]1017        return $objQuery->select($col, $from, 'T1.order_id = ?', array($order_id));
1018    }
1019
1020    /**
[20888]1021     * ダウンロード可能フラグを, 受注詳細に設定する.
1022     *
1023     * ダウンロード可能と判断されるのは, 以下の通り.
1024     *
1025     * 1. ダウンロード可能期限が期限内かつ, 入金日が入力されている
1026     * 2. 販売価格が 0 円である
1027     *
1028     * 受注詳細行には, is_downloadable という真偽値が設定される.
1029     * @param array 受注詳細の配列
1030     * @return void
1031     */
1032    function setDownloadableFlgTo(&$arrOrderDetail) {
1033        foreach (array_keys($arrOrderDetail) as $key) {
1034            // 販売価格が 0 円
1035            if ($arrOrderDetail[$key]['price'] == '0') {
1036                $arrOrderDetail[$key]['is_downloadable'] = true;
1037            }
1038            // ダウンロード期限内かつ, 入金日あり
1039            elseif ($arrOrderDetail[$key]['effective'] == '1'
1040                    && !SC_Utils_Ex::isBlank($arrOrderDetail[$key]['payment_date'])) {
1041                $arrOrderDetail[$key]['is_downloadable'] = true;
1042            } else {
1043                $arrOrderDetail[$key]['is_downloadable'] = false;
1044            }
1045        }
1046    }
1047
1048    /**
[20167]1049     * 配送情報を取得する.
1050     *
1051     * @param integer $order_id 受注ID
1052     * @param boolean $has_items 結果に配送商品も含める場合 true
1053     * @return array 配送情報の配列
1054     */
1055    function getShippings($order_id, $has_items = true) {
[20507]1056        $objQuery =& SC_Query_Ex::getSingletonInstance();
[20167]1057        $arrResults = array();
1058        $objQuery->setOrder('shipping_id');
[21515]1059        $arrShippings = $objQuery->select('*', 'dtb_shipping', 'order_id = ?',
[20167]1060                                          array($order_id));
1061        // shipping_id ごとの配列を生成する
1062        foreach ($arrShippings as $shipping) {
1063            foreach ($shipping as $key => $val) {
1064                $arrResults[$shipping['shipping_id']][$key] = $val;
1065            }
1066        }
1067
1068        if ($has_items) {
[20487]1069            $objProduct = new SC_Product_Ex();
[20167]1070            foreach (array_keys($arrResults) as $shipping_id) {
1071                $arrResults[$shipping_id]['shipment_item']
1072                        =& $this->getShipmentItems($order_id, $shipping_id);
1073            }
1074        }
1075        return $arrResults;
1076    }
1077
1078    /**
1079     * 配送商品を取得する.
1080     *
1081     * @param integer $order_id 受注ID
1082     * @param integer $shipping_id 配送先ID
1083     * @param boolean $has_detail 商品詳細も取得する場合 true
1084     * @return array 商品規格IDをキーにした配送商品の配列
1085     */
1086    function getShipmentItems($order_id, $shipping_id, $has_detail = true) {
[20507]1087        $objQuery =& SC_Query_Ex::getSingletonInstance();
[20487]1088        $objProduct = new SC_Product_Ex();
[20167]1089        $arrResults = array();
[21514]1090        $arrItems = $objQuery->select('*', 'dtb_shipment_item',
1091                                      'order_id = ? AND shipping_id = ?',
[20167]1092                                      array($order_id, $shipping_id));
1093
1094        foreach ($arrItems as $key => $arrItem) {
1095            $product_class_id = $arrItem['product_class_id'];
1096
1097            foreach ($arrItem as $detailKey => $detailVal) {
[20180]1098                $arrResults[$key][$detailKey] = $detailVal;
[20167]1099            }
1100            // 商品詳細を関連づける
1101            if ($has_detail) {
[20180]1102                $arrResults[$key]['productsClass']
[20167]1103                    =& $objProduct->getDetailAndProductsClass($product_class_id);
1104            }
1105        }
1106        return $arrResults;
1107    }
1108
1109    /**
[18860]1110     * 受注完了メールを送信する.
1111     *
1112     * HTTP_USER_AGENT の種別により, 携帯電話の場合は携帯用の文面,
1113     * PC の場合は PC 用の文面でメールを送信する.
1114     *
1115     * @param integer $orderId 受注ID
1116     * @return void
1117     */
1118    function sendOrderMail($orderId) {
1119        $mailHelper = new SC_Helper_Mail_Ex();
[21302]1120        $template_id =
1121            SC_Display_Ex::detectDevice() == DEVICE_TYPE_MOBILE ? 2 : 1;
1122        $mailHelper->sfSendOrderMail($orderId, $template_id);
[18860]1123    }
[20180]1124
1125    /**
1126     * 受注.対応状況の更新
1127     *
[20660]1128     * 必ず呼び出し元でトランザクションブロックを開いておくこと。
[20180]1129     *
1130     * @param integer $orderId 注文番号
1131     * @param integer|null $newStatus 対応状況 (null=変更無し)
1132     * @param integer|null $newAddPoint 加算ポイント (null=変更無し)
1133     * @param integer|null $newUsePoint 使用ポイント (null=変更無し)
[20970]1134     * @param array $sqlval 更新後の値をリファレンスさせるためのパラメーター
[20180]1135     * @return void
1136     */
1137    function sfUpdateOrderStatus($orderId, $newStatus = null, $newAddPoint = null, $newUsePoint = null, &$sqlval) {
[20507]1138        $objQuery =& SC_Query_Ex::getSingletonInstance();
[20180]1139        $arrOrderOld = $objQuery->getRow('status, add_point, use_point, customer_id', 'dtb_order', 'order_id = ?', array($orderId));
1140
1141        // 対応状況が変更無しの場合、DB値を引き継ぐ
1142        if (is_null($newStatus)) {
1143            $newStatus = $arrOrderOld['status'];
1144        }
1145
1146        // 使用ポイント、DB値を引き継ぐ
1147        if (is_null($newUsePoint)) {
1148            $newUsePoint = $arrOrderOld['use_point'];
1149        }
1150
1151        // 加算ポイント、DB値を引き継ぐ
1152        if (is_null($newAddPoint)) {
1153            $newAddPoint = $arrOrderOld['add_point'];
1154        }
1155
1156        if (USE_POINT !== false) {
[20953]1157            // 会員.ポイントの加減値
[20180]1158            $addCustomerPoint = 0;
1159
1160            // ▼使用ポイント
1161            // 変更前の対応状況が利用対象の場合、変更前の使用ポイント分を戻す
[20660]1162            if ($this->isUsePoint($arrOrderOld['status'])) {
[20180]1163                $addCustomerPoint += $arrOrderOld['use_point'];
1164            }
1165
1166            // 変更後の対応状況が利用対象の場合、変更後の使用ポイント分を引く
[20660]1167            if ($this->isUsePoint($newStatus)) {
[20180]1168                $addCustomerPoint -= $newUsePoint;
1169            }
[20660]1170
[20180]1171            // ▲使用ポイント
1172
1173            // ▼加算ポイント
1174            // 変更前の対応状況が加算対象の場合、変更前の加算ポイント分を戻す
[20660]1175            if ($this->isAddPoint($arrOrderOld['status'])) {
[20180]1176                $addCustomerPoint -= $arrOrderOld['add_point'];
1177            }
1178
1179            // 変更後の対応状況が加算対象の場合、変更後の加算ポイント分を足す
[20660]1180            if ($this->isAddPoint($newStatus)) {
[20180]1181                $addCustomerPoint += $newAddPoint;
1182            }
1183            // ▲加算ポイント
1184
1185            if ($addCustomerPoint != 0) {
[20953]1186                // ▼会員テーブルの更新
[21185]1187                $objQuery->update('dtb_customer', array('update_date' => 'CURRENT_TIMESTAMP'),
[21115]1188                                  'customer_id = ?', array($arrOrderOld['customer_id']),
1189                                  array('point' => 'point + ?'), array($addCustomerPoint));
[20953]1190                // ▲会員テーブルの更新
[20180]1191
[20953]1192                // 会員.ポイントをマイナスした場合、
[20180]1193                if ($addCustomerPoint < 0) {
1194                    $sql = 'SELECT point FROM dtb_customer WHERE customer_id = ?';
1195                    $point = $objQuery->getOne($sql, array($arrOrderOld['customer_id']));
[20953]1196                    // 変更後の会員.ポイントがマイナスの場合、
[20180]1197                    if ($point < 0) {
1198                        // ロールバック
1199                        $objQuery->rollback();
1200                        // エラー
1201                        SC_Utils_Ex::sfDispSiteError(LACK_POINT);
1202                    }
1203                }
1204            }
1205        }
1206
1207        // ▼受注テーブルの更新
1208        if (empty($sqlval)) {
1209            $sqlval = array();
1210        }
1211
1212        if (USE_POINT !== false) {
1213            $sqlval['add_point'] = $newAddPoint;
1214            $sqlval['use_point'] = $newUsePoint;
1215        }
[21182]1216        // 対応状況が発送済みに変更の場合、発送日を更新
[20180]1217        if ($arrOrderOld['status'] != ORDER_DELIV && $newStatus == ORDER_DELIV) {
[21185]1218            $sqlval['commit_date'] = 'CURRENT_TIMESTAMP';
[20180]1219        }
[21182]1220        // 対応状況が入金済みに変更の場合、入金日を更新
[20180]1221        elseif ($arrOrderOld['status'] != ORDER_PRE_END && $newStatus == ORDER_PRE_END) {
[21185]1222            $sqlval['payment_date'] = 'CURRENT_TIMESTAMP';
[20180]1223        }
1224
1225        $sqlval['status'] = $newStatus;
[21185]1226        $sqlval['update_date'] = 'CURRENT_TIMESTAMP';
[20180]1227
[20660]1228        $dest = $objQuery->extractOnlyColsOf('dtb_order', $sqlval);
[20180]1229        $objQuery->update('dtb_order', $dest, 'order_id = ?', array($orderId));
1230        // ▲受注テーブルの更新
[21259]1231
1232        //会員情報の最終購入日、購入合計を更新
[21441]1233        if ($arrOrderOld['customer_id'] > 0 and $arrOrderOld['status'] != $newStatus) {
[21259]1234            SC_Customer_Ex::updateOrderSummary($arrOrderOld['customer_id']);
1235        }
[20180]1236    }
1237
1238    /**
1239     * 受注の名称列を更新する
1240     *
1241     * @param integer $order_id 更新対象の注文番号
1242     * @param boolean $temp_table 更新対象は「受注_Temp」か
1243     * @static
1244     */
1245    function sfUpdateOrderNameCol($order_id, $temp_table = false) {
[20507]1246        $objQuery =& SC_Query_Ex::getSingletonInstance();
[20180]1247
1248        if ($temp_table) {
1249            $tgt_table = 'dtb_order_temp';
1250            $sql_where = 'order_temp_id = ?';
1251        } else {
1252            $tgt_table = 'dtb_order';
1253            $sql_where = 'order_id = ?';
1254
[21551]1255            $sql_sub = <<< __EOS__
1256                SELECT deliv_time
1257                FROM dtb_delivtime
1258                WHERE time_id = dtb_shipping.time_id
1259                    AND deliv_id = (SELECT dtb_order.deliv_id FROM dtb_order WHERE order_id = dtb_shipping.order_id)
1260__EOS__;
[20180]1261            $objQuery->update('dtb_shipping', array(),
1262                              $sql_where,
1263                              array($order_id),
[21551]1264                              array('shipping_time' => "($sql_sub)"));
[20180]1265
1266        }
1267
1268        $objQuery->update($tgt_table, array(),
1269                          $sql_where,
1270                          array($order_id),
1271                          array('payment_method' =>
[21514]1272                                '(SELECT payment_method FROM dtb_payment WHERE payment_id = ' . $tgt_table . '.payment_id)'));
[20180]1273    }
[20660]1274
1275    /**
1276     * ポイント使用するかの判定
1277     *
1278     * $status が null の場合は false を返す.
1279     *
1280     * @param integer $status 対応状況
[20953]1281     * @return boolean 使用するか(会員テーブルから減算するか)
[20660]1282     */
1283    function isUsePoint($status) {
1284        if ($status == null) {
1285            return false;
1286        }
1287        switch ($status) {
1288            case ORDER_CANCEL:      // キャンセル
1289                return false;
1290            default:
1291                break;
1292        }
1293
1294        return true;
1295    }
1296
1297    /**
1298     * ポイント加算するかの判定
1299     *
1300     * @param integer $status 対応状況
1301     * @return boolean 加算するか
1302     */
1303    function isAddPoint($status) {
1304        switch ($status) {
1305            case ORDER_NEW:         // 新規注文
1306            case ORDER_PAY_WAIT:    // 入金待ち
1307            case ORDER_PRE_END:     // 入金済み
1308            case ORDER_CANCEL:      // キャンセル
1309            case ORDER_BACK_ORDER:  // 取り寄せ中
1310                return false;
1311
1312            case ORDER_DELIV:       // 発送済み
1313                return true;
1314
1315            default:
1316                break;
1317        }
1318
1319        return false;
1320    }
[20771]1321
1322    /**
1323     * セッションに保持している情報を破棄する.
1324     *
1325     * 通常、受注処理(completeOrder)完了後に呼び出され、
1326     * セッション情報を破棄する.
1327     *
1328     * 決済モジュール画面から確認画面に「戻る」場合を考慮し、
1329     * セッション情報を破棄しないカスタマイズを、モジュール側で
1330     * 加える機会を与える.
1331     *
1332     * @param integer $orderId 注文番号
1333     * @param SC_CartSession $objCartSession カート情報のインスタンス
1334     * @param SC_Customer $objCustomer SC_Customer インスタンス
1335     * @param integer $cartKey 登録を行うカート情報のキー
1336     */
1337    function cleanupSession($orderId, &$objCartSession, &$objCustomer, $cartKey) {
[21016]1338        // カートの内容を削除する.
[20771]1339        $objCartSession->delAllProducts($cartKey);
1340        SC_SiteSession_Ex::unsetUniqId();
1341
1342        // セッションの配送情報を破棄する.
[21571]1343        $this->unsetAllShippingTemp(true);
[20771]1344        $objCustomer->updateSession();
1345    }
[21571]1346
1347    /**
1348     * 単一配送指定用に配送商品を設定する
1349     *
1350     * @param SC_CartSession $objCartSession カート情報のインスタンス
1351     * @param integer $shipping_id 配送先ID
1352     * @return void
1353     */
1354    function setShipmentItemTempForSole(&$objCartSession, $shipping_id = 0) {
1355        $objCartSess = new SC_CartSession_Ex();
1356
1357        $this->clearShipmentItemTemp();
1358
1359        $arrCartList =& $objCartSession->getCartList($objCartSess->getKey());
1360        foreach ($arrCartList as $arrCartRow) {
1361            if ($arrCartRow['quantity'] == 0) continue;
1362            $this->setShipmentItemTemp($shipping_id, $arrCartRow['id'], $arrCartRow['quantity']);
1363        }
1364    }
[18860]1365}
Note: See TracBrowser for help on using the repository browser.