source: branches/version-2_13-dev/data/class/helper/SC_Helper_Purchase.php @ 23605

Revision 23605, 54.2 KB checked in by kimoto, 10 years ago (diff)

#2448 typo修正・ソース整形・ソースコメントの改善 for 2.13.3

Scrutinizer Auto-Fixes

This patch was automatically generated as part of the following inspection:
 https://scrutinizer-ci.com/g/nobuhiko/EC-CUBE/inspections/d8722894-69a6-4b1b-898d-43618035c60d

Enabled analysis tools:

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