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

Revision 22251, 33.8 KB checked in by undertree, 9 years ago (diff)

#1609 CSV_SIZE による制限を画面上に表示する

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