source: branches/feature-module-update/data/class/pages/admin/products/LC_Page_Admin_Products_UploadCSV.php @ 16582

Revision 16582, 22.2 KB checked in by nanasess, 17 years ago (diff)

ライセンス表記変更

  • Property svn:keywords set to Id Revision Date
  • 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-2007 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_PATH . "pages/LC_Page.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 */
34class LC_Page_Admin_Products_UploadCSV extends LC_Page {
35
36    // }}}
37    // {{{ functions
38
39    /** フォームパラメータ */
40    var $objFormParam;
41
42    /** SC_UploadFile インスタンス */
43    var $objUpfile;
44
45    /**
46     * Page を初期化する.
47     *
48     * @return void
49     */
50    function init() {
51        parent::init();
52        $this->tpl_mainpage = 'products/upload_csv.tpl';
53        $this->tpl_subnavi = 'products/subnavi.tpl';
54        $this->tpl_mainno = 'products';
55        $this->tpl_subno = 'upload_csv';
56    }
57
58    /**
59     * Page のプロセス.
60     *
61     * @return void
62     */
63    function process() {
64        $conn = new SC_DBConn();
65        $objView = new SC_AdminView();
66        $objSess = new SC_Session();
67        $objDb = new SC_Helper_DB_Ex();
68
69        // 認証可否の判定
70        SC_Utils_Ex::sfIsSuccess($objSess);
71
72        // ファイル管理クラス
73        $this->objUpFile = new SC_UploadFile(IMAGE_TEMP_DIR, IMAGE_SAVE_DIR);
74        // ファイル情報の初期化
75        $this->lfInitFile();
76        // パラメータ管理クラス
77        $this->objFormParam = new SC_FormParam();
78        // パラメータ情報の初期化
79        $this->lfInitParam();
80        $colmax = $this->objFormParam->getCount();
81        $this->objFormParam->setHtmlDispNameArray();
82        $this->arrTitle = $this->objFormParam->getHtmlDispNameArray();
83
84        if (!isset($_POST['mode'])) $_POST['mode'] = "";
85
86        switch($_POST['mode']) {
87        case 'csv_upload':
88            $err = false;
89            // エラーチェック
90            $arrErr['csv_file'] = $this->objUpFile->makeTempFile('csv_file');
91
92            if($arrErr['css_file'] == "") {
93                $arrErr = $this->objUpFile->checkEXISTS();
94            }
95
96            // 実行時間を制限しない
97            set_time_limit(0);
98
99            // 出力をバッファリングしない(==日本語自動変換もしない)
100            ob_end_clean();
101
102            // IEのために256バイト空文字出力
103            echo str_pad('',256);
104
105            if(empty($arrErr['csv_file'])) {
106                // 一時ファイル名の取得
107                $filepath = $this->objUpFile->getTempFilePath('csv_file');
108                // エンコード
109                $enc_filepath = SC_Utils_Ex::sfEncodeFile($filepath,
110                                                          CHAR_CODE, CSV_TEMP_DIR);
111
112                // レコード数を得る
113                $rec_count = $this->lfCSVRecordCount($enc_filepath);
114
115                $fp = fopen($enc_filepath, "r");
116                $line = 0;      // 行数
117                $regist = 0;    // 登録数
118
119                $objQuery = new SC_Query();
120                $objQuery->begin();
121
122                echo "■ CSV登録進捗状況 <br/><br/>\n";
123
124                while(!feof($fp) && !$err) {
125                    $arrCSV = fgetcsv($fp, CSV_LINE_MAX);
126
127                    // 行カウント
128                    $line++;
129
130                    if($line <= 1) {
131                        continue;
132                    }
133
134                    // 項目数カウント
135                    $max = count($arrCSV);
136
137                    // 項目数が1以下の場合は無視する
138                    if($max <= 1) {
139                        continue;
140                    }
141
142                    // 項目数チェック
143                    if($max != $colmax) {
144                        echo "※ 項目数が" . $max . "個検出されました。項目数は" . $colmax . "個になります。</br>\n";
145                        $err = true;
146                    } else {
147                        // シーケンス配列を格納する。
148                        $this->objFormParam->setParam($arrCSV, true);
149                        $arrRet = $this->objFormParam->getHashArray();
150                        $this->objFormParam->setParam($arrRet);
151                        // 入力値の変換
152                        $this->objFormParam->convParam();
153                        // <br>なしでエラー取得する。
154                        $arrCSVErr = $this->lfCheckError();
155                    }
156
157                    // 入力エラーチェック
158                    if(count($arrCSVErr) > 0) {
159                        echo "<font color=\"red\">■" . $line . "行目でエラーが発生しました。</font></br>\n";
160                        foreach($arrCSVErr as $val) {
161                            $this->printError($val);
162                        }
163                        $err = true;
164                    }
165
166                    if(!$err) {
167                        $this->lfRegistProduct($objQuery, $line);
168                        $regist++;
169                    }
170                    $arrParam = $this->objFormParam->getHashArray();
171
172                    if(!$err) echo $line." / ".$rec_count. "行目 (商品ID:".$arrParam['product_id']." / 商品名:".$arrParam['name'].")\n<br />";
173                    flush();
174                }
175                fclose($fp);
176
177                if(!$err) {
178                    $objQuery->commit();
179                    echo "■" . $regist . "件のレコードを登録しました。";
180                    // 商品件数カウント関数の実行
181                    $objDb->sfCategory_Count($objQuery);
182                } else {
183                    $objQuery->rollback();
184                }
185            } else {
186                foreach($arrErr as $val) {
187                    $this->printError($val);
188                }
189            }
190            echo "<br/><a href=\"javascript:window.close()\">→閉じる</a>";
191            flush();
192            exit;
193            break;
194        default:
195            break;
196        }
197
198        $objView->assignobj($this);
199        $objView->display(MAIN_FRAME);
200    }
201
202    /**
203     * デストラクタ.
204     *
205     * @return void
206     */
207    function destroy() {
208        parent::destroy();
209    }
210
211
212    /**
213     * ファイル情報の初期化を行う.
214     *
215     * @return void
216     */
217    function lfInitFile() {
218        $this->objUpFile->addFile("CSVファイル", 'csv_file', array('csv'),
219                                  CSV_SIZE, true, 0, 0, false);
220    }
221
222    /**
223     * 入力情報の初期化を行う.
224     *
225     * @return void
226     */
227    function lfInitParam() {
228
229        $this->objFormParam->addParam("商品ID", "product_id", INT_LEN, "n", array("MAX_LENGTH_CHECK","NUM_CHECK"));
230        $this->objFormParam->addParam("商品規格ID", "product_class_id", INT_LEN, "n", array("MAX_LENGTH_CHECK","NUM_CHECK"));
231
232        $this->objFormParam->addParam("規格名1", "dummy1");
233        $this->objFormParam->addParam("規格名2", "dummy2");
234
235        $this->objFormParam->addParam("商品名", "name", STEXT_LEN, "KVa", array("EXIST_CHECK","SPTAB_CHECK","MAX_LENGTH_CHECK"));
236        $this->objFormParam->addParam("公開フラグ(1:公開 2:非公開)", "status", INT_LEN, "n", array("EXIST_CHECK","MAX_LENGTH_CHECK","NUM_CHECK"));
237        $this->objFormParam->addParam("商品ステータス", "product_flag", INT_LEN, "n", array("EXIST_CHECK","MAX_LENGTH_CHECK","NUM_CHECK"));
238        $this->objFormParam->addParam("商品コード", "product_code", STEXT_LEN, "KVa", array("SPTAB_CHECK","MAX_LENGTH_CHECK"));
239        $this->objFormParam->addParam(NORMAL_PRICE_TITLE, "price01", PRICE_LEN, "n", array("MAX_LENGTH_CHECK","NUM_CHECK"));
240        $this->objFormParam->addParam(SALE_PRICE_TITLE, "price02", PRICE_LEN, "n", array("EXIST_CHECK","MAX_LENGTH_CHECK","NUM_CHECK"));
241        $this->objFormParam->addParam("在庫数", "stock", INT_LEN, "n", array("MAX_LENGTH_CHECK","NUM_CHECK"));
242        $this->objFormParam->addParam("送料", "deliv_fee", PRICE_LEN, "n", array("MAX_LENGTH_CHECK","NUM_CHECK"));
243        $this->objFormParam->addParam("ポイント付与率", "point_rate", PERCENTAGE_LEN, "n", array("EXIST_CHECK","MAX_LENGTH_CHECK","NUM_CHECK"));
244        $this->objFormParam->addParam("購入制限", "sale_limit", INT_LEN, "n", array("MAX_LENGTH_CHECK","NUM_CHECK"));
245        $this->objFormParam->addParam("メーカーURL", "comment1", URL_LEN, "KVa", array("SPTAB_CHECK","URL_CHECK","MAX_LENGTH_CHECK"));
246        $this->objFormParam->addParam("検索ワード", "comment3", LLTEXT_LEN, "KVa", array("SPTAB_CHECK","MAX_LENGTH_CHECK"));
247        $this->objFormParam->addParam("一覧-メインコメント", "main_list_comment", LTEXT_LEN, "KVa", array("EXIST_CHECK","SPTAB_CHECK","MAX_LENGTH_CHECK"));
248        $this->objFormParam->addParam("一覧-メイン画像", "main_list_image", LTEXT_LEN, "KVa", array("EXIST_CHECK","FILE_EXISTS","SPTAB_CHECK","MAX_LENGTH_CHECK"));
249        $this->objFormParam->addParam("メインコメント", "main_comment", LTEXT_LEN, "KVa", array("EXIST_CHECK","SPTAB_CHECK","MAX_LENGTH_CHECK"));
250        $this->objFormParam->addParam("メイン画像", "main_image", LTEXT_LEN, "KVa", array("EXIST_CHECK","FILE_EXISTS","SPTAB_CHECK","MAX_LENGTH_CHECK"));
251        $this->objFormParam->addParam("メイン拡大画像", "main_large_image", LTEXT_LEN, "KVa", array("FILE_EXISTS","SPTAB_CHECK","MAX_LENGTH_CHECK"));
252        $this->objFormParam->addParam("カラー比較画像", "file1", LTEXT_LEN, "KVa", array("FILE_EXISTS","SPTAB_CHECK","MAX_LENGTH_CHECK"));
253        $this->objFormParam->addParam("商品詳細ファイル", "file2", LTEXT_LEN, "KVa", array("FILE_EXISTS","SPTAB_CHECK","MAX_LENGTH_CHECK"));
254        $this->objFormParam->addParam("詳細-サブタイトル(1)", "sub_title1", LTEXT_LEN, "KVa", array("SPTAB_CHECK","MAX_LENGTH_CHECK"));
255        $this->objFormParam->addParam("詳細-サブコメント(1)", "sub_comment1", LTEXT_LEN, "KVa", array("SPTAB_CHECK","MAX_LENGTH_CHECK"));
256        $this->objFormParam->addParam("詳細-サブ画像(1)", "sub_image1", LTEXT_LEN, "KVa", array("FILE_EXISTS","SPTAB_CHECK","MAX_LENGTH_CHECK"));
257        $this->objFormParam->addParam("詳細-サブ拡大画像(1)", "sub_large_image1", LTEXT_LEN, "KVa", array("FILE_EXISTS","SPTAB_CHECK","MAX_LENGTH_CHECK"));
258
259        $this->objFormParam->addParam("詳細-サブタイトル(2)", "sub_title2", LTEXT_LEN, "KVa", array("SPTAB_CHECK","MAX_LENGTH_CHECK"));
260        $this->objFormParam->addParam("詳細-サブコメント(2)", "sub_comment2", LTEXT_LEN, "KVa", array("SPTAB_CHECK","MAX_LENGTH_CHECK"));
261        $this->objFormParam->addParam("詳細-サブ画像(2)", "sub_image2", LTEXT_LEN, "KVa", array("FILE_EXISTS","SPTAB_CHECK","MAX_LENGTH_CHECK"));
262        $this->objFormParam->addParam("詳細-サブ拡大画像(2)", "sub_large_image2", LTEXT_LEN, "KVa", array("FILE_EXISTS","SPTAB_CHECK","MAX_LENGTH_CHECK"));
263
264        $this->objFormParam->addParam("詳細-サブタイトル(3)", "sub_title3", LTEXT_LEN, "KVa", array("SPTAB_CHECK","MAX_LENGTH_CHECK"));
265        $this->objFormParam->addParam("詳細-サブコメント(3)", "sub_comment3", LTEXT_LEN, "KVa", array("SPTAB_CHECK","MAX_LENGTH_CHECK"));
266        $this->objFormParam->addParam("詳細-サブ画像(3)", "sub_image3", LTEXT_LEN, "KVa", array("FILE_EXISTS","SPTAB_CHECK","MAX_LENGTH_CHECK"));
267        $this->objFormParam->addParam("詳細-サブ拡大画像(3)", "sub_large_image3", LTEXT_LEN, "KVa", array("FILE_EXISTS","SPTAB_CHECK","MAX_LENGTH_CHECK"));
268
269        $this->objFormParam->addParam("詳細-サブタイトル(4)", "sub_title4", LTEXT_LEN, "KVa", array("SPTAB_CHECK","MAX_LENGTH_CHECK"));
270        $this->objFormParam->addParam("詳細-サブコメント(4)", "sub_comment4", LTEXT_LEN, "KVa", array("SPTAB_CHECK","MAX_LENGTH_CHECK"));
271        $this->objFormParam->addParam("詳細-サブ画像(4)", "sub_image4", LTEXT_LEN, "KVa", array("FILE_EXISTS","SPTAB_CHECK","MAX_LENGTH_CHECK"));
272        $this->objFormParam->addParam("詳細-サブ拡大画像(4)", "sub_large_image4", LTEXT_LEN, "KVa", array("FILE_EXISTS","SPTAB_CHECK","MAX_LENGTH_CHECK"));
273
274        $this->objFormParam->addParam("詳細-サブタイトル(5)", "sub_title5", LTEXT_LEN, "KVa", array("SPTAB_CHECK","MAX_LENGTH_CHECK"));
275        $this->objFormParam->addParam("詳細-サブコメント(5)", "sub_comment5", LTEXT_LEN, "KVa", array("SPTAB_CHECK","MAX_LENGTH_CHECK"));
276        $this->objFormParam->addParam("詳細-サブ画像(5)", "sub_image5", LTEXT_LEN, "KVa", array("FILE_EXISTS","SPTAB_CHECK","MAX_LENGTH_CHECK"));
277        $this->objFormParam->addParam("詳細-サブ拡大画像(5)", "sub_large_image5", LTEXT_LEN, "KVa", array("FILE_EXISTS","SPTAB_CHECK","MAX_LENGTH_CHECK"));
278
279        $this->objFormParam->addParam("発送日目安", "deliv_date_id", INT_LEN, "n", array("MAX_LENGTH_CHECK","NUM_CHECK"));
280
281        for ($cnt = 1; $cnt <= RECOMMEND_PRODUCT_MAX; $cnt++) {
282            $this->objFormParam->addParam("おすすめ商品($cnt)", "recommend_product_id$cnt", INT_LEN, "n", array("MAX_LENGTH_CHECK","NUM_CHECK"));
283            $this->objFormParam->addParam("詳細-サブコメント($cnt)", "recommend_comment$cnt", LTEXT_LEN, "KVa", array("SPTAB_CHECK","MAX_LENGTH_CHECK"));
284        }
285
286        $this->objFormParam->addParam("商品カテゴリ", "category_id", STEXT_LEN, "n", array("EXIST_CHECK", "SPTAB_CHECK"));
287    }
288
289    /**
290     * 商品登録を行う.
291     *
292     * @param SC_Query $objQuery SC_Queryインスタンス
293     * @param string|integer $line 処理中の行数
294     * @return void
295     */
296    function lfRegistProduct($objQuery, $line = "") {
297
298        $objDb = new SC_Helper_DB_Ex();
299
300        $arrRet = $this->objFormParam->getHashArray();
301
302        // dtb_products以外に登録される値を除外する。
303        foreach($arrRet as $key => $val) {
304            switch($key) {
305            case 'product_code':
306            case 'price01':
307            case 'price02':
308            case 'stock':
309            case 'product_class_id':
310            case 'recommend_product_id1':
311            case 'recommend_product_id2':
312            case 'recommend_product_id3':
313            case 'recommend_product_id4':
314            case 'recommend_product_id5':
315            case 'recommend_product_id6':
316            case 'recommend_comment1':
317            case 'recommend_comment2':
318            case 'recommend_comment3':
319            case 'recommend_comment4':
320            case 'recommend_comment5':
321            case 'recommend_comment6':
322            case 'category_id':
323                break;
324            default:
325                if(!ereg("^dummy", $key)) {
326                    $sqlval[$key] = $val;
327                }
328                break;
329            }
330        }
331
332        // 登録時間を生成(DBのnow()だとcommitした際、すべて同一の時間になってしまう)
333        $time = date("Y-m-d H:i:s");
334        // 秒以下を生成
335        if($line != "") {
336            $microtime = sprintf("%06d", $line);
337            $time .= ".$microtime";
338        }
339        $sqlval['update_date'] = $time;
340        $sqlval['creator_id'] = $_SESSION['member_id'];
341
342        if($sqlval['sale_limit'] == "") {
343            $sqlval['sale_unlimited'] = '1';
344        } else {
345            $sqlval['sale_unlimited'] = '0';
346        }
347
348        if($sqlval['status'] == "") {
349            $sqlval['status'] = 2;
350        }
351
352        if($arrRet['product_id'] != "" && $arrRet['product_class_id'] != "") {
353
354            // UPDATEの実行
355            $where = "product_id = ?";
356            $objQuery->update("dtb_products", $sqlval, $where, array($sqlval['product_id']));
357        } else {
358
359            // 新規登録
360            // postgresqlとmysqlとで処理を分ける
361            if (DB_TYPE == "pgsql") {
362                $product_id = $objQuery->nextval("dtb_products","product_id");
363            }elseif (DB_TYPE == "mysql") {
364                $product_id = $objQuery->get_auto_increment("dtb_products");
365            }
366            $sqlval['product_id'] = $product_id;
367            $sqlval['create_date'] = $time;
368
369            // INSERTの実行
370            $objQuery->insert("dtb_products", $sqlval);
371        }
372
373        // カテゴリ登録
374        $arrCategory_id = explode("|", $arrRet["category_id"]);
375        $objDb->updateProductCategories($arrCategory_id, $sqlval['product_id']);
376
377        // 規格登録
378        $this->lfRegistProductClass($objQuery, $arrRet, $sqlval['product_id'], $arrRet['product_class_id']);
379
380        // おすすめ商品登録
381        $objQuery->delete("dtb_recommend_products", "product_id = ?", array($sqlval['product_id']));
382        for($i = 1; $i <= RECOMMEND_PRODUCT_MAX; $i++) {
383            $keyname = "recommend_product_id" . $i;
384            $comment_key = "recommend_comment" . $i;
385            if($arrRet[$keyname] != "") {
386                $arrProduct = $objQuery->select("product_id", "dtb_products", "product_id = ?", array($arrRet[$keyname]));
387                if($arrProduct[0]['product_id'] != "") {
388                    $arrval['product_id'] = $sqlval['product_id'];
389                    $arrval['recommend_product_id'] = $arrProduct[0]['product_id'];
390                    $arrval['comment'] = $arrRet[$comment_key];
391                    $arrval['update_date'] = "Now()";
392                    $arrval['create_date'] = "Now()";
393                    $arrval['creator_id'] = $_SESSION['member_id'];
394                    $arrval['rank'] = RECOMMEND_PRODUCT_MAX - $i + 1;
395                    $objQuery->insert("dtb_recommend_products", $arrval);
396                }
397            }
398        }
399    }
400
401    /**
402     * 商品規格登録を行う.
403     *
404     * @param SC_Query $objQuery SC_Queryインスタンス
405     * @param array $arrList 商品規格情報配列
406     * @param integer $product_id 商品ID
407     * @param integer $product_class_id 商品規格ID
408     * @return void
409     */
410    function lfRegistProductClass($objQuery, $arrList, $product_id, $product_class_id) {
411        $sqlval['product_code'] = $arrList["product_code"];
412        $sqlval['stock'] = $arrList["stock"];
413        if($sqlval['stock'] == "") {
414            $sqlval['stock_unlimited'] = '1';
415        } else {
416            $sqlval['stock_unlimited'] = '0';
417        }
418        $sqlval['price01'] = $arrList['price01'];
419        $sqlval['price02'] = $arrList['price02'];
420        $sqlval['creator_id'] = $_SESSION['member_id'];
421
422        // TODO $sqlval['member_id'] は何処から出てくる?
423        if($sqlval['member_id'] == "") {
424            $sqlval['creator_id'] = '0';
425        }
426
427        if($product_class_id == "") {
428            // 新規登録
429            $where = "product_id = ?";
430            // 念のために既存の規格を削除
431            $objQuery->delete("dtb_products_class", $where, array($product_id));
432            $sqlval['product_id'] = $product_id;
433            $sqlval['classcategory_id1'] = '0';
434            $sqlval['classcategory_id2'] = '0';
435            $sqlval['create_date'] = "now()";
436            $objQuery->insert("dtb_products_class", $sqlval);
437        } else {
438            // 既存編集
439            $where = "product_id = ? AND product_class_id = ?";
440            $objQuery->update("dtb_products_class", $sqlval, $where, array($product_id, $product_class_id));
441        }
442    }
443
444    /**
445     * 入力チェックを行う.
446     *
447     * @return void
448     */
449    function lfCheckError() {
450
451        // 入力データを渡す。
452        $arrRet =  $this->objFormParam->getHashArray();
453        $objErr = new SC_CheckError($arrRet);
454        $objErr->arrErr = $this->objFormParam->checkError(false);
455
456        if(count($objErr->arrErr) == 0) {
457            $objQuery = new SC_Query();
458            // 商品ID、規格IDの存在チェック
459            if($arrRet['product_id'] != "") {
460                $count = $objQuery->count("dtb_products", "product_id = ?", array($arrRet['product_id']));
461                if($count == 0) {
462                    $objErr->arrErr['product_id'] = "※ 指定の商品IDは、登録されていません。";
463                }
464            }
465
466            if($arrRet['product_class_id'] != "") {
467                $count = 0;
468                if($arrRet['product_id'] != "") {
469                    $count = $objQuery->count("dtb_products_class", "product_id = ? AND product_class_id = ?", array($arrRet['product_id'], $arrRet['product_class_id']));
470                }
471                if($count == 0) {
472                    $objErr->arrErr['product_class_id'] = "※ 指定の規格IDは、登録されていません。";
473                }
474            }
475
476            // 存在するカテゴリIDかチェック
477            $arrCategory_id = explode("|", $arrRet['category_id']);
478            foreach ($arrCategory_id as $category_id) {
479                $count = $objQuery->count("dtb_category", "category_id = ?", array($category_id));
480                if($count == 0) {
481                    $objErr->arrErr['product_id'] = "※ 指定のカテゴリIDは、登録されていません。";
482                }
483            }
484        }
485        return $objErr->arrErr;
486    }
487
488    /**
489     * CSVのカウント数を得る.
490     *
491     * @param string $file_name ファイルパス
492     * @return integer CSV のカウント数
493     */
494    function lfCSVRecordCount($file_name) {
495
496        $count = 0;
497        $fp = fopen($file_name, "r");
498        while(!feof($fp)) {
499            $arrCSV = fgetcsv($fp, CSV_LINE_MAX);
500            $count++;
501        }
502
503        return $count-1;
504    }
505
506    /**
507     * 引数の文字列をエラー出力する.
508     *
509     * 引数 $val の内容は, htmlspecialchars() によってサニタイズされる
510     *
511     * @param string $val 出力する文字列
512     * @return void
513     */
514    function printError($val) {
515         echo "<font color=\"red\">"
516             . htmlspecialchars($val, ENT_QUOTES)
517             . "</font></br>\n";
518    }
519}
520?>
Note: See TracBrowser for help on using the repository browser.