source: branches/version-2_5-dev/data/class/helper/SC_Helper_Purchase.php @ 20771

Revision 20771, 41.9 KB checked in by takashi, 13 years ago (diff)

決済モジュール画面から確認画面に「戻る」場合を実装する為の変更

  • 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-2011 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     * 受注を完了する.
38     *
39     * 下記のフローで受注を完了する.
40     *
41     * 1. トランザクションを開始する
42     * 2. カートの内容を検証する.
43     * 3. 受注一時テーブルから受注データを読み込む
44     * 4. ユーザーがログインしている場合はその他の発送先へ登録する
45     * 5. 受注データを受注テーブルへ登録する
46     * 6. トランザクションをコミットする
47     *
48     * 実行中に, 何らかのエラーが発生した場合, 処理を中止しエラーページへ遷移する
49     *
50     * 決済モジュールを使用する場合は受注ステータスを「決済処理中」に設定し,
51     * 決済完了後「新規受付」に変更すること
52     *
53     * @param integer $orderStatus 受注処理を完了する際に設定する受注ステータス
54     * @return void
55     */
56    function completeOrder($orderStatus = ORDER_NEW) {
57        $objQuery =& SC_Query_Ex::getSingletonInstance();
58        $objSiteSession = new SC_SiteSession_Ex();
59        $objCartSession = new SC_CartSession_Ex();
60        $objCustomer = new SC_Customer_Ex();
61        $customerId = $objCustomer->getValue('customer_id');
62
63        $objQuery->begin();
64        if (!$objSiteSession->isPrePage()) {
65            SC_Utils_Ex::sfDispSiteError(PAGE_ERROR, $objSiteSession);
66        }
67
68        $uniqId = $objSiteSession->getUniqId();
69        $this->verifyChangeCart($uniqId, $objCartSession);
70
71        $orderTemp = $this->getOrderTemp($uniqId);
72
73        $orderTemp['status'] = $orderStatus;
74        $cartkey = $objCartSession->getKey();
75        $orderId = $this->registerOrderComplete($orderTemp, $objCartSession,
76                                                $cartkey);
77        $shippingTemp =& $this->getShippingTemp();
78        if (count($shippingTemp) > 1) {
79            foreach ($shippingTemp as $shippingId => $val) {
80                $this->registerShipmentItem($orderId, $shippingId,
81                                            $val['shipment_item']);
82            }
83        }
84
85        $this->registerShipping($orderId, $shippingTemp);
86        $objQuery->commit();
87        $this->cleanupSession($orderId, $objCartSession, $objCustomer, $cartkey);
88    }
89
90    /**
91     * カートに変化が無いか検証する.
92     *
93     * ユニークIDとセッションのユニークIDを比較し, 異なる場合は
94     * エラー画面を表示する.
95     *
96     * カートが空の場合, 購入ボタン押下後にカートが変更された場合は
97     * カート画面へ遷移する.
98     *
99     * @param string $uniqId ユニークID
100     * @param SC_CartSession $objCartSession
101     * @return void
102     */
103    function verifyChangeCart($uniqId, &$objCartSession) {
104        $cartKeys = $objCartSession->getKeys();
105
106        // カート内が空でないか
107        if (SC_Utils_Ex::isBlank($cartKeys)) {
108            SC_Response_Ex::sendRedirect(CART_URLPATH);
109            exit;
110        }
111
112        foreach ($cartKeys as $cartKey) {
113            // 初回のみカートの内容を保存
114            $objCartSession->saveCurrentCart($uniqId, $cartKey);
115
116            /*
117             * POSTのユニークIDとセッションのユニークIDを比較
118             *(ユニークIDがPOSTされていない場合はスルー)
119             */
120            if(!SC_SiteSession_Ex::checkUniqId()) {
121                SC_Utils_Ex::sfDispSiteError(CANCEL_PURCHASE);
122                exit;
123            }
124
125            // 購入ボタンを押してから変化がないか
126            $quantity = $objCartSession->getTotalQuantity($cartKey);
127            if($objCartSession->checkChangeCart($cartKey) || !($quantity > 0)) {
128                SC_Response_Ex::sendRedirect(CART_URLPATH);
129                exit;
130            }
131        }
132    }
133
134    /**
135     * 受注一時情報を取得する.
136     *
137     * @param integer $uniqId 受注一時情報ID
138     * @return array 受注一時情報の配列
139     */
140    function getOrderTemp($uniqId) {
141        $objQuery =& SC_Query_Ex::getSingletonInstance();
142        return $objQuery->getRow("*", "dtb_order_temp", "order_temp_id = ?",
143                                 array($uniqId));
144    }
145
146    /**
147     * 受注一時情報を保存する.
148     *
149     * 既存のデータが存在しない場合は新規保存. 存在する場合は更新する.
150     * 既存のデータが存在せず, ユーザーがログインしている場合は,
151     * 会員情報をコピーする.
152     *
153     * @param integer $uniqId 受注一時情報ID
154     * @param array $params 登録する受注情報の配列
155     * @param SC_Customer $objCustomer SC_Customer インスタンス
156     * @return array void
157     */
158    function saveOrderTemp($uniqId, $params, &$objCustomer) {
159        if (SC_Utils_Ex::isBlank($uniqId)) {
160            return;
161        }
162        $params['device_type_id'] = SC_Display::detectDevice();
163        $objQuery =& SC_Query_Ex::getSingletonInstance();
164        // 存在するカラムのみを対象とする
165        $cols = $objQuery->listTableFields('dtb_order_temp');
166        foreach ($params as $key => $val) {
167            if (in_array($key, $cols)) {
168                $sqlval[$key] = $val;
169            }
170        }
171
172        $sqlval['session'] = serialize($_SESSION);
173        $exists = $this->getOrderTemp($uniqId);
174        if (SC_Utils_Ex::isBlank($exists)) {
175            $this->copyFromCustomer($sqlval, $objCustomer);
176            $sqlval['order_temp_id'] = $uniqId;
177            $sqlval['create_date'] = "now()";
178            $objQuery->insert("dtb_order_temp", $sqlval);
179        } else {
180            $objQuery->update("dtb_order_temp", $sqlval, 'order_temp_id = ?',
181                              array($uniqId));
182        }
183    }
184
185    /**
186     * セッションの配送情報を取得する.
187     */
188    function getShippingTemp() {
189        return $_SESSION['shipping'];
190    }
191
192    /**
193     * 配送商品を設定する.
194     *
195     * @param integer $shipping_id 配送先ID
196     * @param integer $product_class_id 商品規格ID
197     * @param integer $quantity 数量
198     * @return void
199     */
200    function setShipmentItemTemp($shipping_id, $product_class_id, $quantity) {
201        // 配列が長くなるので, リファレンスを使用する
202        $arrItems =& $_SESSION['shipping'][$shipping_id]['shipment_item'];
203
204        $arrItems[$product_class_id]['shipping_id'] = $shipping_id;
205        $arrItems[$product_class_id]['product_class_id'] = $product_class_id;
206        $arrItems[$product_class_id]['quantity'] = $quantity;
207
208        $objProduct = new SC_Product_Ex();
209        if (empty($arrItems[$product_class_id]['productsClass'])) {
210            $product =& $objProduct->getDetailAndProductsClass($product_class_id);
211            $arrItems[$product_class_id]['productsClass'] = $product;
212        }
213        $incTax = SC_Helper_DB_Ex::sfCalcIncTax($arrItems[$product_class_id]['productsClass']['price02']);
214        $arrItems[$product_class_id]['total_inctax'] = $incTax * $arrItems[$product_class_id]['quantity'];
215    }
216
217    /**
218     * 配送商品の情報でカートを更新する.
219     *
220     * @param SC_CartSession $objCartSession SC_CartSession インスタンス
221     */
222    function shippingItemTempToCart(&$objCartSession) {
223        $arrShipmentItems = array();
224        foreach (array_keys($_SESSION['shipping']) as $shipping_id) {
225            foreach (array_keys($_SESSION['shipping'][$shipping_id]['shipment_item']) as $product_class_id) {
226                $arrShipmentItems[$product_class_id] += $_SESSION['shipping'][$shipping_id]['shipment_item'][$product_class_id]['quantity'];
227           }
228        }
229        foreach ($arrShipmentItems as $product_class_id => $quantity) {
230            $objCartSession->setProductValue($product_class_id, 'quantity',
231                                             $quantity,$objCartSession->getKey());
232        }
233    }
234
235    /**
236     * 配送先都道府県の配列を返す.
237     */
238    function getShippingPref() {
239        $results = array();
240        foreach ($_SESSION['shipping'] as $val) {
241            $results[] = $val['shipping_pref'];
242        }
243        return $results;
244    }
245
246    /**
247     * 複数配送指定の購入かどうか.
248     *
249     * @return boolean 複数配送指定の購入の場合 true
250     */
251    function isMultiple() {
252        return count($this->getShippingTemp()) > 1;
253    }
254
255    /**
256     * 配送情報をセッションに保存する.
257     *
258     * @param array $arrSrc 配送情報の連想配列
259     * @param integer $shipping_id 配送先ID
260     * @return void
261     */
262    function saveShippingTemp(&$arrSrc, $shipping_id = 0) {
263        if (empty($_SESSION['shipping'][$shipping_id])) {
264            $_SESSION['shipping'][$shipping_id] = $arrSrc;
265            $_SESSION['shipping'][$shipping_id]['shipping_id'] = $shipping_id;
266        } else {
267            $_SESSION['shipping'][$shipping_id] = array_merge($_SESSION['shipping'][$shipping_id], $arrSrc);
268            $_SESSION['shipping'][$shipping_id]['shipping_id'] = $shipping_id;
269        }
270    }
271
272    /**
273     * セッションの配送情報を破棄する.
274     */
275    function unsetShippingTemp() {
276        unset($_SESSION['shipping']);
277        unset($_SESSION['multiple_temp']);
278    }
279
280    /**
281     * 会員情報を受注情報にコピーする.
282     *
283     * ユーザーがログインしていない場合は何もしない.
284     * 会員情報を $dest の order_* へコピーする.
285     * customer_id は強制的にコピーされる.
286     *
287     * @param array $dest コピー先の配列
288     * @param SC_Customer $objCustomer SC_Customer インスタンス
289     * @param string $prefix コピー先の接頭辞. デフォルト order
290     * @param array $keys コピー対象のキー
291     * @return void
292     */
293    function copyFromCustomer(&$dest, &$objCustomer, $prefix = 'order',
294                              $keys = array('name01', 'name02', 'kana01', 'kana02',
295                                            'sex', 'zip01', 'zip02', 'pref',
296                                            'addr01', 'addr02',
297                                            'tel01', 'tel02', 'tel03', 'job',
298                                            'birth', 'email')) {
299        if ($objCustomer->isLoginSuccess(true)) {
300
301            foreach ($keys as $key) {
302                if (in_array($key, $keys)) {
303                    $dest[$prefix . '_' . $key] = $objCustomer->getValue($key);
304                }
305            }
306
307            if (Net_UserAgent_Mobile::isMobile()
308                && in_array('email', $keys)) {
309                $email_mobile = $objCustomer->getValue('email_mobile');
310                if (empty($email_mobile)) {
311                    $dest[$prefix . '_email'] = $objCustomer->getValue('email');
312                } else {
313                    $dest[$prefix . '_email'] = $email_mobile;
314                }
315            }
316
317            $dest['customer_id'] = $objCustomer->getValue('customer_id');
318            $dest['update_date'] = 'Now()';
319        }
320    }
321
322    /**
323     * 受注情報を配送情報にコピーする.
324     *
325     * 受注情報($src)を $dest の order_* へコピーする.
326     *
327     * TODO 汎用的にして SC_Utils へ移動
328     *
329     * @param array $dest コピー先の配列
330     * @param array $src コピー元の配列
331     * @param array $keys コピー対象のキー
332     * @param string $prefix コピー先の接頭辞. デフォルト shipping
333     * @param string $src_prefix コピー元の接頭辞. デフォルト order
334     * @return void
335     */
336    function copyFromOrder(&$dest, $src,
337                           $prefix = 'shipping', $src_prefix = 'order',
338                           $keys = array('name01', 'name02', 'kana01', 'kana02',
339                                         'sex', 'zip01', 'zip02', 'pref',
340                                         'addr01', 'addr02',
341                                         'tel01', 'tel02', 'tel03')) {
342        if (!SC_Utils_Ex::isBlank($prefix)) {
343            $prefix = $prefix . '_';
344        }
345        if (!SC_Utils_Ex::isBlank($src_prefix)) {
346            $src_prefix = $src_prefix . '_';
347        }
348        foreach ($keys as $key) {
349            if (in_array($key, $keys)) {
350                $dest[$prefix . $key] = $src[$src_prefix . $key];
351            }
352        }
353    }
354
355    /**
356     * 購入金額に応じた支払方法を取得する.
357     *
358     * @param integer $total 購入金額
359     * @param integer $deliv_id 配送業者ID
360     * @return array 購入金額に応じた支払方法の配列
361     */
362    function getPaymentsByPrice($total, $deliv_id) {
363
364        $arrPaymentIds = $this->getPayments($deliv_id);
365        if (SC_Utils_Ex::isBlank($arrPaymentIds)) {
366            return array();
367        }
368
369        $objQuery =& SC_Query_Ex::getSingletonInstance();
370
371        // 削除されていない支払方法を取得
372        $where = 'del_flg = 0 AND payment_id IN (' . implode(', ', array_pad(array(), count($arrPaymentIds), '?')) . ')';
373        $objQuery->setOrder("rank DESC");
374        $payments = $objQuery->select("payment_id, payment_method, rule, upper_rule, note, payment_image, charge", "dtb_payment", $where, $arrPaymentIds);
375        foreach ($payments as $data) {
376            // 下限と上限が設定されている
377            if (strlen($data['rule']) != 0 && strlen($data['upper_rule']) != 0) {
378                if ($data['rule'] <= $total && $data['upper_rule'] >= $total) {
379                    $arrPayment[] = $data;
380                }
381            }
382            // 下限のみ設定されている
383            elseif (strlen($data['rule']) != 0) {
384                if($data['rule'] <= $total) {
385                    $arrPayment[] = $data;
386                }
387            }
388            // 上限のみ設定されている
389            elseif (strlen($data['upper_rule']) != 0) {
390                if($data['upper_rule'] >= $total) {
391                    $arrPayment[] = $data;
392                }
393            }
394            // いずれも設定なし
395            else {
396                $arrPayment[] = $data;
397            }
398          }
399        return $arrPayment;
400    }
401
402    /**
403     * お届け日一覧を取得する.
404     */
405    function getDelivDate(&$objCartSess, $productTypeId) {
406        $cartList = $objCartSess->getCartList($productTypeId);
407        $delivDateIds = array();
408        foreach ($cartList as $item) {
409            $delivDateIds[] = $item['productsClass']['deliv_date_id'];
410        }
411        $max_date = max($delivDateIds);
412        //発送目安
413        switch($max_date) {
414        //即日発送
415        case '1':
416            $start_day = 1;
417            break;
418            //1-2日後
419        case '2':
420            $start_day = 3;
421            break;
422            //3-4日後
423        case '3':
424            $start_day = 5;
425            break;
426            //1週間以内
427        case '4':
428            $start_day = 8;
429            break;
430            //2週間以内
431        case '5':
432            $start_day = 15;
433            break;
434            //3週間以内
435        case '6':
436            $start_day = 22;
437            break;
438            //1ヶ月以内
439        case '7':
440            $start_day = 32;
441            break;
442            //2ヶ月以降
443        case '8':
444            $start_day = 62;
445            break;
446            //お取り寄せ(商品入荷後)
447        case '9':
448            $start_day = "";
449            break;
450        default:
451            //お届け日が設定されていない場合
452            $start_day = "";
453        }
454        //お届け可能日のスタート値から、お届け日の配列を取得する
455        $arrDelivDate = $this->getDateArray($start_day, DELIV_DATE_END_MAX);
456        return $arrDelivDate;
457    }
458
459    /**
460     * お届け可能日のスタート値から, お届け日の配列を取得する.
461     */
462    function getDateArray($start_day, $end_day) {
463        $masterData = new SC_DB_MasterData();
464        $arrWDAY = $masterData->getMasterData("mtb_wday");
465        //お届け可能日のスタート値がセットされていれば
466        if($start_day >= 1) {
467            $now_time = time();
468            $max_day = $start_day + $end_day;
469            // 集計
470            for ($i = $start_day; $i < $max_day; $i++) {
471                // 基本時間から日数を追加していく
472                $tmp_time = $now_time + ($i * 24 * 3600);
473                list($y, $m, $d, $w) = explode(" ", date("Y m d w", $tmp_time));
474                $val = sprintf("%04d/%02d/%02d(%s)", $y, $m, $d, $arrWDAY[$w]);
475                $arrDate[$val] = $val;
476            }
477        } else {
478            $arrDate = false;
479        }
480        return $arrDate;
481    }
482
483    /**
484     * 配送業者IDからお届け時間の配列を取得する.
485     *
486     * @param integer $deliv_id 配送業者ID
487     * @return array お届け時間の配列
488     */
489    function getDelivTime($deliv_id) {
490        $objQuery =& SC_Query_Ex::getSingletonInstance();
491        $objQuery->setOrder('time_id');
492        $results = $objQuery->select('time_id, deliv_time',
493                                     'dtb_delivtime',
494                                     'deliv_id = ?', array($deliv_id));
495        $arrDelivTime = array();
496        foreach ($results as $val) {
497            $arrDelivTime[$val['time_id']] = $val['deliv_time'];
498        }
499        return $arrDelivTime;
500    }
501
502    /**
503     * 商品種別ID から配送業者を取得する.
504     *
505     * @param integer $product_type_id 商品種別ID
506     * @return array 配送業者の配列
507     */
508    function getDeliv($product_type_id) {
509        $objQuery =& SC_Query_Ex::getSingletonInstance();
510        $objQuery->setOrder('rank DESC');
511        return $objQuery->select('*', 'dtb_deliv', 'product_type_id = ? AND del_flg = 0',
512                                 array($product_type_id));
513    }
514
515    /**
516     * 配送業者ID から, 有効な支払方法IDを取得する.
517     *
518     * @param integer $deliv_id 配送業者ID
519     * @return array 有効な支払方法IDの配列
520     */
521    function getPayments($deliv_id) {
522        $objQuery =& SC_Query_Ex::getSingletonInstance();
523        $objQuery->setOrder('rank');
524        return $objQuery->getCol('payment_id', 'dtb_payment_options',
525                                 'deliv_id = ?',
526                                 array($deliv_id), MDB2_FETCHMODE_ORDERED);
527    }
528
529    /**
530     * 配送情報の登録を行う.
531     *
532     * $arrParam のうち, dtb_shipping テーブルに存在するカラムのみを登録する.
533     *
534     * TODO UPDATE/INSERT にする
535     *
536     * @param integer $order_id 受注ID
537     * @param array $arrParams 配送情報の連想配列
538     * @param boolean $convert_shipping_date yyyy/mm/dd(EEE) 形式の配送日付を変換する場合 true
539     * @return void
540     */
541    function registerShipping($order_id, $arrParams, $convert_shipping_date = true) {
542        $objQuery =& SC_Query_Ex::getSingletonInstance();
543        $table = 'dtb_shipping';
544        $where = 'order_id = ?';
545        $objQuery->delete($table, $where, array($order_id));
546
547         foreach ($arrParams as $key => $arrShipping) {
548
549            $arrValues = $objQuery->extractOnlyColsOf($table, $arrShipping);
550
551            // 配送日付を timestamp に変換
552            if (!SC_Utils_Ex::isBlank($arrValues['shipping_date'])
553                && $convert_shipping_date) {
554                $d = mb_strcut($arrValues["shipping_date"], 0, 10);
555                $arrDate = explode("/", $d);
556                $ts = mktime(0, 0, 0, $arrDate[1], $arrDate[2], $arrDate[0]);
557                $arrValues['shipping_date'] = date("Y-m-d", $ts);
558            }
559
560            // 非会員購入の場合は shipping_id が存在しない
561            if (!isset($arrValues['shipping_id'])) {
562                $arrValues['shipping_id'] = $key;
563            }
564            $arrValues['order_id'] = $order_id;
565            $arrValues['create_date'] = 'Now()';
566            $arrValues['update_date'] = 'Now()';
567            $objQuery->insert($table, $arrValues);
568        }
569    }
570
571    /**
572     * 配送商品を登録する.
573     *
574     * @param integer $order_id 受注ID
575     * @param integer $shipping_id 配送先ID
576     * @param array $arrParams 配送商品の配列
577     * @return void
578     */
579    function registerShipmentItem($order_id, $shipping_id, $arrParams) {
580        $objQuery =& SC_Query_Ex::getSingletonInstance();
581        $table = 'dtb_shipment_item';
582        $where = 'order_id = ? AND shipping_id = ?';
583        $objQuery->delete($table, $where, array($order_id, $shipping_id));
584
585        $objProduct = new SC_Product_Ex();
586        foreach ($arrParams as $arrValues) {
587            if (SC_Utils_Ex::isBlank($arrValues['product_class_id'])) {
588                continue;
589            }
590            $d = $objProduct->getDetailAndProductsClass($arrValues['product_class_id']);
591            $name = SC_Utils_Ex::isBlank($arrValues['product_name'])
592                ? $d['name']
593                : $arrValues['product_name'];
594
595            $code = SC_Utils_Ex::isBlank($arrValues['product_code'])
596                ? $d['product_code']
597                : $arrValues['product_code'];
598
599            $cname1 = SC_Utils_Ex::isBlank($arrValues['classcategory_name1'])
600                ? $d['classcategory_name1']
601                : $arrValues['classcategory_name1'];
602
603            $cname2 = SC_Utils_Ex::isBlank($arrValues['classcategory_name2'])
604                ? $d['classcategory_name2']
605                : $arrValues['classcategory_name2'];
606
607            $price = SC_Utils_Ex::isBlank($arrValues['price'])
608                ? $d['price02']
609                : $arrValues['price'];
610
611            $arrValues['order_id'] = $order_id;
612            $arrValues['shipping_id'] = $shipping_id;
613            $arrValues['product_name'] = $name;
614            $arrValues['product_code'] = $code;
615            $arrValues['classcategory_name1'] = $cname1;
616            $arrValues['classcategory_name2'] = $cname2;
617            $arrValues['price'] = $price;
618
619            $arrExtractValues = $objQuery->extractOnlyColsOf($table, $arrValues);
620            $objQuery->insert($table, $arrExtractValues);
621        }
622    }
623
624    /**
625     * 受注登録を完了する.
626     *
627     * 引数の受注情報を受注テーブル及び受注詳細テーブルに登録する.
628     * 登録後, 受注一時テーブルに削除フラグを立てる.
629     *
630     * @param array $orderParams 登録する受注情報の配列
631     * @param SC_CartSession $objCartSession カート情報のインスタンス
632     * @param integer $cartKey 登録を行うカート情報のキー
633     * @param integer 受注ID
634     */
635    function registerOrderComplete($orderParams, &$objCartSession, $cartKey) {
636        $objQuery =& SC_Query_Ex::getSingletonInstance();
637
638        // 不要な変数を unset
639        $unsets = array('mailmaga_flg', 'deliv_check', 'point_check', 'password',
640                        'reminder', 'reminder_answer', 'mail_flag', 'session');
641        foreach ($unsets as $unset) {
642            unset($orderParams[$unset]);
643        }
644
645        // 注文ステータスの指定が無い場合は新規受付
646        if(SC_Utils_Ex::isBlank($orderParams['status'])) {
647            $orderParams['status'] = ORDER_NEW;
648        }
649
650        $orderParams['create_date'] = 'Now()';
651        $orderParams['update_date'] = 'Now()';
652
653        $this->registerOrder($orderParams['order_id'], $orderParams);
654
655        // 詳細情報を取得
656        $cartItems = $objCartSession->getCartList($cartKey);
657
658        // 詳細情報を生成
659        $objProduct = new SC_Product_Ex();
660        $i = 0;
661        foreach ($cartItems as $item) {
662            $p =& $item['productsClass'];
663            $arrDetail[$i]['order_id'] = $orderParams['order_id'];
664            $arrDetail[$i]['product_id'] = $p['product_id'];
665            $arrDetail[$i]['product_class_id'] = $p['product_class_id'];
666            $arrDetail[$i]['product_name'] = $p['name'];
667            $arrDetail[$i]['product_code'] = $p['product_code'];
668            $arrDetail[$i]['classcategory_name1'] = $p['classcategory_name1'];
669            $arrDetail[$i]['classcategory_name2'] = $p['classcategory_name2'];
670            $arrDetail[$i]['point_rate'] = $item['point_rate'];
671            $arrDetail[$i]['price'] = $item['price'];
672            $arrDetail[$i]['quantity'] = $item['quantity'];
673
674            // 在庫の減少処理
675            if (!$objProduct->reduceStock($p['product_class_id'], $item['quantity'])) {
676                $objQuery->rollback();
677                SC_Utils_Ex::sfDispSiteError(SOLD_OUT, "", true);
678            }
679            $i++;
680        }
681        $this->registerOrderDetail($orderParams['order_id'], $arrDetail);
682
683        $objQuery->update("dtb_order_temp", array('del_flg' => 1),
684                          "order_temp_id = ?",
685                          array(SC_SiteSession_Ex::getUniqId()));
686
687        return $orderParams['order_id'];
688    }
689
690    /**
691     * 受注情報を登録する.
692     *
693     * 既に受注IDが存在する場合は, 受注情報を更新する.
694     * 引数の受注IDが, 空白又は null の場合は, 新しく受注IDを発行して登録する.
695     *
696     * @param integer $order_id 受注ID
697     * @param array $arrParams 受注情報の連想配列
698     * @return integer 受注ID
699     */
700    function registerOrder($order_id, $arrParams) {
701        $table = 'dtb_order';
702        $where = 'order_id = ?';
703        $objQuery = SC_Query_Ex::getSingletonInstance();
704        $arrValues = $objQuery->extractOnlyColsOf($table, $arrParams);
705
706        $exists = $objQuery->count($table, $where, array($order_id));
707        if ($exists > 0) {
708
709            $this->sfUpdateOrderStatus($order_id, $arrValues['status'],
710                                       $arrValues['add_point'],
711                                       $arrValues['use_point'],
712                                       $arrValues);
713            $this->sfUpdateOrderNameCol($order_id);
714
715            $arrValues['update_date'] = 'now()';
716            $objQuery->update($table, $arrValues, $where, array($order_id));
717        } else {
718            if (SC_Utils_Ex::isBlank($order_id)) {
719                $order_id = $objQuery->nextVal('dtb_order_order_id');
720            }
721            /*
722             * 新規受付の場合は受注ステータス null で insert し,
723             * sfUpdateOrderStatus で ORDER_NEW に変更する.
724             */
725            $status = $arrValues['status'];
726            $arrValues['status'] = null;
727            $arrValues['order_id'] = $order_id;
728            $arrValues['customer_id'] =
729                    SC_Utils_Ex::isBlank($arrValues['customer_id'])
730                    ? 0 : $arrValues['customer_id'];
731            $arrValues['create_date'] = 'now()';
732            $arrValues['update_date'] = 'now()';
733            $objQuery->insert($table, $arrValues);
734
735            $this->sfUpdateOrderStatus($order_id, $status,
736                                       $arrValues['add_point'],
737                                       $arrValues['use_point'],
738                                       $arrValues);
739            $this->sfUpdateOrderNameCol($order_id);
740
741        }
742        return $order_id;
743    }
744
745    /**
746     * 受注詳細情報を登録する.
747     *
748     * 既に, 該当の受注が存在する場合は, 受注情報を削除し, 登録する.
749     *
750     * @param integer $order_id 受注ID
751     * @param array $arrParams 受注情報の連想配列
752     * @return void
753     */
754    function registerOrderDetail($order_id, $arrParams) {
755        $table = 'dtb_order_detail';
756        $where = 'order_id = ?';
757        $objQuery = SC_Query_Ex::getSingletonInstance();
758
759        $objQuery->delete($table, $where, array($order_id));
760        foreach ($arrParams as $arrDetail) {
761            $arrValues = $objQuery->extractOnlyColsOf($table, $arrDetail);
762            $arrValues['order_detail_id'] = $objQuery->nextVal('dtb_order_detail_order_detail_id');
763            $arrValues['order_id'] = $order_id;
764            $objQuery->insert($table, $arrValues);
765        }
766    }
767
768    /**
769     * 受注情報を取得する.
770     *
771     * @param integer $order_id 受注ID
772     * @param integer $customer_id 顧客ID
773     * @return array 受注情報の配列
774     */
775    function getOrder($order_id, $customer_id = null) {
776        $objQuery =& SC_Query_Ex::getSingletonInstance();
777        $where = 'order_id = ?';
778        $arrValues = array($order_id);
779        if (!SC_Utils_Ex::isBlank($customer_id)) {
780            $where .= ' AND customer_id = ?';
781            $arrValues[] = $customer_id;
782        }
783        return $objQuery->getRow('*', 'dtb_order', $where, $arrValues);
784    }
785
786    /**
787     * 受注詳細を取得する.
788     *
789     * @param integer $order_id 受注ID
790     * @param boolean $has_order_status 受注ステータス, 入金日も含める場合 true
791     * @return array 受注詳細の配列
792     */
793    function getOrderDetail($order_id, $has_order_status = true) {
794        $objQuery =& SC_Query_Ex::getSingletonInstance();
795        $dbFactory  = SC_DB_DBFactory_Ex::getInstance();
796        $col = <<< __EOS__
797            T3.product_id,
798            T3.product_class_id as product_class_id,
799            T3.product_type_id AS product_type_id,
800            T2.product_code,
801            T2.product_name,
802            T2.classcategory_name1 AS classcategory_name1,
803            T2.classcategory_name2 AS classcategory_name2,
804            T2.price,
805            T2.quantity,
806            T2.point_rate,
807__EOS__;
808        if ($has_order_status) {
809            $col .= 'T1.status AS status, T1.payment_date AS payment_date,';
810
811        }
812        $col .= <<< __EOS__
813
814            CASE WHEN EXISTS(
815                    SELECT 1 FROM dtb_products
816                     WHERE product_id = T3.product_id
817                       AND del_flg = 0
818                       AND status = 1)
819                 THEN '1' ELSE '0'
820                  END AS enable,
821__EOS__;
822        $col .= $dbFactory->getDownloadableDaysWhereSql('T1') . ' AS effective';
823        $from = <<< __EOS__
824                 dtb_order T1
825            JOIN dtb_order_detail T2
826              ON T1.order_id = T2.order_id
827            JOIN dtb_products_class T3
828              ON T2.product_class_id = T3.product_class_id
829__EOS__;
830        $objQuery->setOrder('T2.order_detail_id');
831        return $objQuery->select($col, $from, 'T1.order_id = ?', array($order_id));
832    }
833
834    /**
835     * 配送情報を取得する.
836     *
837     * @param integer $order_id 受注ID
838     * @param boolean $has_items 結果に配送商品も含める場合 true
839     * @return array 配送情報の配列
840     */
841    function getShippings($order_id, $has_items = true) {
842        $objQuery =& SC_Query_Ex::getSingletonInstance();
843        $arrResults = array();
844        $objQuery->setOrder('shipping_id');
845        $arrShippings = $objQuery->select("*", "dtb_shipping", "order_id = ?",
846                                          array($order_id));
847        // shipping_id ごとの配列を生成する
848        foreach ($arrShippings as $shipping) {
849            foreach ($shipping as $key => $val) {
850                $arrResults[$shipping['shipping_id']][$key] = $val;
851            }
852        }
853
854        if ($has_items) {
855            $objProduct = new SC_Product_Ex();
856            foreach (array_keys($arrResults) as $shipping_id) {
857                $arrResults[$shipping_id]['shipment_item']
858                        =& $this->getShipmentItems($order_id, $shipping_id);
859            }
860        }
861        return $arrResults;
862    }
863
864    /**
865     * 配送商品を取得する.
866     *
867     * @param integer $order_id 受注ID
868     * @param integer $shipping_id 配送先ID
869     * @param boolean $has_detail 商品詳細も取得する場合 true
870     * @return array 商品規格IDをキーにした配送商品の配列
871     */
872    function getShipmentItems($order_id, $shipping_id, $has_detail = true) {
873        $objQuery =& SC_Query_Ex::getSingletonInstance();
874        $objProduct = new SC_Product_Ex();
875        $arrResults = array();
876        $arrItems = $objQuery->select("*", "dtb_shipment_item",
877                                      "order_id = ? AND shipping_id = ?",
878                                      array($order_id, $shipping_id));
879
880        foreach ($arrItems as $key => $arrItem) {
881            $product_class_id = $arrItem['product_class_id'];
882
883            foreach ($arrItem as $detailKey => $detailVal) {
884                $arrResults[$key][$detailKey] = $detailVal;
885            }
886            // 商品詳細を関連づける
887            if ($has_detail) {
888                $arrResults[$key]['productsClass']
889                    =& $objProduct->getDetailAndProductsClass($product_class_id);
890            }
891        }
892        return $arrResults;
893    }
894
895    /**
896     * 受注完了メールを送信する.
897     *
898     * HTTP_USER_AGENT の種別により, 携帯電話の場合は携帯用の文面,
899     * PC の場合は PC 用の文面でメールを送信する.
900     *
901     * @param integer $orderId 受注ID
902     * @return void
903     */
904    function sendOrderMail($orderId) {
905        $mailHelper = new SC_Helper_Mail_Ex();
906        $mailHelper->sfSendOrderMail($orderId,
907                                     SC_MobileUserAgent_Ex::isMobile() ? 2 : 1);
908    }
909
910    /**
911     * 受注.対応状況の更新
912     *
913     * 必ず呼び出し元でトランザクションブロックを開いておくこと。
914     *
915     * @param integer $orderId 注文番号
916     * @param integer|null $newStatus 対応状況 (null=変更無し)
917     * @param integer|null $newAddPoint 加算ポイント (null=変更無し)
918     * @param integer|null $newUsePoint 使用ポイント (null=変更無し)
919     * @param array $sqlval 更新後の値をリファレンスさせるためのパラメータ
920     * @return void
921     */
922    function sfUpdateOrderStatus($orderId, $newStatus = null, $newAddPoint = null, $newUsePoint = null, &$sqlval) {
923        $objQuery =& SC_Query_Ex::getSingletonInstance();
924
925        $arrOrderOld = $objQuery->getRow('status, add_point, use_point, customer_id', 'dtb_order', 'order_id = ?', array($orderId));
926
927        // 対応状況が変更無しの場合、DB値を引き継ぐ
928        if (is_null($newStatus)) {
929            $newStatus = $arrOrderOld['status'];
930        }
931
932        // 使用ポイント、DB値を引き継ぐ
933        if (is_null($newUsePoint)) {
934            $newUsePoint = $arrOrderOld['use_point'];
935        }
936
937        // 加算ポイント、DB値を引き継ぐ
938        if (is_null($newAddPoint)) {
939            $newAddPoint = $arrOrderOld['add_point'];
940        }
941
942        if (USE_POINT !== false) {
943            // 顧客.ポイントの加減値
944            $addCustomerPoint = 0;
945
946            // ▼使用ポイント
947            // 変更前の対応状況が利用対象の場合、変更前の使用ポイント分を戻す
948            if ($this->isUsePoint($arrOrderOld['status'])) {
949                $addCustomerPoint += $arrOrderOld['use_point'];
950            }
951
952            // 変更後の対応状況が利用対象の場合、変更後の使用ポイント分を引く
953            if ($this->isUsePoint($newStatus)) {
954                $addCustomerPoint -= $newUsePoint;
955            }
956
957            // ▲使用ポイント
958
959            // ▼加算ポイント
960            // 変更前の対応状況が加算対象の場合、変更前の加算ポイント分を戻す
961            if ($this->isAddPoint($arrOrderOld['status'])) {
962                $addCustomerPoint -= $arrOrderOld['add_point'];
963            }
964
965            // 変更後の対応状況が加算対象の場合、変更後の加算ポイント分を足す
966            if ($this->isAddPoint($newStatus)) {
967                $addCustomerPoint += $newAddPoint;
968            }
969            // ▲加算ポイント
970
971            if ($addCustomerPoint != 0) {
972                // ▼顧客テーブルの更新
973                $sqlval = array();
974                $where = '';
975                $arrVal = array();
976                $arrRawSql = array();
977                $arrRawSqlVal = array();
978
979                $sqlval['update_date'] = 'Now()';
980                $arrRawSql['point'] = 'point + ?';
981                $arrRawSqlVal[] = $addCustomerPoint;
982                $where .= 'customer_id = ?';
983                $arrVal[] = $arrOrderOld['customer_id'];
984
985                $objQuery->update('dtb_customer', $sqlval, $where, $arrVal, $arrRawSql, $arrRawSqlVal);
986                // ▲顧客テーブルの更新
987
988                // 顧客.ポイントをマイナスした場合、
989                if ($addCustomerPoint < 0) {
990                    $sql = 'SELECT point FROM dtb_customer WHERE customer_id = ?';
991                    $point = $objQuery->getOne($sql, array($arrOrderOld['customer_id']));
992                    // 変更後の顧客.ポイントがマイナスの場合、
993                    if ($point < 0) {
994                        // ロールバック
995                        $objQuery->rollback();
996                        // エラー
997                        SC_Utils_Ex::sfDispSiteError(LACK_POINT);
998                    }
999                }
1000            }
1001        }
1002
1003        // ▼受注テーブルの更新
1004        if (empty($sqlval)) {
1005            $sqlval = array();
1006        }
1007
1008        if (USE_POINT !== false) {
1009            $sqlval['add_point'] = $newAddPoint;
1010            $sqlval['use_point'] = $newUsePoint;
1011        }
1012        // ステータスが発送済みに変更の場合、発送日を更新
1013        if ($arrOrderOld['status'] != ORDER_DELIV && $newStatus == ORDER_DELIV) {
1014            $sqlval['commit_date'] = 'Now()';
1015        }
1016        // ステータスが入金済みに変更の場合、入金日を更新
1017        elseif ($arrOrderOld['status'] != ORDER_PRE_END && $newStatus == ORDER_PRE_END) {
1018            $sqlval['payment_date'] = 'Now()';
1019        }
1020
1021        $sqlval['status'] = $newStatus;
1022        $sqlval['update_date'] = 'Now()';
1023
1024        $dest = $objQuery->extractOnlyColsOf('dtb_order', $sqlval);
1025        $objQuery->update('dtb_order', $dest, 'order_id = ?', array($orderId));
1026        // ▲受注テーブルの更新
1027    }
1028
1029    /**
1030     * 受注の名称列を更新する
1031     *
1032     * @param integer $order_id 更新対象の注文番号
1033     * @param boolean $temp_table 更新対象は「受注_Temp」か
1034     * @static
1035     */
1036    function sfUpdateOrderNameCol($order_id, $temp_table = false) {
1037        $objQuery =& SC_Query_Ex::getSingletonInstance();
1038
1039        if ($temp_table) {
1040            $tgt_table = 'dtb_order_temp';
1041            $sql_where = 'order_temp_id = ?';
1042        } else {
1043            $tgt_table = 'dtb_order';
1044            $sql_where = 'order_id = ?';
1045
1046            $objQuery->update('dtb_shipping', array(),
1047                              $sql_where,
1048                              array($order_id),
1049                              array('shipping_time' =>
1050                                    "(SELECT deliv_time FROM dtb_delivtime WHERE time_id = dtb_shipping.time_id AND deliv_id = dtb_shipping.deliv_id)"));
1051
1052        }
1053
1054        $objQuery->update($tgt_table, array(),
1055                          $sql_where,
1056                          array($order_id),
1057                          array('payment_method' =>
1058                                "(SELECT payment_method FROM dtb_payment WHERE payment_id = " . $tgt_table . ".payment_id)"));
1059    }
1060
1061    /**
1062     * ポイント使用するかの判定
1063     *
1064     * $status が null の場合は false を返す.
1065     *
1066     * @param integer $status 対応状況
1067     * @return boolean 使用するか(顧客テーブルから減算するか)
1068     */
1069    function isUsePoint($status) {
1070        if ($status == null) {
1071            return false;
1072        }
1073        switch ($status) {
1074            case ORDER_CANCEL:      // キャンセル
1075                return false;
1076            default:
1077                break;
1078        }
1079
1080        return true;
1081    }
1082
1083    /**
1084     * ポイント加算するかの判定
1085     *
1086     * @param integer $status 対応状況
1087     * @return boolean 加算するか
1088     */
1089    function isAddPoint($status) {
1090        switch ($status) {
1091            case ORDER_NEW:         // 新規注文
1092            case ORDER_PAY_WAIT:    // 入金待ち
1093            case ORDER_PRE_END:     // 入金済み
1094            case ORDER_CANCEL:      // キャンセル
1095            case ORDER_BACK_ORDER:  // 取り寄せ中
1096                return false;
1097
1098            case ORDER_DELIV:       // 発送済み
1099                return true;
1100
1101            default:
1102                break;
1103        }
1104
1105        return false;
1106    }
1107
1108    /**
1109     * セッションに保持している情報を破棄する.
1110     *
1111     * 通常、受注処理(completeOrder)完了後に呼び出され、
1112     * セッション情報を破棄する.
1113     *
1114     * 決済モジュール画面から確認画面に「戻る」場合を考慮し、
1115     * セッション情報を破棄しないカスタマイズを、モジュール側で
1116     * 加える機会を与える.
1117     *
1118     * @param integer $orderId 注文番号
1119     * @param SC_CartSession $objCartSession カート情報のインスタンス
1120     * @param SC_Customer $objCustomer SC_Customer インスタンス
1121     * @param integer $cartKey 登録を行うカート情報のキー
1122     */
1123    function cleanupSession($orderId, &$objCartSession, &$objCustomer, $cartKey) {
1124        // カートの内容を削除する.
1125        $objCartSession->delAllProducts($cartKey);
1126        SC_SiteSession_Ex::unsetUniqId();
1127
1128        // セッションの配送情報を破棄する.
1129        $this->unsetShippingTemp();
1130        $objCustomer->updateSession();
1131    }
1132}
Note: See TracBrowser for help on using the repository browser.