Files
invyone/frontend/styles/builder-ide.css
T
gbpark 1aa48cc0bb INVYONE 화면 디자이너 개편 + 컴포넌트 86→8 통합
## 디자인 개편
- IDE 톤 CSS 오버라이드 (builder-ide.css)
- 컴팩트화 (폰트/간격/패딩 축소)
- INVYONE STUDIO 로고 추가 (SlimToolbar)
- 좌측 수평 탭 → 수직 아코디언 (details/summary)
- 우측 속성 패널 신설 (V2PropertiesPanel 완전 이주)
- 다크모드 지원 (7개 통합 컴포넌트 inline hex → CSS 변수)

## 기반 시스템
- ScreenDefinition.fields/connections 타입 확장
- ComponentDefinition.dataPorts 타입 확장
- FieldConfig adapters (fieldsToColumns/Search/Form)
- DataPortBus + setupConnections runtime
- FieldsPanel (화면 수준 필드 관리 패널)

## 컴포넌트 통합 (Phase A~C)
- divider (3→1): 가로/세로 + 텍스트 구분선
- title (2→1): h1~h6/body/caption variant
- button (3→1): 6 variant × 13 actionType
- search (3→1): inline/stacked 검색 필터
- input (20+→1): FieldConfig.type 10종 내부 분기
- stats (6→1): card/chip/bigNumber 3종 스타일
- table (9→1): table/split/grouped/pivot/card 5종 displayMode
- container (11→1): tabs/section/accordion/repeater/conditional 5종

## 버그 수정 (기존 VEX 코드)
- 드래그 드롭 불가 (defaultSize camelCase 불일치)
- 설정 변경 미반영 (componentConfig vs component_config)
- ConfigPanel 미인식 (config_panel vs configPanel)
- v2- 자동 매핑 함정 (INVYONE_UNIFIED_IDS 화이트리스트)
- LayerManagerPanel 무한 API 호출 (useEffect deps)
- Button size 이름 충돌 (visual size 객체 vs config string)

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-12 20:37:23 +09:00

600 lines
20 KiB
CSS

/* ═══════════════════════════════════════════════════════════════════════════
★ Builder IDE Theme — ScreenDesigner 전용 IDE 톤 오버라이드
═══════════════════════════════════════════════════════════════════════════
목적:
/admin/builder 의 ScreenDesigner 를 기존 v5 Cosmic 톤에서 IDE 톤으로 전환.
JSX / 로직은 건드리지 않고 shadcn HSL 변수만 스코프 안에서 덮어쓴다.
스코프:
.ide-builder 가 붙은 루트 아래에서만 효과.
바깥 (헤더 / 사이드바 / 대시보드 등) 은 Cosmic 톤 그대로 유지.
기준 색 (mockup/css/09-developer.css 와 frontend/styles/developer.css):
다크 #121218 / #1a1a22 / #22222c / #3a3a48 / #e8e8ee / #5b9ef5
라이트 #f5f5f8 / #ededf2 / #e4e4ec / #d8d8e2 / #1a1a24 / #3b7dd8
═══════════════════════════════════════════════════════════════════════════ */
/* ─── 라이트 모드 ─── */
.ide-builder {
/* 배경 f5f5f8 */
--background: 240 12% 97%;
/* 카드 (패널) ededf2 */
--card: 240 13% 94%;
--card-foreground: 240 17% 12%;
/* 팝오버 ededf2 */
--popover: 240 13% 94%;
--popover-foreground: 240 17% 12%;
/* 전경 1a1a24 */
--foreground: 240 17% 12%;
/* muted e4e4ec / 5a5a6e */
--muted: 240 12% 91%;
--muted-foreground: 240 10% 39%;
/* accent (hover 배경) e4e4ec */
--accent: 240 12% 91%;
--accent-foreground: 240 17% 12%;
/* secondary ededf2 */
--secondary: 240 13% 94%;
--secondary-foreground: 240 17% 12%;
/* 보더 d8d8e2 */
--border: 240 13% 87%;
--input: 240 13% 87%;
/* 액센트 (primary) 3b7dd8 — IDE 블루 */
--primary: 214 66% 54%;
--primary-foreground: 0 0% 100%;
--ring: 214 66% 54%;
/* destructive dc2626 */
--destructive: 0 73% 51%;
--destructive-foreground: 0 0% 100%;
/* 사이드바 (패널 안의 보조 영역) */
--sidebar-background: 240 13% 94%;
--sidebar-foreground: 240 17% 12%;
--sidebar-primary: 214 66% 54%;
--sidebar-primary-foreground: 0 0% 100%;
--sidebar-accent: 240 12% 91%;
--sidebar-accent-foreground: 240 17% 12%;
--sidebar-border: 240 13% 87%;
--sidebar-ring: 214 66% 54%;
/* 반경을 살짝 더 타이트하게 — IDE 느낌 */
--radius: 0.375rem;
}
/* ─── 다크 모드 ─── */
.dark .ide-builder {
/* 배경 121218 */
--background: 240 13% 9%;
/* 카드 (패널) 1a1a22 */
--card: 240 14% 12%;
--card-foreground: 240 14% 92%;
/* 팝오버 1a1a22 */
--popover: 240 14% 12%;
--popover-foreground: 240 14% 92%;
/* 전경 e8e8ee */
--foreground: 240 14% 92%;
/* muted 22222c / 78788a */
--muted: 240 13% 15%;
--muted-foreground: 240 8% 51%;
/* accent 2a2a36 */
--accent: 240 13% 19%;
--accent-foreground: 240 14% 92%;
/* secondary 1a1a22 */
--secondary: 240 14% 12%;
--secondary-foreground: 240 14% 92%;
/* 보더 3a3a48 */
--border: 240 10% 25%;
--input: 240 10% 25%;
/* 액센트 5b9ef5 — IDE 라이트 블루 */
--primary: 214 89% 66%;
--primary-foreground: 0 0% 100%;
--ring: 214 89% 66%;
/* destructive f87171 */
--destructive: 0 91% 71%;
--destructive-foreground: 0 0% 100%;
/* 사이드바 (패널 안의 보조 영역) */
--sidebar-background: 240 14% 12%;
--sidebar-foreground: 240 14% 92%;
--sidebar-primary: 214 89% 66%;
--sidebar-primary-foreground: 0 0% 100%;
--sidebar-accent: 240 13% 19%;
--sidebar-accent-foreground: 240 14% 92%;
--sidebar-border: 240 10% 25%;
--sidebar-ring: 214 89% 66%;
}
/* ═══════════════════════════════════════════════════════════════════════════
캔버스 배경 — 도트 그리드 (IDE 특유의 느낌)
기존 ScreenDesigner 의 캔버스 영역에 은은한 도트가 찍힘.
ScreenDesigner 가 캔버스 컨테이너에 특정 클래스를 박고 있지 않으니
.ide-builder 하위의 최상위 bg-background 영역을 폴백으로 커버한다.
실제 캔버스 ref 는 overflow-auto 이므로 내부 inner wrapper 에도 먹게 함.
═══════════════════════════════════════════════════════════════════════════ */
/* 라이트 기본 — 은은한 검정 도트 */
.ide-builder .bg-background {
background-image: radial-gradient(
circle,
rgba(0, 0, 0, 0.05) 0.5px,
transparent 0.5px
);
background-size: 20px 20px;
}
/* 다크 덮어쓰기 — 은은한 흰색 도트 */
.dark .ide-builder .bg-background {
background-image: radial-gradient(
circle,
rgba(255, 255, 255, 0.03) 0.5px,
transparent 0.5px
);
background-size: 20px 20px;
}
/* ═══════════════════════════════════════════════════════════════════════════
★ 컴팩트화 — IDE 특유의 촘촘한 레이아웃
═══════════════════════════════════════════════════════════════════════════
원칙: JSX 는 건드리지 않는다. Tailwind arbitrary 클래스까지 CSS selector
specificity 로 덮어쓴다 (.ide-builder .w-\[300px\] 등).
폰트 기준: 기본 13px → 0.72rem (11.5px) 촘촘. 완전한 mockup 0.42rem 은 읽기
어려워서 웹 최소 가독선인 11~12px 에 맞춤.
═══════════════════════════════════════════════════════════════════════════ */
/* ─── 기본 폰트 살짝 축소 ─── */
.ide-builder {
font-size: 0.8125rem; /* 13px */
line-height: 1.3;
}
/* ─── 패널 너비 축소 (shadcn w-[300px] → 240px) ─── */
.ide-builder .w-\[300px\] {
width: 240px;
}
.ide-builder .w-\[280px\] {
width: 220px;
}
.ide-builder .w-\[320px\] {
width: 250px;
}
/* ─── Tailwind 텍스트 크기 다운그레이드 ─── */
.ide-builder .text-xs {
font-size: 0.625rem;
line-height: 0.9rem;
}
.ide-builder .text-sm {
font-size: 0.72rem;
line-height: 1rem;
}
.ide-builder .text-base {
font-size: 0.8125rem;
line-height: 1.15rem;
}
.ide-builder .text-lg {
font-size: 0.88rem;
line-height: 1.25rem;
}
.ide-builder .text-xl {
font-size: 1rem;
line-height: 1.35rem;
}
/* ─── 헤딩 ─── */
.ide-builder h1 {
font-size: 0.95rem;
}
.ide-builder h2 {
font-size: 0.85rem;
}
.ide-builder h3 {
font-size: 0.78rem;
}
.ide-builder h4,
.ide-builder h5,
.ide-builder h6 {
font-size: 0.72rem;
}
/* ─── 입력/선택 크기 ─── */
.ide-builder input,
.ide-builder select,
.ide-builder textarea {
font-size: 0.72rem;
}
.ide-builder input[type="text"],
.ide-builder input[type="number"],
.ide-builder input[type="search"],
.ide-builder select {
min-height: 26px;
height: 26px;
}
.ide-builder textarea {
min-height: 48px;
}
/* ─── 버튼 ─── */
.ide-builder button {
font-size: 0.72rem;
}
.ide-builder button[data-slot="button"] {
min-height: 26px;
}
/* ─── shadcn Tabs 크기 축소 ─── */
.ide-builder [role="tablist"],
.ide-builder .h-8 {
height: 28px;
min-height: 28px;
}
.ide-builder [role="tab"] {
font-size: 0.68rem;
padding: 0.22rem 0.5rem;
}
/* ─── 패딩 / 간격 다운그레이드 ─── */
.ide-builder .p-4 {
padding: 0.6rem;
}
.ide-builder .p-3 {
padding: 0.45rem;
}
.ide-builder .p-2 {
padding: 0.3rem;
}
.ide-builder .px-4 {
padding-left: 0.7rem;
padding-right: 0.7rem;
}
.ide-builder .px-3 {
padding-left: 0.55rem;
padding-right: 0.55rem;
}
.ide-builder .px-2 {
padding-left: 0.35rem;
padding-right: 0.35rem;
}
.ide-builder .py-3 {
padding-top: 0.4rem;
padding-bottom: 0.4rem;
}
.ide-builder .py-2 {
padding-top: 0.3rem;
padding-bottom: 0.3rem;
}
.ide-builder .py-1 {
padding-top: 0.15rem;
padding-bottom: 0.15rem;
}
.ide-builder .gap-4 {
gap: 0.5rem;
}
.ide-builder .gap-3 {
gap: 0.4rem;
}
.ide-builder .gap-2 {
gap: 0.3rem;
}
.ide-builder .space-y-4 > :not([hidden]) ~ :not([hidden]) {
margin-top: 0.5rem;
}
.ide-builder .space-y-3 > :not([hidden]) ~ :not([hidden]) {
margin-top: 0.4rem;
}
.ide-builder .space-y-2 > :not([hidden]) ~ :not([hidden]) {
margin-top: 0.3rem;
}
/* ─── 아이콘 크기 (lucide) ─── */
.ide-builder svg {
flex-shrink: 0;
}
.ide-builder .size-4 {
width: 0.85rem;
height: 0.85rem;
}
.ide-builder .size-5 {
width: 1rem;
height: 1rem;
}
.ide-builder .size-6 {
width: 1.1rem;
height: 1.1rem;
}
.ide-builder .h-4,
.ide-builder .w-4 {
height: 0.85rem;
width: 0.85rem;
}
/* ─── 라벨 (속성 패널의 Label 컴포넌트) ─── */
.ide-builder label {
font-size: 0.62rem;
font-weight: 600;
letter-spacing: 0.01em;
}
/* ─── 보더 반경 타이트 ─── */
.ide-builder .rounded-lg {
border-radius: 0.45rem;
}
.ide-builder .rounded-md {
border-radius: 0.35rem;
}
.ide-builder .rounded-sm {
border-radius: 0.25rem;
}
/* ─── 테이블 row 높이 축소 ─── */
.ide-builder table {
font-size: 0.68rem;
}
.ide-builder th,
.ide-builder td {
padding: 0.25rem 0.4rem;
}
/* ─── 카드 (Card) 패딩 축소 ─── */
.ide-builder [data-slot="card"] {
padding: 0;
}
.ide-builder [data-slot="card-header"] {
padding: 0.5rem 0.6rem;
}
.ide-builder [data-slot="card-content"] {
padding: 0.5rem 0.6rem;
}
.ide-builder [data-slot="card-footer"] {
padding: 0.4rem 0.6rem;
}
/* ─── Dialog / Popover 컴팩트 ─── */
.ide-builder [data-slot="popover-content"],
.ide-builder [role="dialog"] {
font-size: 0.72rem;
}
/* ─── Badge 컴팩트 ─── */
.ide-builder [data-slot="badge"] {
font-size: 0.56rem;
padding: 0.08rem 0.3rem;
border-radius: 0.25rem;
}
/* ─── Separator 얇게 ─── */
.ide-builder [data-slot="separator"],
.ide-builder hr {
border-color: hsl(var(--border));
}
/* ─── 포커스 링 은은하게 (IDE 느낌) ─── */
.ide-builder *:focus-visible {
outline: 1.5px solid hsl(var(--ring));
outline-offset: 1px;
box-shadow: none !important;
}
/* ═══════════════════════════════════════════════════════════════════════════
★ 좌측 패널 섹션 (수평 탭 해체 → 수직 아코디언)
═══════════════════════════════════════════════════════════════════════════
기존 <Tabs> 의 수평 탭을 없애고 <details>/<summary> 네이티브 아코디언으로
대체. 한 패널 안에서 컴포넌트/필드/레이어/편집 4개 섹션이 세로로 나열되며
각자 독립적으로 접고 펼 수 있다.
복잡성 축소: "탭 선택 후 내용 봄" (2단계) → "섹션 펼쳐서 내용 봄" (1단계).
═══════════════════════════════════════════════════════════════════════════ */
.ide-builder details.ide-section {
border-bottom: 1px solid hsl(var(--border));
flex-shrink: 0;
min-height: 0;
}
.ide-builder details.ide-section[open] {
display: flex;
flex-direction: column;
min-height: 0;
flex: 1 1 0;
}
.ide-builder details.ide-section > summary.ide-section-header {
list-style: none;
cursor: pointer;
padding: 0.5rem 0.65rem;
font-size: 0.66rem;
font-weight: 700;
text-transform: uppercase;
letter-spacing: 0.04em;
color: hsl(var(--primary));
background: hsl(var(--muted) / 0.5);
border-bottom: 1px solid hsl(var(--border));
display: flex;
align-items: center;
gap: 0.4rem;
user-select: none;
transition: background 0.1s;
}
.ide-builder details.ide-section > summary.ide-section-header:hover {
background: hsl(var(--muted));
}
.ide-builder details.ide-section > summary.ide-section-header::-webkit-details-marker {
display: none;
}
.ide-builder details.ide-section > summary.ide-section-header::before {
content: "▶";
font-size: 0.5rem;
color: hsl(var(--muted-foreground));
transition: transform 0.15s;
display: inline-block;
}
.ide-builder details.ide-section[open] > summary.ide-section-header::before {
transform: rotate(90deg);
}
.ide-builder details.ide-section > .ide-section-body {
flex: 1 1 0;
min-height: 0;
max-height: 70vh;
overflow-y: auto;
}
/* ═══════════════════════════════════════════════════════════════════════════
★ v5 Cosmic 토큰 → IDE 블루 오버라이드 (잔존 보라 제거)
═══════════════════════════════════════════════════════════════════════════
v5-layout.css 가 정의한 --v5-primary (#6c5ce7 / #a29bfe) 같은 토큰이 여러
.v5-* 클래스에서 직접 쓰이고 있어서 INVYONE 빌더 안에도 보라가 잔존한다.
.ide-builder 스코프 안에서 이 토큰들을 IDE 블루로 전부 덮어씀.
═══════════════════════════════════════════════════════════════════════════ */
.ide-builder {
/* 라이트 v5 토큰 */
--v5-primary: #3b7dd8;
--v5-primary-light: #5b9ef5;
--v5-primary-glow: rgba(59, 125, 216, 0.18);
--v5-cyan: #0891b2;
--v5-cyan-glow: rgba(8, 145, 178, 0.15);
--v5-pink: #db2777;
--v5-pink-glow: rgba(219, 39, 119, 0.12);
--v5-glass: hsl(240 13% 94% / 0.6);
--v5-glass-strong: hsl(240 13% 94% / 0.8);
--v5-glass-border: rgba(59, 125, 216, 0.15);
--v5-glow-sm: 0 0 0 1px rgba(59, 125, 216, 0.1);
--v5-glow-md: 0 0 12px rgba(59, 125, 216, 0.15);
--v5-glow-lg: 0 0 24px rgba(59, 125, 216, 0.2);
--v5-surface: hsl(240 12% 97%);
}
.dark .ide-builder {
/* 다크 v5 토큰 */
--v5-primary: #5b9ef5;
--v5-primary-light: #93bffc;
--v5-primary-glow: rgba(91, 158, 245, 0.2);
--v5-cyan: #22d3ee;
--v5-cyan-glow: rgba(34, 211, 238, 0.15);
--v5-pink: #ec4899;
--v5-pink-glow: rgba(236, 72, 153, 0.12);
--v5-glass: hsl(240 14% 12% / 0.6);
--v5-glass-strong: hsl(240 14% 12% / 0.8);
--v5-glass-border: rgba(91, 158, 245, 0.15);
--v5-glow-sm: 0 0 0 1px rgba(91, 158, 245, 0.12);
--v5-glow-md: 0 0 12px rgba(91, 158, 245, 0.18);
--v5-glow-lg: 0 0 24px rgba(91, 158, 245, 0.22);
--v5-surface: hsl(240 13% 9%);
}
/* ═══════════════════════════════════════════════════════════════════════════
★ Tailwind primary 유틸리티 강제 덮기 (!important)
═══════════════════════════════════════════════════════════════════════════
Tailwind v4 의 --color-primary 가 런타임 --primary 업데이트를 따라가지
못하는 경우가 있어서 .bg-primary / .text-primary / .border-primary 등을
강제로 덮는다. specificity 는 .ide-builder 스코프로 제한.
═══════════════════════════════════════════════════════════════════════════ */
.ide-builder .bg-primary {
background-color: hsl(var(--primary)) !important;
}
.ide-builder .text-primary {
color: hsl(var(--primary)) !important;
}
.ide-builder .border-primary {
border-color: hsl(var(--primary)) !important;
}
.ide-builder .bg-primary\/5 {
background-color: hsl(var(--primary) / 0.05) !important;
}
.ide-builder .bg-primary\/10 {
background-color: hsl(var(--primary) / 0.1) !important;
}
.ide-builder .bg-primary\/20 {
background-color: hsl(var(--primary) / 0.2) !important;
}
.ide-builder .bg-primary\/90 {
background-color: hsl(var(--primary) / 0.9) !important;
}
.ide-builder .hover\:bg-primary\/90:hover {
background-color: hsl(var(--primary) / 0.9) !important;
}
.ide-builder .hover\:bg-primary\/80:hover {
background-color: hsl(var(--primary) / 0.8) !important;
}
.ide-builder .hover\:bg-primary\/20:hover {
background-color: hsl(var(--primary) / 0.2) !important;
}
.ide-builder .ring-primary {
--tw-ring-color: hsl(var(--primary)) !important;
}
/* shadcn Button default variant 강제 블루 */
.ide-builder [data-slot="button"][data-variant="default"],
.ide-builder button.bg-primary,
.ide-builder [class*="bg-primary"]:not([class*="/"]) {
background-color: hsl(var(--primary)) !important;
color: hsl(var(--primary-foreground)) !important;
}
/* ═══════════════════════════════════════════════════════════════════════════
★ SlimToolbar 슬림화 (h-14 → h-10)
═══════════════════════════════════════════════════════════════════════════ */
.ide-builder .h-14 {
height: 42px !important;
min-height: 42px;
}
.ide-builder .h-14.border-b {
padding-top: 0;
padding-bottom: 0;
}
.ide-builder .h-9 {
height: 28px !important;
min-height: 28px;
}
.ide-builder .text-lg {
font-size: 0.82rem !important;
line-height: 1.15rem;
font-weight: 700;
}
/* ═══════════════════════════════════════════════════════════════════════════
★ INVYONE 브랜드 로고 (SlimToolbar 좌측)
═══════════════════════════════════════════════════════════════════════════ */
.ide-builder .ide-brand {
display: flex;
align-items: center;
gap: 0.4rem;
padding-right: 0.7rem;
margin-right: 0.3rem;
border-right: 1px solid hsl(var(--border));
flex-shrink: 0;
}
.ide-builder .ide-brand-text {
font-size: 0.78rem;
font-weight: 800;
letter-spacing: -0.03em;
color: hsl(var(--primary));
font-family:
-apple-system, BlinkMacSystemFont, "Segoe UI", system-ui, sans-serif;
}
.ide-builder .ide-brand-badge {
font-size: 0.5rem;
font-weight: 800;
padding: 0.1rem 0.32rem;
background: hsl(var(--primary));
color: hsl(var(--primary-foreground));
border-radius: 3px;
letter-spacing: 0.04em;
text-transform: uppercase;
line-height: 1.1;
}
/* ─── 스크롤바 IDE 스타일 ─── */
.ide-builder *::-webkit-scrollbar {
width: 10px;
height: 10px;
}
.ide-builder *::-webkit-scrollbar-track {
background: transparent;
}
.ide-builder *::-webkit-scrollbar-thumb {
background: hsl(var(--border));
border-radius: 5px;
border: 2px solid transparent;
background-clip: padding-box;
}
.ide-builder *::-webkit-scrollbar-thumb:hover {
background: hsl(var(--muted-foreground) / 0.5);
background-clip: padding-box;
border: 2px solid transparent;
}