source: branches/version-2_11-dev/data/class/pages/admin/products/LC_Page_Admin_Products_UploadCSV.php @ 20829

Revision 20829, 33.5 KB checked in by saiteisan, 11 years ago (diff)

refs #1195 (product_idが入っている場合、該当IDがあればupdateなければinsert)

  • 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// {{{ requires
25require_once CLASS_EX_REALDIR . 'page_extends/admin/LC_Page_Admin_Ex.php';
26require_once CLASS_EX_REALDIR . 'helper_extends/SC_Helper_CSV_Ex.php';
27
28/**
29 * 商品登録CSVのページクラス.
30 *
31 * @package Page
32 * @author LOCKON CO.,LTD.
33 * @version $Id:LC_Page_Admin_Products_UploadCSV.php 15532 2007-08-31 14:39:46Z nanasess $
34 *
35 * FIXME 同一商品IDで商品規格違いを登録できない。(更新は可能)
36 */
37class LC_Page_Admin_Products_UploadCSV extends LC_Page_Admin_Ex {
38
39    // }}}
40    // {{{ functions
41
42    /** TAGエラーチェックフィールド情報 */
43    var $arrTagCheckItem;
44
45    /** 商品テーブルカラム情報 (登録処理用) **/
46    var $arrProductColumn;
47
48    /** 商品規格テーブルカラム情報 (登録処理用) **/
49    var $arrProductClassColumn;
50
51    /** 登録フォームカラム情報 **/
52    var $arrFormKeyList;
53
54    var $arrRowErr;
55
56    var $arrRowResult;
57
58    /**
59     * Page を初期化する.
60     *
61     * @return void
62     */
63    function init() {
64        parent::init();
65        $this->tpl_mainpage = 'products/upload_csv.tpl';
66        $this->tpl_subnavi = 'products/subnavi.tpl';
67        $this->tpl_mainno = 'products';
68        $this->tpl_subno = 'upload_csv';
69        $this->tpl_subtitle = '商品登録CSV';
70        $this->csv_id = '1';
71
72        $masterData = new SC_DB_MasterData_Ex();
73        $this->arrDISP = $masterData->getMasterData("mtb_disp");
74        $this->arrSTATUS = $masterData->getMasterData("mtb_status");
75        $this->arrDELIVERYDATE = $masterData->getMasterData("mtb_delivery_date");
76        $this->arrProductType = $masterData->getMasterData("mtb_product_type");
77        $this->arrMaker = SC_Helper_DB_Ex::sfGetIDValueList("dtb_maker", "maker_id", 'name');
78        $this->arrPayments = SC_Helper_DB_Ex::sfGetIDValueList("dtb_payment", "payment_id", "payment_method");
79        $this->arrInfo = SC_Helper_DB_Ex::sfGetBasisData();
80        $this->arrAllowedTag = $masterData->getMasterData("mtb_allowed_tag");
81        $this->arrTagCheckItem = array();
82    }
83
84    /**
85     * Page のプロセス.
86     *
87     * @return void
88     */
89    function process() {
90        $this->action();
91        $this->sendResponse();
92    }
93
94    /**
95     * Page のアクション.
96     *
97     * @return void
98     */
99    function action() {
100        $this->objDb = new SC_Helper_DB_Ex();
101
102        // CSV管理ヘルパー
103        $objCSV = new SC_Helper_CSV_Ex();
104        // CSV構造読み込み
105        $arrCSVFrame = $objCSV->sfGetCsvOutput($this->csv_id);
106
107        // CSV構造がインポート可能かのチェック
108        if(!$objCSV->sfIsImportCSVFrame($arrCSVFrame) ) {
109            // 無効なフォーマットなので初期状態に強制変更
110            $arrCSVFrame = $objCSV->sfGetCsvOutput($this->csv_id, '', array(), 'no');
111            $this->tpl_is_format_default = true;
112        }
113        // CSV構造は更新可能なフォーマットかのフラグ取得
114        $this->tpl_is_update = $objCSV->sfIsUpdateCSVFrame($arrCSVFrame);
115
116        // CSVファイルアップロード情報の初期化
117        $objUpFile = new SC_UploadFile_Ex(IMAGE_TEMP_REALDIR, IMAGE_SAVE_REALDIR);
118        $this->lfInitFile($objUpFile);
119
120        // パラメータ情報の初期化
121        $objFormParam = new SC_FormParam_Ex();
122        $this->lfInitParam($objFormParam, $arrCSVFrame);
123
124        $objFormParam->setHtmlDispNameArray();
125        $this->arrTitle = $objFormParam->getHtmlDispNameArray();
126
127        switch($this->getMode()) {
128        case 'csv_upload':
129            $this->doUploadCsv($objFormParam, $objUpFile);
130            break;
131        default:
132            break;
133        }
134    }
135
136    /**
137     * 登録/編集結果のメッセージをプロパティへ追加する
138     *
139     * @param integer $line_count 行数
140     * @param stirng $message メッセージ
141     * @return void
142     */
143    function addRowResult($line_count, $message) {
144        $this->arrRowResult[] = $line_count . "行目:" . $message;
145    }
146
147    /**
148     * 登録/編集結果のエラーメッセージをプロパティへ追加する
149     *
150     * @param integer $line_count 行数
151     * @param stirng $message メッセージ
152     * @return void
153     */
154    function addRowErr($line_count, $message) {
155        $this->arrRowErr[] = $line_count . "行目:" . $message;
156    }
157
158    /**
159     * CSVアップロードを実行します.
160     *
161     * @return void
162     */
163    function doUploadCsv(&$objFormParam, &$objUpFile) {
164        // ファイルアップロードのチェック
165        $objUpFile->makeTempFile('csv_file');
166        $arrErr = $objUpFile->checkExists();
167        if (count($arrErr) > 0) {
168            $this->arrErr = $arrErr;
169            return;
170        }
171        // 一時ファイル名の取得
172        $filepath = $objUpFile->getTempFilePath('csv_file');
173        // CSVファイルの文字コード変換
174        $enc_filepath = SC_Utils_Ex::sfEncodeFile($filepath, CHAR_CODE, CSV_TEMP_REALDIR);
175        // CSVファイルのオープン
176        $fp = fopen($enc_filepath, 'r');
177        // 失敗した場合はエラー表示
178        if (!$fp) {
179             SC_Utils_Ex::sfDispError("");
180        }
181
182        // 登録先テーブル カラム情報の初期化
183        $this->lfInitTableInfo();
184
185        // 登録フォーム カラム情報
186        $this->arrFormKeyList = $objFormParam->getKeyList();
187
188        // 登録対象の列数
189        $col_max_count = $objFormParam->getCount();
190        // 行数
191        $line_count = 0;
192
193        $objQuery =& SC_Query_Ex::getSingletonInstance();
194        $objQuery->begin();
195
196        $errFlag = false;
197        $all_line_checked = false;
198
199        while (!feof($fp)) {
200            $arrCSV = fgetcsv($fp, CSV_LINE_MAX);
201
202            // 全行入力チェック後に、ファイルポインターを先頭に戻す
203            if (feof($fp) && !$all_line_checked) {
204                rewind($fp);
205                $line_count = 0;
206                $all_line_checked = true;
207                continue;
208            }
209
210            // 行カウント
211            $line_count++;
212            // ヘッダ行はスキップ
213            if ($line_count == 1) {
214                continue;
215            }
216            // 空行はスキップ
217            if (empty($arrCSV)) {
218                continue;
219            }
220            // 列数が異なる場合はエラー
221            $col_count = count($arrCSV);
222            if ($col_max_count != $col_count) {
223                $this->addRowErr($line_count, "※ 項目数が" . $col_count . "個検出されました。項目数は" . $col_max_count . "個になります。");
224                $errFlag = true;
225                break;
226            }
227            // シーケンス配列を格納する。
228            $objFormParam->setParam($arrCSV, true);
229            $arrRet = $objFormParam->getHashArray();
230            $objFormParam->setParam($arrRet);
231            // 入力値の変換
232            $objFormParam->convParam();
233            // <br>なしでエラー取得する。
234            $arrCSVErr = $this->lfCheckError($objFormParam);
235
236            // 入力エラーチェック
237            if (count($arrCSVErr) > 0) {
238                foreach ($arrCSVErr as $err) {
239                    $this->addRowErr($line_count, $err);
240                }
241                $errFlag = true;
242                break;
243            }
244
245            if ($all_line_checked) {
246                $this->lfRegistProduct($objQuery, $line_count, $objFormParam);
247                $arrParam = $objFormParam->getHashArray();
248
249                $this->addRowResult($line_count, "商品ID:".$arrParam['product_id'] . " / 商品名:" . $arrParam['name']);
250            }
251        }
252
253        // 実行結果画面を表示
254        $this->tpl_mainpage = 'products/upload_csv_complete.tpl';
255
256        fclose($fp);
257
258        if ($errFlag) {
259            $objQuery->rollback();
260            return;
261        }
262
263        $objQuery->commit();
264
265        // 商品件数カウント関数の実行
266        $this->objDb->sfCountCategory($objQuery);
267        $this->objDb->sfCountMaker($objQuery);
268        return;
269    }
270
271    /**
272     * デストラクタ.
273     *
274     * @return void
275     */
276    function destroy() {
277        parent::destroy();
278    }
279
280    /**
281     * ファイル情報の初期化を行う.
282     *
283     * @return void
284     */
285    function lfInitFile(&$objUpFile) {
286        $objUpFile->addFile("CSVファイル", 'csv_file', array('csv'), CSV_SIZE, true, 0, 0, false);
287    }
288
289    /**
290     * 入力情報の初期化を行う.
291     *
292     * @param array CSV構造設定配列
293     * @return void
294     */
295    function lfInitParam(&$objFormParam, &$arrCSVFrame) {
296        // 固有の初期値調整
297        $arrCSVFrame = $this->lfSetParamDefaultValue($arrCSVFrame);
298        // CSV項目毎の処理
299        foreach($arrCSVFrame as $item) {
300            if($item['status'] == CSV_COLUMN_STATUS_FLG_DISABLE) continue;
301            //サブクエリ構造の場合は AS名 を使用
302            if(preg_match_all('/\(.+\) as (.+)$/i', $item['col'], $match, PREG_SET_ORDER)) {
303                $col = $match[0][1];
304            }else{
305                $col = $item['col'];
306            }
307            // HTML_TAG_CHECKは別途実行なので除去し、別保存しておく
308            if(strpos(strtoupper($item['error_check_types']), 'HTML_TAG_CHECK') !== FALSE) {
309                $this->arrTagCheckItem[] = $item;
310                $error_check_types = str_replace('HTML_TAG_CHECK', '', $item['error_check_types']);
311            }else{
312                $error_check_types = $item['error_check_types'];
313            }
314            $arrErrorCheckTypes = explode(',', $error_check_types);
315            foreach($arrErrorCheckTypes as $key => $val) {
316                if(trim($val) == "") {
317                    unset($arrErrorCheckTypes[$key]);
318                }else{
319                    $arrErrorCheckTypes[$key] = trim($val);
320                }
321            }
322            // パラメーター登録
323            $objFormParam->addParam(
324                    $item['disp_name']
325                    , $col
326                    , constant($item['size_const_type'])
327                    , $item['mb_convert_kana_option']
328                    , $arrErrorCheckTypes
329                    , $item['default']
330                    , ($item['rw_flg'] != CSV_COLUMN_RW_FLG_READ_ONLY) ? true : false
331                    );
332        }
333    }
334
335    /**
336     * 入力チェックを行う.
337     *
338     * @return void
339     */
340    function lfCheckError(&$objFormParam) {
341        // 入力データを渡す。
342        $arrRet =  $objFormParam->getHashArray();
343        $objErr = new SC_CheckError_Ex($arrRet);
344        $objErr->arrErr = $objFormParam->checkError(false);
345        // HTMLタグチェックの実行
346        foreach($this->arrTagCheckItem as $item) {
347            $objErr->doFunc(array( $item['disp_name'], $item['col'], $this->arrAllowedTag), array("HTML_TAG_CHECK"));
348        }
349        // このフォーム特有の複雑系のエラーチェックを行う
350        if(count($objErr->arrErr) == 0) {
351            $objErr->arrErr = $this->lfCheckErrorDetail($arrRet, $objErr->arrErr);
352        }
353        return $objErr->arrErr;
354    }
355
356    /**
357     * 保存先テーブル情報の初期化を行う.
358     *
359     * @return void
360     */
361    function lfInitTableInfo() {
362        $objQuery =& SC_Query_Ex::getSingletonInstance();
363        $this->arrProductColumn = $objQuery->listTableFields('dtb_products');
364        $this->arrProductClassColumn = $objQuery->listTableFields('dtb_products_class');
365    }
366
367    /**
368     * 商品登録を行う.
369     *
370     * FIXME: 商品登録の実処理自体は、LC_Page_Admin_Products_Productと共通化して欲しい。
371     *
372     * @param SC_Query $objQuery SC_Queryインスタンス
373     * @param string|integer $line 処理中の行数
374     * @return void
375     */
376    function lfRegistProduct($objQuery, $line = "", &$objFormParam) {
377        $objProduct = new SC_Product_Ex();
378        // 登録データ対象取得
379        $arrList = $objFormParam->getHashArray();
380        // 登録時間を生成(DBのnow()だとcommitした際、すべて同一の時間になってしまう)
381        $arrList['update_date'] = $this->lfGetDbFormatTimeWithLine($line);
382
383        // 商品登録情報を生成する。
384        // 商品テーブルのカラムに存在しているもののうち、Form投入設定されていないデータは上書きしない。
385        $sqlval = SC_Utils_Ex::sfArrayIntersectKeys($arrList, $this->arrProductColumn);
386
387        // 必須入力では無い項目だが、空文字では問題のある特殊なカラム値の初期値設定
388        $sqlval = $this->lfSetProductDefaultData($sqlval);
389
390        if($sqlval['product_id'] != "") {
391            // 同じidが存在すればupdate存在しなければinsert
392            $where = "product_id = ?";
393            $product_count = $objQuery->count("dtb_products", $where, array($sqlval['product_id']));
394            if($product_count > 0){
395                $objQuery->update("dtb_products", $sqlval, $where, array($sqlval['product_id']));
396            }else{
397                $sqlval['create_date'] = $arrList['update_date'];
398                // INSERTの実行
399                $objQuery->insert("dtb_products", $sqlval);
400                // シーケンスの調整
401                $seq_count = $objQuery->currVal('dtb_products_product_id');
402                if($seq_count < $sqlval['product_id']){
403                    $objQuery->setVal('dtb_products_product_id', $sqlval['product_id'] + 1);
404                }
405            }
406            $product_id = $sqlval['product_id'];
407        } else {
408            // 新規登録
409            $sqlval['product_id'] = $objQuery->nextVal('dtb_products_product_id');
410            $product_id = $sqlval['product_id'];
411            $sqlval['create_date'] = $arrList['update_date'];
412            // INSERTの実行
413            $objQuery->insert("dtb_products", $sqlval);
414        }
415
416        // カテゴリ登録
417        if($arrList['category_ids'] != "") {
418            $arrCategory_id = explode(',', $arrList['category_ids']);
419            $this->objDb->updateProductCategories($arrCategory_id, $product_id);
420        }
421        // ステータス登録
422        if($arrList['product_statuses'] != "") {
423            $arrStatus_id = explode(',', $arrList['product_statuses']);
424            $objProduct->setProductStatus($product_id, $arrStatus_id);
425        }
426
427        // 商品規格情報を登録する
428        $this->lfRegistProductClass($objQuery, $arrList, $product_id, $arrList['product_class_id']);
429
430        // 関連商品登録
431        $this->lfRegistReccomendProducts($objQuery, $arrList, $product_id);
432    }
433
434    /**
435     * 商品規格登録を行う.
436     *
437     * FIXME: 商品規格登録の実処理自体は、LC_Page_Admin_Products_Productと共通化して欲しい。
438     *
439     * @param SC_Query $objQuery SC_Queryインスタンス
440     * @param array $arrList 商品規格情報配列
441     * @param integer $product_id 商品ID
442     * @param integer $product_class_id 商品規格ID
443     * @return void
444     */
445    function lfRegistProductClass($objQuery, $arrList, $product_id, $product_class_id) {
446        $objProduct = new SC_Product_Ex();
447        // 商品規格登録情報を生成する。
448        // 商品規格テーブルのカラムに存在しているもののうち、Form投入設定されていないデータは上書きしない。
449        $sqlval = SC_Utils_Ex::sfArrayIntersectKeys($arrList, $this->arrProductClassColumn);
450        // 必須入力では無い項目だが、空文字では問題のある特殊なカラム値の初期値設定
451        $sqlval = $this->lfSetProductClassDefaultData($sqlval);
452
453        if($product_class_id == "") {
454            // 新規登録
455            $sqlval['product_id'] = $product_id;
456            $sqlval['product_class_id'] = $objQuery->nextVal('dtb_products_class_product_class_id');
457            $sqlval['create_date'] = $arrList['update_date'];
458            // INSERTの実行
459            $objQuery->insert("dtb_products_class", $sqlval);
460            $product_class_id = $sqlval['product_class_id'];
461        } else {
462            // UPDATEの実行
463            $where = "product_class_id = ?";
464            $objQuery->update("dtb_products_class", $sqlval, $where, array($product_class_id));
465        }
466        // 支払い方法登録
467        if($arrList['product_payment_ids'] != "") {
468            $arrPayment_id = explode(',', $arrList['product_payment_ids']);
469            $objProduct->setPaymentOptions($product_class_id, $arrPayment_id);
470        }
471    }
472
473    /**
474     * 関連商品登録を行う.
475     *
476     * FIXME: 商品規格登録の実処理自体は、LC_Page_Admin_Products_Productと共通化して欲しい。
477     *        DELETE/INSERT ではなく UPDATEへの変更も・・・
478     *
479     * @param SC_Query $objQuery SC_Queryインスタンス
480     * @param array $arrList 商品規格情報配列
481     * @param integer $product_id 商品ID
482     * @return void
483     */
484    function lfRegistReccomendProducts($objQuery, $arrList, $product_id) {
485        $objQuery->delete("dtb_recommend_products", "product_id = ?", array($product_id));
486        for($i = 1; $i <= RECOMMEND_PRODUCT_MAX; $i++) {
487            $keyname = "recommend_product_id" . $i;
488            $comment_key = "recommend_comment" . $i;
489            if($arrList[$keyname] != "") {
490                $arrProduct = $objQuery->select("product_id", "dtb_products", "product_id = ?", array($arrList[$keyname]));
491                if($arrProduct[0]['product_id'] != "") {
492                    $arrval['product_id'] = $product_id;
493                    $arrval['recommend_product_id'] = $arrProduct[0]['product_id'];
494                    $arrval['comment'] = $arrList[$comment_key];
495                    $arrval['update_date'] = $arrList['update_date'];
496                    $arrval['create_date'] = $arrList['update_date'];
497                    $arrval['creator_id'] = $_SESSION['member_id'];
498                    $arrval['rank'] = RECOMMEND_PRODUCT_MAX - $i + 1;
499                    $objQuery->insert("dtb_recommend_products", $arrval);
500                }
501            }
502        }
503    }
504
505    /**
506     * 初期値の設定
507     *
508     * @param array $arrCSVFrame CSV構造配列
509     * @return array $arrCSVFrame CSV構造配列
510     */
511    function lfSetParamDefaultValue(&$arrCSVFrame) {
512        foreach($arrCSVFrame as $key => $val) {
513            switch($val['col']) {
514                case 'status':
515                    $arrCSVFrame[$key]['default'] = DEFAULT_PRODUCT_DISP;
516                    break;
517                case 'del_flg':
518                    $arrCSVFrame[$key]['default'] = '0';
519                    break;
520                case 'point_rate':
521                    $arrCSVFrame[$key]['default'] = $this->arrInfo['point_rate'];
522                    break;
523                case 'product_type_id':
524                    $arrCSVFrame[$key]['default'] = DEFAULT_PRODUCT_DOWN;
525                    break;
526                case 'product_payment_ids':
527                    $arrCSVFrame[$key]['default'] = implode(',',array_keys($this->arrPayments));
528                    break;
529                case 'stock_unlimited':
530                    $arrCSVFrame[$key]['default'] = UNLIMITED_FLG_LIMITED;
531                default:
532                    break;
533            }
534        }
535        return $arrCSVFrame;
536    }
537
538    /**
539     * 商品データ登録前に特殊な値の持ち方をする部分のデータ部分の初期値補正を行う
540     *
541     * @param array $sqlval 商品登録情報配列
542     * @return $sqlval 登録情報配列
543     */
544    function lfSetProductDefaultData(&$sqlval) {
545        //新規登録時のみ設定する項目
546        if( $sqlval['product_id'] == "") {
547            if($sqlval['status'] == "") {
548                $sqlval['status'] = DEFAULT_PRODUCT_DISP;
549            }
550        }
551        //共通で空欄時に上書きする項目
552        if($sqlval['del_flg'] == ""){
553            $sqlval['del_flg'] = '0'; //有効
554        }
555        if($sqlval['creator_id'] == "") {
556            $sqlval['creator_id'] = $_SESSION['member_id'];
557        }
558        return $sqlval;
559    }
560
561    /**
562     * 商品規格データ登録前に特殊な値の持ち方をする部分のデータ部分の初期値補正を行う
563     *
564     * @param array $sqlval 商品登録情報配列
565     * @return $sqlval 登録情報配列
566     */
567    function lfSetProductClassDefaultData(&$sqlval) {
568        //新規登録時のみ設定する項目
569        if($sqlval['product_class_id'] == "") {
570            if($sqlval['point_rate'] == "") {
571                $sqlval['point_rate'] = $this->arrInfo['point_rate'];
572            }
573            if($sqlval['product_type_id'] == "") {
574                $sqlval['product_type_id'] = DEFAULT_PRODUCT_DOWN;
575            }
576            // TODO: 在庫数、無制限フラグの扱いについて仕様がぶれているので要調整
577            if($sqlval['stock'] == "" and $sqlval['stock_unlimited'] != UNLIMITED_FLG_UNLIMITED) {
578                //在庫数設定がされておらず、かつ無制限フラグが設定されていない場合、強制無制限
579                $sqlval['stock_unlimited'] = UNLIMITED_FLG_UNLIMITED;
580            }elseif($sqlval['stock'] != "" and $sqlval['stock_unlimited'] != UNLIMITED_FLG_UNLIMITED) {
581                //在庫数設定時は在庫無制限フラグをクリア
582                $sqlval['stock_unlimited'] = UNLIMITED_FLG_LIMITED;
583            }elseif($sqlval['stock'] != "" and $sqlval['stock_unlimited'] == UNLIMITED_FLG_UNLIMITED) {
584                //在庫無制限フラグ設定時は在庫数をクリア
585                $sqlval['stock'] = '';
586            }
587        }else{
588            //更新時のみ設定する項目
589            if(array_key_exists('stock_unlimited', $sqlval) and $sqlval['stock_unlimited'] == UNLIMITED_FLG_UNLIMITED) {
590                $sqlval['stock'] = '';
591            }
592        }
593        //共通で設定する項目
594        if($sqlval['del_flg'] == ""){
595            $sqlval['del_flg'] = '0'; //有効
596        }
597        if($sqlval['creator_id'] == "") {
598            $sqlval['creator_id'] = $_SESSION['member_id'];
599        }
600        return $sqlval;
601    }
602
603    /**
604     * このフォーム特有の複雑な入力チェックを行う.
605     *
606     * @param array 確認対象データ
607     * @param array エラー配列
608     * @return array エラー配列
609     */
610    function lfCheckErrorDetail($item, $arrErr) {
611        // 規格IDの存在チェック
612        if(!$this->lfIsDbRecord('dtb_products_class', 'product_class_id', $item)) {
613            $arrErr['product_class_id'] = "※ 指定の商品規格IDは、登録されていません。";
614        }
615        // 商品ID、規格IDの組合せチェック
616        if(array_search('product_class_id', $this->arrFormKeyList) !== FALSE
617                and $item['product_class_id'] != "") {
618            if($item['product_id'] == "") {
619                $arrErr['product_class_id'] = "※ 商品規格ID指定時には商品IDの指定が必須です。";
620            }else{
621                if(!$this->objDb->sfIsRecord('dtb_products_class', 'product_id, product_class_id'
622                        , array($item['product_id'], $item['product_class_id']))) {
623                    $arrErr['product_class_id'] = "※ 指定の商品IDと商品規格IDの組合せは正しくありません。";
624                }
625            }
626        }
627        // 規格組合せIDの存在チェック
628//        if(!$this->lfIsDbRecord('dtb_class_combination', 'class_combination_id', $item)) {
629//      SC_Utils::sfIsRecord が del_flg が無いと使えない為、個別処理
630        if(array_search('class_combination_id', $this->arrFormKeyList) !== FALSE
631                and $item['class_combination_id'] != "" ) {
632            $objQuery =& SC_Query_Ex::getSingletonInstance();
633            $ret = $objQuery->get('class_combination_id', 'dtb_class_combination', 'class_combination_id = ?', array($item['class_combination_id']));
634            if($ret == "") {
635                $arrErr['class_combination_id'] = "※ 指定の規格組合せIDは、登録されていません。";
636            }
637        }
638        // 表示ステータスの存在チェック
639        if(!$this->lfIsArrayRecord($this->arrDISP, 'status', $item)) {
640            $arrErr['status'] = "※ 指定の表示ステータスは、登録されていません。";
641        }
642        // メーカーIDの存在チェック
643        if(!$this->lfIsArrayRecord($this->arrMaker, 'maker_id', $item)) {
644            $arrErr['maker_id'] = "※ 指定のメーカーIDは、登録されていません。";
645        }
646        // 発送日目安IDの存在チェック
647        if(!$this->lfIsArrayRecord($this->arrDELIVERYDATE, 'deliv_date_id', $item)) {
648            $arrErr['deliv_date_id'] = "※ 指定の発送日目安IDは、登録されていません。";
649        }
650        // 発送日目安IDの存在チェック
651        if(!$this->lfIsArrayRecord($this->arrProductType, 'product_type_id', $item)) {
652            $arrErr['product_type_id'] = "※ 指定の商品種別IDは、登録されていません。";
653        }
654        // 関連商品IDの存在チェック
655        for($i = 1; $i <= RECOMMEND_PRODUCT_MAX; $i++) {
656            if(array_search('recommend_product_id' . $i, $this->arrFormKeyList) !== FALSE
657                    and $item['recommend_product_id' . $i] != ""
658                    and !$this->objDb->sfIsRecord('dtb_products', 'product_id', (array)$item['recommend_product_id' . $i]) ) {
659                $arrErr['recommend_product_id' . $i] = "※ 指定の関連商品ID($i)は、登録されていません。";
660            }
661        }
662        // カテゴリIDの存在チェック
663        if(!$this->lfIsDbRecordMulti('dtb_category', 'category_id', 'category_ids', $item, ',')) {
664            $arrErr['category_ids'] = "※ 指定のカテゴリIDは、登録されていません。";
665        }
666        // ステータスIDの存在チェック
667        if(!$this->lfIsArrayRecordMulti($this->arrSTATUS, 'product_statuses', $item, ',')) {
668            $arrErr['product_statuses'] = "※ 指定のステータスIDは、登録されていません。";
669        }
670        // 支払い方法IDの存在チェック
671        if(!$this->lfIsArrayRecordMulti($this->arrPayments, 'product_payment_ids', $item, ',')) {
672            $arrErr['product_payment_ids'] = "※ 指定の支払い方法IDは、登録されていません。";
673        }
674        // 削除フラグのチェック
675        if(array_search('del_flg', $this->arrFormKeyList) !== FALSE
676                and $item['del_flg'] != "") {
677            if(!($item['del_flg'] == "0" or $item['del_flg'] == "1")) {
678                $arrErr['del_flg'] = "※ 削除フラグは「0」(有効)、「1」(削除)のみが有効な値です。";
679            }
680        }
681/*
682    TODO: 在庫数の扱いが2.4仕様ではぶれているのでどうするか・・
683        // 在庫数/在庫無制限フラグの有効性に関するチェック
684        if($item['stock'] == "") {
685            if(array_search('stock_unlimited', $this->arrFormKeyList) === FALSE) {
686                $arrErr['stock'] = "※ 在庫数は必須です(無制限フラグ項目がある場合のみ空欄許可)。";
687            }else if($item['stock_unlimited'] != UNLIMITED_FLG_UNLIMITED) {
688                $arrErr['stock'] = "※ 在庫数または在庫無制限フラグのいずれかの入力が必須です。";
689            }
690        }
691*/
692        // ダウンロード商品チェック
693        if(array_search('product_type_id', $this->arrFormKeyList) !== FALSE
694                 and $item['product_type_id'] == PRODUCT_TYPE_NORMAL) {
695            //実商品の場合
696            if( $item['down_filename'] != "") {
697                $arrErr['down_filename'] = "※ 実商品の場合はダウンロードファイル名は入力できません。";
698            }
699            if( $item['down_realfilename'] != "") {
700                $arrErr['down_realfilename'] = "※ 実商品の場合はダウンロード商品用ファイルアップロードは入力できません。";
701            }
702        }elseif(array_search('product_type_id', $this->arrFormKeyList) !== FALSE
703                and $item['product_type_id'] == PRODUCT_TYPE_DOWNLOAD) {
704            //ダウンロード商品の場合
705            if( $item['down_filename'] == "") {
706                $arrErr['down_filename'] = "※ ダウンロード商品の場合はダウンロードファイル名は必須です。";
707            }
708            if( $item['down_realfilename'] == "") {
709                $arrErr['down_realfilename'] = "※ ダウンロード商品の場合はダウンロード商品用ファイルアップロードは必須です。";
710            }
711        }
712        return $arrErr;
713    }
714
715    // TODO: ここから下のルーチンは汎用ルーチンとして移動が望ましい
716
717    /**
718     * 指定された行番号をmicrotimeに付与してDB保存用の時間を生成する。
719     * トランザクション内のnow()は全てcommit()時の時間に統一されてしまう為。
720     *
721     * @param string $line_no 行番号
722     * @return string $time DB保存用の時間文字列
723     */
724    function lfGetDbFormatTimeWithLine($line_no = '') {
725        $time = date("Y-m-d H:i:s");
726        // 秒以下を生成
727        if($line_no != '') {
728            $microtime = sprintf("%06d", $line_no);
729            $time .= ".$microtime";
730        }
731        return $time;
732    }
733
734    /**
735     * 指定されたキーと複数値の有効性の配列内確認
736     *
737     * @param string $arr チェック対象配列
738     * @param string $keyname フォームキー名
739     * @param array  $item 入力データ配列
740     * @param string $delimiter 分割文字
741     * @return boolean true:有効なデータがある false:有効ではない
742     */
743    function lfIsArrayRecordMulti($arr, $keyname, $item, $delimiter = ',') {
744        if(array_search($keyname, $this->arrFormKeyList) === FALSE) {
745            return true;
746        }
747        if($item[$keyname] == "") {
748            return true;
749        }
750        $arrItems = explode($delimiter, $item[$keyname]);
751        //空項目のチェック 1つでも空指定があったら不正とする。
752        if(array_search("", $arrItems) !== FALSE) {
753            return false;
754        }
755        //キー項目への存在チェック
756        foreach($arrItems as $item) {
757            if(!array_key_exists($item, $arr)) {
758                return false;
759            }
760        }
761        return true;
762    }
763
764    /**
765     * 指定されたキーと複数値の有効性のDB確認
766     *
767     * @param string $table テーブル名
768     * @param string $tblkey テーブルキー名
769     * @param string $keyname フォームキー名
770     * @param array  $item 入力データ配列
771     * @param string $delimiter 分割文字
772     * @return boolean true:有効なデータがある false:有効ではない
773     */
774    function lfIsDbRecordMulti($table, $tblkey, $keyname, $item, $delimiter = ',') {
775        if(array_search($keyname, $this->arrFormKeyList) === FALSE) {
776            return true;
777        }
778        if($item[$keyname] == "") {
779            return true;
780        }
781        $arrItems = explode($delimiter, $item[$keyname]);
782        //空項目のチェック 1つでも空指定があったら不正とする。
783        if(array_search("", $arrItems) !== FALSE) {
784            return false;
785        }
786        $count = count($arrItems);
787        $where = $tblkey ." IN (" . implode(",", array_fill(0, $count, "?")) . ")";
788
789        $objQuery =& SC_Query_Ex::getSingletonInstance();
790        $db_count = $objQuery->count($table, $where, $arrItems);
791        if($count != $db_count) {
792            return false;
793        }
794        return true;
795    }
796
797    /**
798     * 指定されたキーと値の有効性のDB確認
799     *
800     * @param string $table テーブル名
801     * @param string $keyname キー名
802     * @param array  $item 入力データ配列
803     * @return boolean true:有効なデータがある false:有効ではない
804     */
805    function lfIsDbRecord($table, $keyname, $item) {
806        if(array_search($keyname, $this->arrFormKeyList) !== FALSE  //入力対象である
807                and $item[$keyname] != ""   // 空ではない
808                and !$this->objDb->sfIsRecord($table, $keyname, (array)$item[$keyname]) //DBに存在するか
809                ) {
810            return false;
811        }
812        return true;
813    }
814
815    /**
816     * 指定されたキーと値の有効性の配列内確認
817     *
818     * @param string $arr チェック対象配列
819     * @param string $keyname キー名
820     * @param array  $item 入力データ配列
821     * @return boolean true:有効なデータがある false:有効ではない
822     */
823    function lfIsArrayRecord($arr, $keyname, $item) {
824        if(array_search($keyname, $this->arrFormKeyList) !== FALSE //入力対象である
825                and $item[$keyname] != "" // 空ではない
826                and !array_key_exists($item[$keyname], $arr) //配列に存在するか
827                ) {
828            return false;
829        }
830        return true;
831    }
832}
Note: See TracBrowser for help on using the repository browser.