tpl_maintitle = '商品管理';
$this->tpl_subtitle = 'カテゴリ登録';
$this->tpl_mainpage = 'products/category.tpl';
$this->tpl_mainno = 'products';
$this->tpl_subno = 'category';
$this->tpl_onload = " eccube.setFocus('category_name'); ";
}
/**
* Page のプロセス.
*
* @return void
*/
public function process()
{
$this->action();
$this->sendResponse();
}
/**
* Page のアクション.
*
* @return void
*/
public function action()
{
$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);
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);
} elseif ($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 = $objCategory->getTree();
$this->arrParentID = $objCategory->getTreeTrail($parent_category_id);
// ぱんくずの生成
$arrBread = $objCategory->getTreeTrail($this->arrForm['parent_category_id'], FALSE);
$this->tpl_bread_crumbs = SC_Utils_Ex::jsonEncode(array_reverse($arrBread));
}
/**
* カテゴリの削除を実行する.
*
* 下記の場合は削除を実施せず、エラーメッセージを表示する.
*
* - 削除対象のカテゴリに、子カテゴリが1つ以上ある場合
* - 削除対象のカテゴリを、登録商品が使用している場合
*
* カテゴリの削除は、物理削除で行う.
*
* @param SC_FormParam $objFormParam
* @return void
*/
public function doDelete(&$objFormParam)
{
$objCategory = new SC_Helper_Category_Ex(false);
$category_id = $objFormParam->getValue('category_id');
// 子カテゴリのチェック
$arrBranch = $objCategory->getTreeBranch($category_id);
if (count($arrBranch) > 0) {
$this->arrErr['category_name'] = '※ 子カテゴリが存在するため削除できません。
';
return;
}
// 登録商品のチェック
$arrCategory = $objCategory->get($category_id);
if ($arrCategory['product_count'] > 0) {
$this->arrErr['category_name'] = '※ カテゴリ内に商品が存在するため削除できません。
';
return;
}
// ランク付きレコードの削除(※処理負荷を考慮してレコードごと削除する。)
$objCategory->delete($category_id);
}
/**
* 編集対象のカテゴリ名を, 入力ボックスへ表示する.
*
* @param SC_FormParam $objFormParam
* @return void
*/
public 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
*/
public 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
*/
public 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
*/
public function doUp(&$objFormParam)
{
$objCategory = new SC_Helper_Category_Ex(false);
$category_id = $objFormParam->getValue('category_id');
$objCategory->rankUp($category_id);
}
/**
* カテゴリの表示順序を下へ移動する.
*
* @param SC_FormParam $objFormParam
* @return void
*/
public function doDown(&$objFormParam)
{
$objCategory = new SC_Helper_Category_Ex(false);
$category_id = $objFormParam->getValue('category_id');
$objCategory->rankDown($category_id);
}
/**
* パラメーターの初期化を行う
*
* @param SC_FormParam $objFormParam
* @return void
*/
public 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 カテゴリの配列
*/
public 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
*/
public 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
*/
public 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 int $parent_category_id 親カテゴリID
* @return bool 超えている場合 true
*/
public function isOverLevel($parent_category_id)
{
$objCategory = new SC_Helper_Category_Ex();
$arrCategory = $objCategory->get($parent_category_id);
return $arrCategory['level'] >= LEVEL_MAX;
}
public 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);
}
public 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);
}
public 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);
}
}