source: branches/version-2_11-dev/data/class/helper/SC_Helper_Purchase.php @ 20970

Revision 20970, 47.3 KB checked in by Seasoft, 10 years ago (diff)

#1288 (「-er」カタカナ表記の統一)

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