create-standard-component
skill표준 RNBT 컴포넌트를 생성합니다. 페이지가 GlobalDataPublisher로 데이터를 제어합니다.
apm::install
apm install @jskim-90/create-standard-componentapm::skill.md
---
name: create-standard-component
description: 표준 RNBT 컴포넌트를 생성합니다. 페이지가 GlobalDataPublisher로 데이터를 제어합니다.
---
# 표준 컴포넌트 생성
페이지가 GlobalDataPublisher로 데이터를 제어하고, 컴포넌트는 구독만 합니다.
> 공통 규칙: [SHARED_INSTRUCTIONS.md](/.claude/skills/SHARED_INSTRUCTIONS.md) 참조
---
## ⚠️ 작업 전 필수 확인
**코드 작성 전 반드시 다음 파일들을 Read 도구로 읽으세요.**
**이전에 읽었더라도 매번 다시 읽어야 합니다 - 캐싱하거나 생략하지 마세요.**
1. [/.claude/skills/SHARED_INSTRUCTIONS.md](/.claude/skills/SHARED_INSTRUCTIONS.md) - 공통 규칙
2. [/RNBT_architecture/README.md](/RNBT_architecture/README.md) - 아키텍처 이해
3. [/.claude/guides/CODING_STYLE.md](/.claude/guides/CODING_STYLE.md) - 코딩 스타일
4. **기존 컴포넌트 패턴 확인** - `/RNBT_architecture/Components/LogViewer/` 등 기존 컴포넌트의 구조와 패턴을 먼저 확인
---
## 🚨 실수 방지 체크리스트
### 기존 컴포넌트 패턴을 먼저 확인
```
❌ 임의로 새 구조를 만들지 않는다
✅ LogViewer, AssetTree 등 기존 컴포넌트의 패턴을 먼저 확인한다
```
---
## 핵심 원칙
### 1. 역할 분리
```
페이지 = 오케스트레이터
- 데이터 정의 (globalDataMappings)
- Interval 관리 (refreshIntervals)
- 이벤트 핸들러 등록 (eventBusHandlers)
컴포넌트 = 독립적 구독자
- topic 구독 (subscriptions)
- 이벤트 발행 (@eventName)
- 렌더링만 집중
```
### 2. 메서드 분리 (재사용성)
**핵심: 컴포넌트 재사용을 위해 메서드를 철저히 분리한다.**
```javascript
// 고정 (재사용)
function renderChart(config, { response }) {
const { optionBuilder, ...chartCfg } = config;
const option = optionBuilder(chartCfg, data);
this.chartInstance.setOption(option, true);
}
// 가변 (컴포넌트별)
const chartConfig = { optionBuilder: getChartOption };
this.renderChart = renderChart.bind(this, chartConfig);
```
| 구분 | 역할 | 재사용 |
|------|------|--------|
| renderChart / renderTable | 규격화된 렌더링 함수 | O (고정) |
| chartConfig / tableConfig | 데이터 매핑 + 옵션 | X (컴포넌트별) |
| optionBuilder | ECharts/Tabulator 옵션 생성 | X (컴포넌트별) |
### 3. 응답 구조
```javascript
// response 키가 한 번 더 감싸져 있음
function renderData(config, { response }) {
const { data, meta } = response;
}
```
---
## register.js 구조
### 전체 구조 (LogViewer 패턴 기준)
> **참고:** 단순 컴포넌트(EventStatus 등)는 STATE, 내부 핸들러 섹션을 생략할 수 있습니다.
```javascript
const { subscribe } = GlobalDataPublisher;
const { bindEvents } = Wkit;
// ======================
// CONFIG (정적 선언)
// ======================
const config = {
titleKey: 'TBD_title',
itemsKey: 'TBD_items',
fields: { name: 'TBD_name', value: 'TBD_value' },
maxItems: 100
};
// ======================
// STATE
// ======================
this._someState = null;
this._internalHandlers = {};
// ======================
// BINDINGS (config 바인딩)
// ======================
this.renderData = renderData.bind(this, config);
this.renderList = renderList.bind(this, config);
// ======================
// SUBSCRIPTIONS (구독 등록)
// ======================
this.subscriptions = {
TBD_topicName: ['renderData']
};
fx.go(
Object.entries(this.subscriptions),
fx.each(([topic, fnList]) =>
fx.each(fn => this[fn] && subscribe(topic, this, this[fn]), fnList)
)
);
// ======================
// EVENT BINDING (이벤트 등록)
// ======================
// 페이지에 전달할 이벤트
this.customEvents = {
click: {
'.btn-refresh': '@refreshClicked',
'.row': '@rowClicked'
}
};
bindEvents(this, this.customEvents);
// 컴포넌트 내부 이벤트
setupInternalHandlers.call(this);
// ======================
// RENDER FUNCTIONS (호이스팅)
// ======================
function renderData(config, { response }) { ... }
function renderList(config, { response }) { ... }
function setupInternalHandlers() { ... }
```
### 이벤트 처리 이중 구조
> **상세: [SHARED_INSTRUCTIONS.md](/.claude/skills/SHARED_INSTRUCTIONS.md#이벤트-처리-이중-구조) 참조**
### fx.go 파이프라인 활용
> **기본 패턴: [SHARED_INSTRUCTIONS.md](/.claude/skills/SHARED_INSTRUCTIONS.md#fxgo-파이프라인-패턴) 참조**
DOM 렌더링에 `fx.go` 파이프라인을 사용합니다. DOM 생성은 순수 함수로 분리합니다.
```javascript
fx.go(
items,
fx.map(item => createItemElement(config, item)),
fx.each(el => container.appendChild(el))
);
function createItemElement(config, item) {
const el = document.createElement('div');
el.className = 'list-item';
el.textContent = item[config.fields.name];
return el;
}
```
### 재귀 DOM 빌딩 (트리 구조)
트리형 데이터(AssetTree 패턴)에서는 재귀적으로 DOM을 구축합니다.
```javascript
function renderTree(config, items, searchTerm) {
const rootEl = this.appendElement.querySelector('.tree-root');
rootEl.innerHTML = '';
fx.go(
items,
fx.each(item => {
const nodeEl = createNodeElement.call(this, config, item, searchTerm);
if (nodeEl) rootEl.appendChild(nodeEl);
})
);
}
function createNodeElement(config, item, searchTerm) {
const children = item[config.fields.children] || [];
const hasChildren = children.length > 0;
const li = document.createElement('li');
// ... 노드 생성
if (hasChildren) {
const childrenUl = document.createElement('ul');
fx.go(
children,
fx.each(child => {
const childEl = createNodeElement.call(this, config, child, searchTerm);
if (childEl) childrenUl.appendChild(childEl);
})
);
li.appendChild(childrenUl);
}
return li;
}
```
---
## beforeDestroy 패턴
> **5단계 정리 순서: [SHARED_INSTRUCTIONS.md](/.claude/skills/SHARED_INSTRUCTIONS.md#beforeDestroy-정리-순서) 참조**
생성의 역순: 구독 해제 → 외부 이벤트 제거 → 내부 핸들러 제거 → 상태 초기화 → 바인딩 메서드 null
```javascript
const { unsubscribe } = GlobalDataPublisher;
const { removeCustomEvents } = Wkit;
```
**핵심:** register에서 생성한 **모든 것**은 beforeDestroy에서 정리합니다.
---
## 출력 구조
```
[ComponentName]/
├── views/component.html
├── styles/component.css
├── scripts/
│ ├── register.js
│ └── beforeDestroy.js
├── preview.html
└── README.md
```
---
## 핵심 패턴
### PUB-SUB (GlobalDataPublisher)
```javascript
this.subscriptions = {
topicA: ['renderData'],
topicB: ['renderList', 'updateCount']
};
```
### Event-Driven (Weventbus)
```javascript
this.customEvents = {
click: { '.btn-refresh': '@refreshClicked' },
change: { '.filter-select': '@filterChanged' }
};
```
---
## 금지 사항
> 공통 금지 사항: [SHARED_INSTRUCTIONS.md](/.claude/skills/SHARED_INSTRUCTIONS.md#금지-사항-전체-공통) 참조
- ❌ 컴포넌트가 직접 fetch (팝업 없이)
- ❌ _internalHandlers에 참조 저장 없이 addEventListener 사용
---
## 관련 자료
| 문서 | 위치 |
|------|------|
| LogViewer (기본 패턴) | [/RNBT_architecture/Components/LogViewer/](/RNBT_architecture/Components/LogViewer/) |
| AssetTree (트리/검색 패턴) | [/RNBT_architecture/Components/AssetTree/](/RNBT_architecture/Components/AssetTree/) |
| EventStatus (단순 패턴) | [/RNBT_architecture/Components/EventStatus/](/RNBT_architecture/Components/EventStatus/) |
| 예제 | [/RNBT_architecture/Examples/SimpleDashboard/](/RNBT_architecture/Examples/SimpleDashboard/) |