tpl_maintitle = '商品管理'; $this->tpl_subtitle = 'カテゴリ登録'; $this->tpl_mainpage = 'products/category.tpl'; $this->tpl_mainno = 'products'; $this->tpl_subno = 'category'; $this->tpl_onload = " fnSetFocus('category_name'); "; } /** * Page のプロセス. * * @return void */ function process() { $this->action(); $this->sendResponse(); } /** * Page のアクション. * * @return void */ function action() { $objDb = new SC_Helper_DB_Ex(); $objFormParam = new SC_FormParam_Ex(); $objCategory = new SC_Helper_Category_Ex(); // 入力パラメーター初期化 $this->initParam($objFormParam); $objFormParam->setParam($_POST); $objFormParam->convParam(); switch ($this->getMode()) { // カテゴリ登録/編集実行 case 'edit': $this->doEdit($objFormParam); break; // 入力ボックスへ編集対象のカテゴリ名をセット case 'pre_edit': $this->doPreEdit($objFormParam); break; // カテゴリ削除 case 'delete': $this->doDelete($objFormParam, $objDb); break; // 表示順を上へ case 'up': $this->doUp($objFormParam); break; // 表示順を下へ case 'down': $this->doDown($objFormParam); break; // FIXME r19909 によってテンプレートが削除されている case 'moveByDnD': // DnDしたカテゴリと移動先のセットを分解する $keys = explode('-', $_POST['keySet']); if ($keys[0] && $keys[1]) { $objQuery =& SC_Query_Ex::getSingletonInstance(); $objQuery->begin(); // 移動したデータのrank、level、parent_category_idを取得 $rank = $objQuery->get('rank', 'dtb_category', 'category_id = ?', array($keys[0])); $level = $objQuery->get('level', 'dtb_category', 'category_id = ?', array($keys[0])); $parent = $objQuery->get('parent_category_id', 'dtb_category', 'category_id = ?', array($keys[0])); // 同一level内のrank配列を作成 $objQuery->setOption('ORDER BY rank DESC'); if ($level == 1) { // 第1階層の時 $arrRet = $objQuery->select('rank', 'dtb_category', 'level = ?', array($level)); } else { // 第2階層以下の時 $arrRet = $objQuery->select('rank', 'dtb_category', 'level = ? AND parent_category_id = ?', array($level, $parent)); } for ($i = 0; $i < sizeof($arrRet); $i++) { $rankAry[$i + 1] = $arrRet[$i]['rank']; } // 移動したデータのグループ内データ数 $my_count = $this->lfCountChilds($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $keys[0]); if ($rankAry[$keys[1]] > $rank) { // データが今の位置より上がった時 $up_count = $rankAry[$keys[1]] - $rank; $decAry = $objQuery->select('category_id', 'dtb_category', 'level = ? AND rank > ? AND rank <= ?', array($level, $rank, $rankAry[$keys[1]])); foreach ($decAry as $value) { // 上のグループから減算 $this->lfDownRankChilds($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $value['category_id'], $my_count); } // 自分のグループに加算 $this->lfUpRankChilds($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $keys[0], $up_count); } else if ($rankAry[$keys[1]] < $rank) { // データが今の位置より下がった時 $down_count = 0; $incAry = $objQuery->select('category_id', 'dtb_category', 'level = ? AND rank < ? AND rank >= ?', array($level, $rank, $rankAry[$keys[1]])); foreach ($incAry as $value) { // 下のグループに加算 $this->lfUpRankChilds($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $value['category_id'], $my_count); // 合計減算値 $down_count += $this->lfCountChilds($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $value['category_id']); } // 自分のグループから減算 $this->lfDownRankChilds($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $keys[0], $down_count); } $objQuery->commit(); } break; // カテゴリツリークリック時 case 'tree': break; // CSVダウンロード case 'csv': // CSVを送信する $objCSV = new SC_Helper_CSV_Ex(); $objCSV->sfDownloadCsv('5', '', array(), '', true); SC_Response_Ex::actionExit(); break; default: break; } $parent_category_id = $objFormParam->getValue('parent_category_id'); // 空の場合は親カテゴリを0にする if (empty($parent_category_id)) { $parent_category_id = 0; } // 親カテゴリIDの保持 $this->arrForm['parent_category_id'] = $parent_category_id; // カテゴリ一覧を取得 $this->arrList = $this->findCategoiesByParentCategoryId($parent_category_id); // カテゴリツリーを取得 // $this->arrTree = $objDb->sfGetCatTree($parent_category_id); $this->arrTree = $objCategory->getTree(); $this->arrParentID = $objCategory->getTreeTrail($parent_category_id); // ぱんくずの生成 $arrBread = array(); $objDb->findTree($this->arrTree, $parent_category_id, $arrBread); $this->tpl_bread_crumbs = SC_Utils_Ex::jsonEncode($arrBread); } /** * カテゴリの削除を実行する. * * 下記の場合は削除を実施せず、エラーメッセージを表示する. * * - 削除対象のカテゴリに、子カテゴリが1つ以上ある場合 * - 削除対象のカテゴリを、登録商品が使用している場合 * * カテゴリの削除は、物理削除で行う. * * @param SC_FormParam $objFormParam * @param SC_Helper_Db $objDb * @return void */ function doDelete(&$objFormParam, &$objDb) { $category_id = $objFormParam->getValue('category_id'); $objQuery =& SC_Query_Ex::getSingletonInstance(); // 子カテゴリのチェック $where = 'parent_category_id = ? AND del_flg = 0'; $exists = $objQuery->exists('dtb_category', $where, array($category_id)); if ($exists) { $this->arrErr['category_name'] = '※ 子カテゴリが存在するため削除できません。
'; return; } // 登録商品のチェック $table = 'dtb_product_categories AS T1 LEFT JOIN dtb_products AS T2 ON T1.product_id = T2.product_id'; $where = 'T1.category_id = ? AND T2.del_flg = 0'; $exists = $objQuery->exists($table, $where, array($category_id)); if ($exists) { $this->arrErr['category_name'] = '※ カテゴリ内に商品が存在するため削除できません。
'; return; } // ランク付きレコードの削除(※処理負荷を考慮してレコードごと削除する。) $objDb->sfDeleteRankRecord('dtb_category', 'category_id', $category_id, '', true); } /** * 編集対象のカテゴリ名を, 入力ボックスへ表示する. * * @param SC_FormParam $objFormParam * @return void */ function doPreEdit(&$objFormParam) { $category_id = $objFormParam->getValue('category_id'); $objCategory = new SC_Helper_Category_Ex(); $arrRes = $objCategory->get($category_id); $objFormParam->setParam($arrRes); $this->arrForm = $objFormParam->getHashArray(); } /** * カテゴリの登録・編集を実行する. * * 下記の場合は, 登録・編集を実行せず、エラーメッセージを表示する * * - カテゴリ名がすでに使用されている場合 * - 階層登録数の上限を超える場合 (登録時のみ評価) * - カテゴリ名がすでに使用されている場合 (登録時のみ評価) * * @param SC_FormParam $objFormParam * @return void */ function doEdit(&$objFormParam) { $category_id = $objFormParam->getValue('category_id'); // 追加か $add = strlen($category_id) === 0; // エラーチェック $this->arrErr = $this->checkError($objFormParam, $add); // エラーがない場合、追加・更新処理 if (empty($this->arrErr)) { $arrCategory = $objFormParam->getDbArray(); // 追加 if ($add) { $this->registerCategory($arrCategory); } // 更新 else { unset($arrCategory['category_id']); $this->updateCategory($category_id, $arrCategory); } } // エラーがある場合、入力値の再表示 else { $this->arrForm = $objFormParam->getHashArray(); } } /** * エラーチェック * * @param SC_FormParam $objFormParam * @param boolean $add 追加か * @return void */ function checkError(&$objFormParam, $add) { $objQuery =& SC_Query_Ex::getSingletonInstance(); // 入力項目チェック $arrErr = $objFormParam->checkError(); if (!empty($arrErr)) { return $arrErr; } $category_id = $objFormParam->getValue('category_id'); $parent_category_id = $objFormParam->getValue('parent_category_id'); $category_name = $objFormParam->getValue('category_name'); // 追加の場合に固有のチェック if ($add) { // 登録数上限チェック $where = 'del_flg = 0'; $count = $objQuery->count('dtb_category', $where); if ($count >= CATEGORY_MAX) { $arrErr['category_name'] = '※ カテゴリの登録最大数を超えました。
'; return $arrErr; } // 階層上限チェック if ($this->isOverLevel($parent_category_id)) { $arrErr['category_name'] = '※ ' . LEVEL_MAX . '階層以上の登録はできません。
'; return $arrErr; } } // 重複チェック $arrWhereVal = array(); $where = 'del_flg = 0 AND parent_category_id = ? AND category_name = ?'; $arrWhereVal[] = $parent_category_id; $arrWhereVal[] = $category_name; // 更新の場合、抽出対象から自己を除外する if (!$add) { $where .= ' AND category_id <> ?'; $arrWhereVal[] = $category_id; } $exists = $objQuery->exists('dtb_category', $where, $arrWhereVal); if ($exists) { $arrErr['category_name'] = '※ 既に同じ内容の登録が存在します。
'; return $arrErr; } return $arrErr; } /** * カテゴリの表示順序を上へ移動する. * * @param SC_FormParam $objFormParam * @return void */ function doUp(&$objFormParam) { $category_id = $objFormParam->getValue('category_id'); $objQuery =& SC_Query_Ex::getSingletonInstance(); $objQuery->begin(); $up_id = $this->lfGetUpRankID($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $category_id); if ($up_id != '') { // 上のグループのrankから減算する数 $my_count = $this->lfCountChilds($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $category_id); // 自分のグループのrankに加算する数 $up_count = $this->lfCountChilds($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $up_id); if ($my_count > 0 && $up_count > 0) { // 自分のグループに加算 $this->lfUpRankChilds($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $category_id, $up_count); // 上のグループから減算 $this->lfDownRankChilds($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $up_id, $my_count); } } $objQuery->commit(); } /** * カテゴリの表示順序を下へ移動する. * * @param SC_FormParam $objFormParam * @return void */ function doDown(&$objFormParam) { $category_id = $objFormParam->getValue('category_id'); $objQuery =& SC_Query_Ex::getSingletonInstance(); $objQuery->begin(); $down_id = $this->lfGetDownRankID($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $category_id); if ($down_id != '') { // 下のグループのrankに加算する数 $my_count = $this->lfCountChilds($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $category_id); // 自分のグループのrankから減算する数 $down_count = $this->lfCountChilds($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $down_id); if ($my_count > 0 && $down_count > 0) { // 自分のグループから減算 $this->lfUpRankChilds($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $down_id, $my_count); // 下のグループに加算 $this->lfDownRankChilds($objQuery, 'dtb_category', 'parent_category_id', 'category_id', $category_id, $down_count); } } $objQuery->commit(); } /** * パラメーターの初期化を行う * * @param SC_FormParam $objFormParam * @return void */ function initParam(&$objFormParam) { $objFormParam->addParam('親カテゴリID', 'parent_category_id', null, null, array()); $objFormParam->addParam('カテゴリID', 'category_id', null, null, array()); $objFormParam->addParam('カテゴリ名', 'category_name', STEXT_LEN, 'KVa', array('EXIST_CHECK', 'SPTAB_CHECK', 'MAX_LENGTH_CHECK')); } /** * 親カテゴリIDでカテゴリを検索する. * * - 表示順の降順でソートする * - 有効なカテゴリを返す(del_flag = 0) * * @param SC_Query $objQuery * @param int $parent_category_id 親カテゴリID * @return array カテゴリの配列 */ function findCategoiesByParentCategoryId($parent_category_id) { if (!$parent_category_id) { $parent_category_id = 0; } $objQuery =& SC_Query_Ex::getSingletonInstance(); $col = 'category_id, category_name, level, rank'; $where = 'del_flg = 0 AND parent_category_id = ?'; $objQuery->setOption('ORDER BY rank DESC'); return $objQuery->select($col, 'dtb_category', $where, array($parent_category_id)); } /** * カテゴリを更新する * * @param SC_FormParam $objFormParam SC_FormParam インスタンス * @return void */ function updateCategory($category_id, $arrCategory) { $objQuery =& SC_Query_Ex::getSingletonInstance(); $arrCategory['update_date'] = 'CURRENT_TIMESTAMP'; $objQuery->begin(); $where = 'category_id = ?'; $objQuery->update('dtb_category', $arrCategory, $where, array($category_id)); $objQuery->commit(); } /** * カテゴリを登録する * * @param SC_FormParam $objFormParam SC_FormParam インスタンス * @return void */ function registerCategory($arrCategory) { $objQuery =& SC_Query_Ex::getSingletonInstance(); $parent_category_id = $arrCategory['parent_category_id']; $objQuery->begin(); $rank = null; if ($parent_category_id == 0) { // ROOT階層で最大のランクを取得する。 $where = 'parent_category_id = ?'; $rank = $objQuery->max('rank', 'dtb_category', $where, array($parent_category_id)) + 1; } else { // 親のランクを自分のランクとする。 $where = 'category_id = ?'; $rank = $objQuery->get('rank', 'dtb_category', $where, array($parent_category_id)); // 追加レコードのランク以上のレコードを一つあげる。 $where = 'rank >= ?'; $arrRawSql = array( 'rank' => '(rank + 1)', ); $objQuery->update('dtb_category', array(), $where, array($rank), $arrRawSql); } $where = 'category_id = ?'; // 自分のレベルを取得する(親のレベル + 1) $level = $objQuery->get('level', 'dtb_category', $where, array($parent_category_id)) + 1; $arrCategory['create_date'] = 'CURRENT_TIMESTAMP'; $arrCategory['update_date'] = 'CURRENT_TIMESTAMP'; $arrCategory['creator_id'] = $_SESSION['member_id']; $arrCategory['rank'] = $rank; $arrCategory['level'] = $level; $arrCategory['category_id'] = $objQuery->nextVal('dtb_category_category_id'); $objQuery->insert('dtb_category', $arrCategory); $objQuery->commit(); // トランザクションの終了 } /** * カテゴリの階層が上限を超えているかを判定する * * @param integer 親カテゴリID * @param 超えている場合 true */ function isOverLevel($parent_category_id) { $objQuery =& SC_Query_Ex::getSingletonInstance(); $level = $objQuery->get('level', 'dtb_category', 'category_id = ?', array($parent_category_id)); return $level >= LEVEL_MAX; } /** * デストラクタ. * * @return void */ function destroy() { parent::destroy(); } // 並びが1つ下のIDを取得する。 function lfGetDownRankID($objQuery, $table, $pid_name, $id_name, $id) { // 親IDを取得する。 $col = "$pid_name"; $where = "$id_name = ?"; $pid = $objQuery->get($col, $table, $where, $id); // すべての子を取得する。 $col = "$id_name"; $where = "del_flg = 0 AND $pid_name = ? ORDER BY rank DESC"; $arrRet = $objQuery->select($col, $table, $where, array($pid)); $max = count($arrRet); $down_id = ''; for ($cnt = 0; $cnt < $max; $cnt++) { if ($arrRet[$cnt][$id_name] == $id) { $down_id = $arrRet[($cnt + 1)][$id_name]; break; } } return $down_id; } // 並びが1つ上のIDを取得する。 function lfGetUpRankID($objQuery, $table, $pid_name, $id_name, $id) { // 親IDを取得する。 $col = "$pid_name"; $where = "$id_name = ?"; $pid = $objQuery->get($col, $table, $where, $id); // すべての子を取得する。 $col = "$id_name"; $where = "del_flg = 0 AND $pid_name = ? ORDER BY rank DESC"; $arrRet = $objQuery->select($col, $table, $where, array($pid)); $max = count($arrRet); $up_id = ''; for ($cnt = 0; $cnt < $max; $cnt++) { if ($arrRet[$cnt][$id_name] == $id) { $up_id = $arrRet[($cnt - 1)][$id_name]; break; } } return $up_id; } function lfCountChilds($objQuery, $table, $pid_name, $id_name, $id) { $objDb = new SC_Helper_DB_Ex(); // 子ID一覧を取得 $arrRet = $objDb->sfGetChildrenArray($table, $pid_name, $id_name, $id); return count($arrRet); } function lfUpRankChilds($objQuery, $table, $pid_name, $id_name, $id, $count) { $objDb = new SC_Helper_DB_Ex(); // 子ID一覧を取得 $arrRet = $objDb->sfGetChildrenArray($table, $pid_name, $id_name, $id); $line = SC_Utils_Ex::sfGetCommaList($arrRet); $where = "$id_name IN ($line) AND del_flg = 0"; $arrRawVal = array( 'rank' => "(rank + $count)", ); return $objQuery->update($table, array(), $where, array(), $arrRawVal); } function lfDownRankChilds($objQuery, $table, $pid_name, $id_name, $id, $count) { $objDb = new SC_Helper_DB_Ex(); // 子ID一覧を取得 $arrRet = $objDb->sfGetChildrenArray($table, $pid_name, $id_name, $id); $line = SC_Utils_Ex::sfGetCommaList($arrRet); $where = "$id_name IN ($line) AND del_flg = 0"; $arrRawVal = array( 'rank' => "(rank - $count)", ); return $objQuery->update($table, array(), $where, array(), $arrRawVal); } }