[UI 개선] 서브메뉴 및 탭 바 컴포넌트 추가, AppHeader 및 기본 레이아웃 수정, API 호출 로직 개선

This commit is contained in:
2025-09-24 16:25:30 +09:00
parent f83782813d
commit f9dde4eb09
10 changed files with 338 additions and 490 deletions

View File

@@ -6,8 +6,8 @@
<!-- HOME 메뉴 -->
<button
class="menu-btn"
:class="{ active: modelValue === 'home' }"
@click="onMenuClick('home')"
:class="{ active: modelValue === 'HOME' }"
@click="onMenuClick('HOME')"
>
HOME
</button>
@@ -72,7 +72,6 @@
<script setup>
import { ref, computed, onMounted, onBeforeUnmount } from "vue";
import { useRouter } from "vue-router";
import { useUserStore } from "~/stores/user";
import { usePermissionsStore } from "~/stores/permissions";
@@ -80,15 +79,9 @@ import { usePermissionsStore } from "~/stores/permissions";
const modelValue = defineModel({ type: String, required: true });
const showDropdown = ref(false);
const router = useRouter();
const userStore = useUserStore();
const permissionStore = usePermissionsStore();
// 권한 초기화
onMounted(async () => {
await permissionStore.fetchPermissions();
});
// 권한이 있고 메뉴에 표시할 페이지그룹들만 필터링
const availableMenus = computed(() => {
return permissionStore.permissions.resources.pageGroups.filter(pageGroup => {
@@ -117,8 +110,7 @@ function handleClickOutside(event) {
async function logout() {
showDropdown.value = false;
await userStore.logout();
router.push("/login");
userStore.logout();
}
onMounted(() => {

View File

@@ -0,0 +1,66 @@
<template>
<nav
v-if="showSubmenuBar && subMenus.length > 0"
class="w-full bg-gray-100 shadow-sm px-4 py-2"
>
<div class="flex items-center space-x-6">
<span class="text-sm font-medium text-gray-600 mr-4">
{{ activeMenu }}
</span>
<div class="flex space-x-4">
<button
v-for="sub in subMenus"
:key="sub.key"
class="submenu-btn"
@click="onSubMenuClick(sub)"
>
{{ sub.label }}
</button>
</div>
</div>
</nav>
</template>
<script setup lang="ts">
interface SubMenu {
key: string;
label: string;
to: string;
componentName: string;
}
interface Props {
showSubmenuBar: boolean;
activeMenu: string;
subMenus: SubMenu[];
}
interface Emits {
(e: "submenu-click", sub: SubMenu): void;
}
defineProps<Props>();
const emit = defineEmits<Emits>();
function onSubMenuClick(sub: SubMenu) {
emit("submenu-click", sub);
}
</script>
<style scoped>
.submenu-btn {
padding: 0.25rem 0.75rem;
font-size: 0.875rem;
color: #374151;
background: none;
border: none;
border-radius: 0.25rem;
cursor: pointer;
transition: all 0.15s ease;
}
.submenu-btn:hover {
color: #2563eb;
background-color: #eff6ff;
}
</style>

View File

@@ -0,0 +1,135 @@
<template>
<div class="tab-bar">
<div
v-for="tab in tabsStore.tabs"
:key="tab.key"
class="tab-item"
:class="{ active: tabsStore.activeTab === tab.key }"
@click="handleTabClick(tab.key)"
>
<span class="tab-label">{{ tab.label }}</span>
<button
v-if="tabsStore.activeTab !== tab.key"
class="close-btn"
type="button"
@click.stop="handleTabClose(tab.key)"
>
×
</button>
</div>
<button class="add-tab-btn" type="button" @click="handleAddTab"></button>
</div>
</template>
<script setup lang="ts">
import { useTabsStore } from "@/stores/tab";
const tabsStore = useTabsStore();
const handleTabClick = (tabKey: number) => tabsStore.setActiveTab(tabKey);
const handleTabClose = (tabKey: number) => tabsStore.removeTab(tabKey);
const handleAddTab = () => tabsStore.addTab();
</script>
<style scoped>
.tab-bar {
display: flex;
gap: 0.5rem;
padding: 0.75rem 1rem;
background: #ffffff;
border-bottom: 1px solid #e5e7eb;
overflow-x: auto;
scrollbar-width: none;
-ms-overflow-style: none;
}
.tab-bar::-webkit-scrollbar {
display: none;
}
.tab-item {
display: flex;
align-items: center;
gap: 0.5rem;
padding: 0.5rem 0.75rem;
background: #f8fafc;
border: 1px solid #e2e8f0;
border-radius: 0.5rem;
cursor: pointer;
transition: all 0.2s ease;
white-space: nowrap;
min-width: fit-content;
position: relative;
}
.tab-item:hover {
background: #f1f5f9;
border-color: #cbd5e1;
}
.tab-item.active {
background: #3b82f6;
border-color: #3b82f6;
color: #ffffff;
}
.tab-item.active:hover {
background: #2563eb;
border-color: #2563eb;
}
.tab-label {
font-size: 0.875rem;
font-weight: 500;
line-height: 1.25rem;
}
.close-btn {
display: flex;
align-items: center;
justify-content: center;
width: 1.25rem;
height: 1.25rem;
background: transparent;
border: none;
border-radius: 0.25rem;
cursor: pointer;
font-size: 1rem;
line-height: 1;
color: inherit;
transition: background-color 0.2s ease;
flex-shrink: 0;
}
.close-btn:hover {
background: rgba(0, 0, 0, 0.1);
}
.tab-item.active .close-btn:hover {
background: rgba(255, 255, 255, 0.2);
}
.add-tab-btn {
display: flex;
align-items: center;
justify-content: center;
padding: 0.5rem 0.75rem;
background: #f1f5f9;
border: 1px solid #e2e8f0;
border-radius: 0.5rem;
cursor: pointer;
font-size: 1.125rem;
line-height: 1;
color: #64748b;
transition: all 0.2s ease;
flex-shrink: 0;
min-width: 2.5rem;
}
.add-tab-btn:hover {
background: #e2e8f0;
border-color: #cbd5e1;
color: #475569;
}
</style>