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

Revision 21514, 33.0 KB checked in by Seasoft, 12 years ago (diff)

#1625 (typo修正・ソース整形・ソースコメントの改善)

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