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

Revision 23681, 54.3 KB checked in by shutta, 9 years ago (diff)

#2649 (購入確認ページにて、ページ遷移の正当性チェックをすり抜けることができる)
r23680 と同様な箇所を修正。

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