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

Revision 21185, 47.2 KB checked in by shutta, 13 years ago (diff)

refs #800 (SQL標準関数を使用する)
CURRENT_TIMESTAMP を使用するように now() を置換。

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