@things-factory/board-ui 10.0.0-beta.92 → 10.0.0-beta.94

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -37,9 +37,4 @@ export declare function suggestBoardMeta(input: {
37
37
  name: string;
38
38
  description: string;
39
39
  }>;
40
- /** @deprecated suggestBoardMeta 사용. name 만 필요한 구버전 호출처 호환용. */
41
- export declare function suggestBoardName(input: {
42
- sessionId?: string;
43
- hint?: string;
44
- }): Promise<string>;
45
40
  export declare function materializeImportSession(input: MaterializeImportSessionInput): Promise<any>;
@@ -70,19 +70,6 @@ export async function suggestBoardMeta(input) {
70
70
  });
71
71
  return response.data?.suggestBoardMeta ?? { name: '', description: '' };
72
72
  }
73
- /** @deprecated suggestBoardMeta 사용. name 만 필요한 구버전 호출처 호환용. */
74
- export async function suggestBoardName(input) {
75
- const response = await client.query({
76
- query: gql `
77
- query SuggestBoardName($input: SuggestBoardNameInput!) {
78
- suggestBoardName(input: $input)
79
- }
80
- `,
81
- variables: { input },
82
- fetchPolicy: 'no-cache'
83
- });
84
- return response.data?.suggestBoardName ?? '';
85
- }
86
73
  export async function materializeImportSession(input) {
87
74
  const response = await client.mutate({
88
75
  mutation: gql `
@@ -1 +1 @@
1
- {"version":3,"file":"board-import.js","sourceRoot":"","sources":["../../client/graphql/board-import.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AACH,OAAO,GAAG,MAAM,aAAa,CAAA;AAE7B,OAAO,EAAE,MAAM,EAAE,MAAM,kBAAkB,CAAA;AAEzC,MAAM,qBAAqB,GAAG;;;;;;;;;;;;;CAa7B,CAAA;AAkBD,MAAM,CAAC,KAAK,UAAU,gBAAgB,CAAC,KAA4B;IACjE,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC;QACnC,QAAQ,EAAE,GAAG,CAAA;;;YAGL,qBAAqB;;;KAG5B;QACD,SAAS,EAAE,EAAE,KAAK,EAAE;KACrB,CAAC,CAAA;IACF,OAAO,QAAQ,CAAC,IAAI,EAAE,gBAAgB,CAAA;AACxC,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,kBAAkB,CAAC,EAAU;IACjD,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,KAAK,CAAC;QAClC,KAAK,EAAE,GAAG,CAAA;;;YAGF,qBAAqB;;;KAG5B;QACD,SAAS,EAAE,EAAE,EAAE,EAAE;QACjB,4CAA4C;QAC5C,WAAW,EAAE,UAAU;KACxB,CAAC,CAAA;IACF,OAAO,QAAQ,CAAC,IAAI,EAAE,aAAa,CAAA;AACrC,CAAC;AAaD;;;;GAIG;AACH,MAAM,CAAC,KAAK,UAAU,gBAAgB,CAAC,KAGtC;IACC,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,KAAK,CAAC;QAClC,KAAK,EAAE,GAAG,CAAA;;;;;;;KAOT;QACD,SAAS,EAAE,EAAE,KAAK,EAAE;QACpB,WAAW,EAAE,UAAU;KACxB,CAAC,CAAA;IACF,OAAO,QAAQ,CAAC,IAAI,EAAE,gBAAgB,IAAI,EAAE,IAAI,EAAE,EAAE,EAAE,WAAW,EAAE,EAAE,EAAE,CAAA;AACzE,CAAC;AAED,+DAA+D;AAC/D,MAAM,CAAC,KAAK,UAAU,gBAAgB,CAAC,KAGtC;IACC,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,KAAK,CAAC;QAClC,KAAK,EAAE,GAAG,CAAA;;;;KAIT;QACD,SAAS,EAAE,EAAE,KAAK,EAAE;QACpB,WAAW,EAAE,UAAU;KACxB,CAAC,CAAA;IACF,OAAO,QAAQ,CAAC,IAAI,EAAE,gBAAgB,IAAI,EAAE,CAAA;AAC9C,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,wBAAwB,CAAC,KAAoC;IACjF,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC;QACnC,QAAQ,EAAE,GAAG,CAAA;;;;;;;;;;;KAWZ;QACD,SAAS,EAAE,EAAE,KAAK,EAAE;KACrB,CAAC,CAAA;IACF,OAAO,QAAQ,CAAC,IAAI,EAAE,wBAAwB,CAAA;AAChD,CAAC","sourcesContent":["/**\n * board-import GraphQL operation client — wizard page 가 사용.\n *\n * 백엔드 정의: @things-factory/board-import 의 ImportSessionResolver.\n * - importBoardAsync(input: ImportBoardAsyncInput!): ImportSession\n * - importSession(id: String!): ImportSession\n * - materializeImportSession(input: MaterializeImportSessionInput!): Board\n */\nimport gql from 'graphql-tag'\n\nimport { client } from '@operato/graphql'\n\nconst IMPORT_SESSION_FIELDS = `\n id\n status\n progress\n message\n totalEntities\n options\n result\n attachmentId\n chatSessionId\n createdAt\n updatedAt\n completedAt\n`\n\nexport interface ImportBoardAsyncInput {\n attachmentId: string\n chatSessionId?: string\n scopes?: string[]\n normalizeOrigin?: boolean\n flipY?: boolean\n scale?: number\n applyBinding?: boolean\n parseOptions?: any\n /**\n * 사용자 자유서술 hint — 도면 설명 / 강조 부분 / 카테고리 가이드 등.\n * VLM 의 view type 분류 + entity 추출에 추가 컨텍스트로 합류.\n */\n userPrompt?: string\n}\n\nexport async function importBoardAsync(input: ImportBoardAsyncInput) {\n const response = await client.mutate({\n mutation: gql`\n mutation ImportBoardAsync($input: ImportBoardAsyncInput!) {\n importBoardAsync(input: $input) {\n ${IMPORT_SESSION_FIELDS}\n }\n }\n `,\n variables: { input }\n })\n return response.data?.importBoardAsync\n}\n\nexport async function fetchImportSession(id: string) {\n const response = await client.query({\n query: gql`\n query ImportSession($id: String!) {\n importSession(id: $id) {\n ${IMPORT_SESSION_FIELDS}\n }\n }\n `,\n variables: { id },\n // polling 시 cache 우회 — 매 호출마다 server fresh.\n fetchPolicy: 'no-cache'\n })\n return response.data?.importSession\n}\n\nexport interface MaterializeImportSessionInput {\n sessionId: string\n name: string\n description?: string\n groupId?: string\n type?: 'main' | 'sub' | 'popup'\n thumbnail?: string\n /** 선택한 시안 id ('as-is' / 'scene' / 'auto-fit' 등). 미지정 시 default. */\n variantId?: string\n}\n\n/**\n * 충돌하지 않는 새 Board 의 짧은 이름 + 자세한 description 제안.\n * 서버가 hint / userPrompt / AI importStrategy / 카테고리 분포 등을 합성해\n * 둘 다 만들어준다. 클라이언트는 review 진입 시 한 번 호출.\n */\nexport async function suggestBoardMeta(input: {\n sessionId?: string\n hint?: string\n}): Promise<{ name: string; description: string }> {\n const response = await client.query({\n query: gql`\n query SuggestBoardMeta($input: SuggestBoardNameInput!) {\n suggestBoardMeta(input: $input) {\n name\n description\n }\n }\n `,\n variables: { input },\n fetchPolicy: 'no-cache'\n })\n return response.data?.suggestBoardMeta ?? { name: '', description: '' }\n}\n\n/** @deprecated suggestBoardMeta 사용. name 만 필요한 구버전 호출처 호환용. */\nexport async function suggestBoardName(input: {\n sessionId?: string\n hint?: string\n}): Promise<string> {\n const response = await client.query({\n query: gql`\n query SuggestBoardName($input: SuggestBoardNameInput!) {\n suggestBoardName(input: $input)\n }\n `,\n variables: { input },\n fetchPolicy: 'no-cache'\n })\n return response.data?.suggestBoardName ?? ''\n}\n\nexport async function materializeImportSession(input: MaterializeImportSessionInput) {\n const response = await client.mutate({\n mutation: gql`\n mutation MaterializeImportSession($input: MaterializeImportSessionInput!) {\n materializeImportSession(input: $input) {\n id\n name\n description\n state\n sourceImportSessionId\n createdAt\n }\n }\n `,\n variables: { input }\n })\n return response.data?.materializeImportSession\n}\n"]}
1
+ {"version":3,"file":"board-import.js","sourceRoot":"","sources":["../../client/graphql/board-import.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AACH,OAAO,GAAG,MAAM,aAAa,CAAA;AAE7B,OAAO,EAAE,MAAM,EAAE,MAAM,kBAAkB,CAAA;AAEzC,MAAM,qBAAqB,GAAG;;;;;;;;;;;;;CAa7B,CAAA;AAkBD,MAAM,CAAC,KAAK,UAAU,gBAAgB,CAAC,KAA4B;IACjE,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC;QACnC,QAAQ,EAAE,GAAG,CAAA;;;YAGL,qBAAqB;;;KAG5B;QACD,SAAS,EAAE,EAAE,KAAK,EAAE;KACrB,CAAC,CAAA;IACF,OAAO,QAAQ,CAAC,IAAI,EAAE,gBAAgB,CAAA;AACxC,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,kBAAkB,CAAC,EAAU;IACjD,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,KAAK,CAAC;QAClC,KAAK,EAAE,GAAG,CAAA;;;YAGF,qBAAqB;;;KAG5B;QACD,SAAS,EAAE,EAAE,EAAE,EAAE;QACjB,4CAA4C;QAC5C,WAAW,EAAE,UAAU;KACxB,CAAC,CAAA;IACF,OAAO,QAAQ,CAAC,IAAI,EAAE,aAAa,CAAA;AACrC,CAAC;AAaD;;;;GAIG;AACH,MAAM,CAAC,KAAK,UAAU,gBAAgB,CAAC,KAGtC;IACC,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,KAAK,CAAC;QAClC,KAAK,EAAE,GAAG,CAAA;;;;;;;KAOT;QACD,SAAS,EAAE,EAAE,KAAK,EAAE;QACpB,WAAW,EAAE,UAAU;KACxB,CAAC,CAAA;IACF,OAAO,QAAQ,CAAC,IAAI,EAAE,gBAAgB,IAAI,EAAE,IAAI,EAAE,EAAE,EAAE,WAAW,EAAE,EAAE,EAAE,CAAA;AACzE,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,wBAAwB,CAAC,KAAoC;IACjF,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC;QACnC,QAAQ,EAAE,GAAG,CAAA;;;;;;;;;;;KAWZ;QACD,SAAS,EAAE,EAAE,KAAK,EAAE;KACrB,CAAC,CAAA;IACF,OAAO,QAAQ,CAAC,IAAI,EAAE,wBAAwB,CAAA;AAChD,CAAC","sourcesContent":["/**\n * board-import GraphQL operation client — wizard page 가 사용.\n *\n * 백엔드 정의: @things-factory/board-import 의 ImportSessionResolver.\n * - importBoardAsync(input: ImportBoardAsyncInput!): ImportSession\n * - importSession(id: String!): ImportSession\n * - materializeImportSession(input: MaterializeImportSessionInput!): Board\n */\nimport gql from 'graphql-tag'\n\nimport { client } from '@operato/graphql'\n\nconst IMPORT_SESSION_FIELDS = `\n id\n status\n progress\n message\n totalEntities\n options\n result\n attachmentId\n chatSessionId\n createdAt\n updatedAt\n completedAt\n`\n\nexport interface ImportBoardAsyncInput {\n attachmentId: string\n chatSessionId?: string\n scopes?: string[]\n normalizeOrigin?: boolean\n flipY?: boolean\n scale?: number\n applyBinding?: boolean\n parseOptions?: any\n /**\n * 사용자 자유서술 hint — 도면 설명 / 강조 부분 / 카테고리 가이드 등.\n * VLM 의 view type 분류 + entity 추출에 추가 컨텍스트로 합류.\n */\n userPrompt?: string\n}\n\nexport async function importBoardAsync(input: ImportBoardAsyncInput) {\n const response = await client.mutate({\n mutation: gql`\n mutation ImportBoardAsync($input: ImportBoardAsyncInput!) {\n importBoardAsync(input: $input) {\n ${IMPORT_SESSION_FIELDS}\n }\n }\n `,\n variables: { input }\n })\n return response.data?.importBoardAsync\n}\n\nexport async function fetchImportSession(id: string) {\n const response = await client.query({\n query: gql`\n query ImportSession($id: String!) {\n importSession(id: $id) {\n ${IMPORT_SESSION_FIELDS}\n }\n }\n `,\n variables: { id },\n // polling 시 cache 우회 — 매 호출마다 server fresh.\n fetchPolicy: 'no-cache'\n })\n return response.data?.importSession\n}\n\nexport interface MaterializeImportSessionInput {\n sessionId: string\n name: string\n description?: string\n groupId?: string\n type?: 'main' | 'sub' | 'popup'\n thumbnail?: string\n /** 선택한 시안 id ('as-is' / 'scene' / 'auto-fit' 등). 미지정 시 default. */\n variantId?: string\n}\n\n/**\n * 충돌하지 않는 새 Board 의 짧은 이름 + 자세한 description 제안.\n * 서버가 hint / userPrompt / AI importStrategy / 카테고리 분포 등을 합성해\n * 둘 다 만들어준다. 클라이언트는 review 진입 시 한 번 호출.\n */\nexport async function suggestBoardMeta(input: {\n sessionId?: string\n hint?: string\n}): Promise<{ name: string; description: string }> {\n const response = await client.query({\n query: gql`\n query SuggestBoardMeta($input: SuggestBoardNameInput!) {\n suggestBoardMeta(input: $input) {\n name\n description\n }\n }\n `,\n variables: { input },\n fetchPolicy: 'no-cache'\n })\n return response.data?.suggestBoardMeta ?? { name: '', description: '' }\n}\n\nexport async function materializeImportSession(input: MaterializeImportSessionInput) {\n const response = await client.mutate({\n mutation: gql`\n mutation MaterializeImportSession($input: MaterializeImportSessionInput!) {\n materializeImportSession(input: $input) {\n id\n name\n description\n state\n sourceImportSessionId\n createdAt\n }\n }\n `,\n variables: { input }\n })\n return response.data?.materializeImportSession\n}\n"]}
@@ -21,7 +21,6 @@ import { notify, notifyError } from '../utils/notify-helper.js';
21
21
  import { findSceneComponent, dispatchBoardAction } from './board-action-dispatch.js';
22
22
  import { dispatchBoardEditOp } from './board-edit-dispatch.js';
23
23
  export { findSceneComponent };
24
- const NOOP = () => { };
25
24
  /** 서버 PatchEntry.reverted=true 플래그 영속용 mutation. */
26
25
  const REVERT_PATCH_MUTATION = gql `
27
26
  mutation RevertPatch($patchId: String!) {
@@ -1 +1 @@
1
- {"version":3,"file":"board-modeller-page.js","sourceRoot":"","sources":["../../client/pages/board-modeller-page.ts"],"names":[],"mappings":"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","sourcesContent":["import './things-scene-components.import'\nimport '@operato/board/ox-board-modeller.js'\nimport '@operato/oops'\nimport '@things-factory/board-ai'\n\nimport gql from 'graphql-tag'\nimport { css, html, nothing } from 'lit'\nimport { customElement, property, query, state } from 'lit/decorators.js'\nimport { BoardModeller } from '@operato/board/ox-board-modeller.js'\nimport { LoadTracker, SceneSearchEngine } from '@hatiolab/things-scene'\nimport { OxPropertyEditor } from '@operato/property-editor'\nimport { PageView, FontController } from '@operato/shell'\nimport { hasPrivilege } from '@things-factory/auth-base/dist-client/index.js'\nimport { applyBoardEditPatchVerbose } from '@things-factory/board-ai'\nimport type { BoardEditOp, BoardEditPatch } from '@things-factory/board-ai'\nimport { client, gqlContext } from '@operato/graphql'\nimport { i18next } from '@operato/i18n'\nimport { OxPrompt } from '@operato/popup/ox-prompt.js'\n\nimport { provider } from '../board-provider.js'\nimport components from './things-scene-components-with-tools.import'\nimport { notify, notifyError } from '../utils/notify-helper.js'\nimport { findSceneComponent, dispatchBoardAction } from './board-action-dispatch.js'\nimport { dispatchBoardEditOp } from './board-edit-dispatch.js'\n\nexport { findSceneComponent }\n\nconst NOOP = () => {}\n\n/** 서버 PatchEntry.reverted=true 플래그 영속용 mutation. */\nconst REVERT_PATCH_MUTATION = gql`\n mutation RevertPatch($patchId: String!) {\n revertPatch(patchId: $patchId)\n }\n`\n\n/**\n * default 위에 override 를 깊게 merge.\n * - override 의 키 값이 undefined 가 아닌 경우 우선 적용\n * - override 의 nested object 는 default 와 deep merge\n * - array / primitive 는 override 가 그대로 우선\n */\nfunction mergeDefaultsDeep(defaults: any, override: any): any {\n if (defaults === null || defaults === undefined) return override\n if (override === null || override === undefined) return defaults\n if (typeof defaults !== 'object' || typeof override !== 'object') return override\n if (Array.isArray(defaults) || Array.isArray(override)) return override\n\n const out: any = { ...defaults }\n for (const key of Object.keys(override)) {\n const dv = (defaults as any)[key]\n const ov = (override as any)[key]\n if (\n dv !== null &&\n ov !== null &&\n typeof dv === 'object' &&\n typeof ov === 'object' &&\n !Array.isArray(dv) &&\n !Array.isArray(ov)\n ) {\n out[key] = mergeDefaultsDeep(dv, ov)\n } else if (ov !== undefined) {\n out[key] = ov\n }\n }\n return out\n}\n\n@customElement('board-modeller-page')\nexport class BoardModellerPage extends PageView {\n constructor() {\n super()\n\n components.forEach(({ templates = [], groups = [] }) => {\n groups.forEach(group => BoardModeller.registerGroup(group))\n BoardModeller.registerTemplate(templates)\n })\n\n /* 컴포넌트에서 정의된 에디터들을 MODELLER_EDITORS에 등록 */\n const addedEditors = {}\n for (const component in components) {\n let { editors } = components[component]\n\n editors &&\n editors.forEach(editor => {\n let { type, element } = editor\n\n addedEditors[type] = element\n })\n }\n\n OxPropertyEditor.register(addedEditors)\n }\n\n static styles = [\n css`\n :host {\n display: flex;\n flex-direction: row;\n\n overflow: hidden;\n position: relative;\n }\n\n .modeller-area {\n flex: 1;\n min-width: 0;\n display: flex;\n flex-direction: column;\n position: relative;\n }\n\n ox-board-modeller {\n flex: 1;\n }\n\n .ai-toggle {\n position: absolute;\n right: 12px;\n bottom: 12px;\n z-index: 10;\n padding: 8px 14px;\n border: 0;\n border-radius: 24px;\n background: #2563eb;\n color: #fff;\n font: 500 13px/1.4 system-ui, -apple-system, sans-serif;\n cursor: pointer;\n box-shadow: 0 4px 12px rgba(37, 99, 235, 0.35);\n transition: background 0.15s, transform 0.1s;\n }\n .ai-toggle:hover { background: #1d4ed8; }\n .ai-toggle:active { transform: scale(0.97); }\n .ai-toggle.open { background: #475569; }\n .ai-toggle.open:hover { background: #334155; }\n\n .ai-panel {\n flex: 0 0 320px;\n display: flex;\n border-left: 1px solid #e2e8f0;\n background: #ffffff;\n }\n\n ox-board-ai-chat { flex: 1; }\n\n ox-oops-note {\n display: block;\n position: absolute;\n left: 50%;\n top: 50%;\n transform: translate(-50%, -50%);\n }\n\n @media (max-width: 900px) {\n .ai-panel {\n flex-basis: 100%;\n position: absolute;\n inset: 0;\n z-index: 20;\n }\n }\n `\n ]\n\n @property({ type: String }) boardId?: string | null\n @property({ type: String }) boardName?: string = ''\n @property({ type: Object }) model: any = null\n @property({ type: Array }) selected: any = []\n @property({ type: Number }) mode?: number = 1\n @property({ type: Boolean }) hideProperty?: boolean = false\n @property({ type: String }) overlay?: string | null = null\n @property({ type: Object }) scene: any = null\n @property({ type: Array }) componentGroupList?: any[] = BoardModeller.groups\n @property({ type: Array }) propertyEditor: any\n @property({ type: Boolean }) preparing?: boolean\n\n /** AI 협력 세션 id — 현재 활성 세션. boardId 변경 시 자동 ensure. 다중 세션 시 활성 1개. */\n @state() chatSessionId?: string\n\n /** 보드의 모든 ChatSession (탭으로 표시). */\n @state() chatSessions: Array<{ id: string; name?: string; createdAt?: string }> = []\n\n /** AI 채팅 패널 열림 상태 (기본 닫힘) */\n @state() aiPanelOpen = false\n\n /**\n * 적용된 patch 별 inverse op 시퀀스 — Revert 기능용.\n *\n * Map<patchId, inverseOps[]>: 각 patch 적용 시점에 계산해 저장. revert 클릭\n * 시 역순 in-place 적용. 페이지 reload 시 손실 — reload 후 revert 시도는\n * notify 로 안내 (메모리 외 영속은 별개 작업).\n */\n private patchInverses = new Map<string, BoardEditOp[]>()\n\n /** 모델러에 등록된 컴포넌트 type 목록 — chat 호출 시 LLM 에 전달 (lazy cache) */\n private _knownTypesCache?: string[]\n /** 등록된 type 의 group 분류 — categories 로 LLM 에 전달 */\n private _knownCategoriesCache?: string[]\n\n private get knownTypes(): string[] {\n if (!this._knownTypesCache) {\n const types = new Set<string>()\n for (const group of BoardModeller.groups || []) {\n for (const t of group.templates || []) {\n if (t && typeof t.type === 'string') types.add(t.type)\n }\n }\n this._knownTypesCache = Array.from(types).sort()\n }\n return this._knownTypesCache\n }\n\n private get knownCategories(): string[] {\n if (!this._knownCategoriesCache) {\n const cats = new Set<string>()\n for (const group of BoardModeller.groups || []) {\n for (const t of group.templates || []) {\n if (t && typeof t.group === 'string') cats.add(t.group)\n }\n }\n this._knownCategoriesCache = Array.from(cats).sort()\n }\n return this._knownCategoriesCache\n }\n\n /**\n * type 별 유효 속성 스킴 — LLM 이 정확한 컴포넌트를 만들도록 (lazy cache).\n *\n * 핵심: 좌표 표현은 type 마다 다르다 (left/top vs cx/cy vs points 등).\n * 따라서 OMIT 하지 않고 `geometryKeys` 로 별도 노출 → LLM 이 type 별 좌표 키 사용 가능.\n */\n private _componentSchemasCache?: Array<{\n type: string\n description?: string\n group?: string\n /** template.model 의 좌표/크기 관련 키들 — type 별 다름 */\n geometryKeys?: string[]\n /** 좌표 외 type 특화 속성 (default 값 또는 nested 키 목록) */\n properties?: Record<string, any>\n }>\n\n private get componentSchemas() {\n if (!this._componentSchemasCache) {\n const out: Array<{\n type: string\n description?: string\n group?: string\n geometryKeys?: string[]\n properties?: Record<string, any>\n }> = []\n const NON_PROPS = new Set(['type', 'id', 'name', 'class'])\n // 좌표/크기/경로 관련 키들 — type 마다 다른 조합으로 사용\n const GEOMETRY_KEYS = new Set([\n 'left',\n 'top',\n 'right',\n 'bottom',\n 'x',\n 'y',\n 'cx',\n 'cy',\n 'width',\n 'height',\n 'rx',\n 'ry',\n 'radius',\n 'rotation',\n 'translate',\n 'scale',\n 'points',\n 'path',\n 'vertices',\n 'verticles',\n 'startX',\n 'startY',\n 'endX',\n 'endY',\n 'startPoint',\n 'endPoint'\n ])\n for (const group of BoardModeller.groups || []) {\n for (const t of group.templates || []) {\n if (!t || typeof t.type !== 'string') continue\n const properties: Record<string, any> = {}\n const geometryKeys: string[] = []\n const model: any = t.model || {}\n for (const key of Object.keys(model)) {\n if (NON_PROPS.has(key)) continue\n if (GEOMETRY_KEYS.has(key)) {\n geometryKeys.push(key)\n continue\n }\n const v = model[key]\n if (v && typeof v === 'object' && !Array.isArray(v)) {\n properties[key] = Object.fromEntries(\n Object.keys(v)\n .slice(0, 8)\n .map(k => [k, null])\n )\n } else if (Array.isArray(v)) {\n properties[key] = []\n } else {\n properties[key] = v\n }\n }\n out.push({\n type: t.type,\n description: t.description || undefined,\n group: t.group || undefined,\n geometryKeys: geometryKeys.length > 0 ? geometryKeys : undefined,\n properties: Object.keys(properties).length > 0 ? properties : undefined\n })\n }\n }\n this._componentSchemasCache = out\n }\n return this._componentSchemasCache\n }\n\n private board: any = null\n private _loadTracker?: LoadTracker\n private _fontCtrl = new FontController(this)\n @query('ox-board-modeller') modeller?: BoardModeller\n\n get context() {\n return {\n title: this.board ? this.boardName : this.preparing ? 'Fetching board...' : '',\n help: 'board-modeller/modeller',\n widebleed: true\n }\n }\n\n get oopsNote() {\n return {\n icon: 'color_lens',\n title: 'EMPTY BOARD',\n description: 'There are no board to be designed'\n }\n }\n\n async refresh() {\n if (!this.boardId) {\n this.board = null\n this.model = null\n\n return\n }\n try {\n // 이전 보드 완전히 클리어\n this.board = null\n this.model = null\n this._loadTracker = new LoadTracker()\n this._loadTracker.setPhase('fetch')\n this.preparing = true\n this.updateContext()\n\n const response = await client.query({\n query: gql`\n query FetchBoardById($id: String!) {\n board(id: $id) {\n id\n name\n model\n }\n }\n `,\n variables: { id: this.boardId },\n context: gqlContext()\n })\n\n this._loadTracker?.setPhase('parse')\n\n const board = response.data.board\n\n if (!board) {\n this.board = null\n throw new Error('board not found')\n }\n\n this.board = {\n ...board,\n model: JSON.parse(board.model)\n }\n\n this.boardName = this.board.name\n this.model = {\n ...this.board.model\n }\n\n // AI 협력 세션 — 보드 전환 시 이전 보드의 세션 / 세션 리스트 정리 (chat panel 의\n // sessionId/sessions 변화 감지 → history 비우고 새로 로드). 패널 열린 채 보드 옮겼다면\n // 새 보드의 세션 리스트를 즉시 ensure (fire-and-forget — 에러는 ensureChatSession 처리).\n this.chatSessionId = undefined\n this.chatSessions = []\n if (this.aiPanelOpen && this.boardId) {\n this.ensureChatSession()\n }\n } catch (ex) {\n notifyError(ex)\n } finally {\n this.preparing = false\n this.updateContext()\n }\n }\n\n /** 패널 토글 시 호출 — 열 때만 chatSession 보장 (세션 리스트 로드 + 활성 세션 결정) */\n private async toggleAIPanel() {\n const willOpen = !this.aiPanelOpen\n if (willOpen && this.boardId) {\n await this.ensureChatSession()\n }\n this.aiPanelOpen = willOpen\n }\n\n /** 보드의 모든 ChatSession 목록을 로드 + 활성 세션 보장.\n * - 세션이 0 개면 startBoardAISession 으로 첫 세션 생성 (단일 세션 UX 호환)\n * - 활성 세션 미설정 시 첫 번째 세션을 활성으로\n * - 이미 활성 세션 set 돼있고 list 에 포함되면 유지 */\n private async ensureChatSession() {\n if (!this.boardId) return\n try {\n const result = await client.query({\n query: gql`\n query ChatSessionsByBoard($boardId: String!) {\n chatSessionsByBoard(boardId: $boardId) {\n id\n name\n createdAt\n }\n }\n `,\n variables: { boardId: this.boardId },\n context: gqlContext(),\n fetchPolicy: 'network-only'\n })\n let sessions: Array<{ id: string; name?: string; createdAt?: string }> =\n result.data?.chatSessionsByBoard ?? []\n\n if (sessions.length === 0) {\n // 첫 세션 ensure (idempotent)\n const created = await client.mutate({\n mutation: gql`\n mutation StartBoardAISession($boardId: String!) {\n startBoardAISession(boardId: $boardId) {\n id\n name\n createdAt\n }\n }\n `,\n variables: { boardId: this.boardId },\n context: gqlContext()\n })\n const s = created.data?.startBoardAISession\n if (s) sessions = [s]\n }\n\n this.chatSessions = sessions\n // 활성 세션 결정 — 기존 active 가 list 에 있으면 유지, 아니면 첫 번째\n if (!this.chatSessionId || !sessions.some(s => s.id === this.chatSessionId)) {\n this.chatSessionId = sessions[0]?.id\n }\n } catch (ex) {\n notifyError(ex)\n }\n }\n\n /** `@` mention popup 의 도메인 사용자 후보 — chat 의 userProvider prop 으로 wire.\n * query 변할 때마다 호출되므로 server query (`domainUsersForMention`) 를 그대로 위임. */\n private fetchDomainUsersForMention = async (\n query: string\n ): Promise<Array<{ id: string; name?: string; email?: string }>> => {\n try {\n const result = await client.query({\n query: gql`\n query DomainUsersForMention($query: String, $limit: Int) {\n domainUsersForMention(query: $query, limit: $limit) {\n id\n name\n email\n }\n }\n `,\n variables: { query: query || null, limit: 50 },\n context: gqlContext(),\n fetchPolicy: 'cache-first'\n })\n return result.data?.domainUsersForMention ?? []\n } catch (ex) {\n // 조용히 — chat panel 이 빈 결과로 처리\n console.warn('[board-modeller-page] fetchDomainUsersForMention failed:', ex)\n return []\n }\n }\n\n /** 채팅 패널의 탭 클릭 — 활성 세션 전환. */\n private onChatSessionSwitch = (e: CustomEvent) => {\n const newId = e.detail?.sessionId\n if (typeof newId === 'string' && newId !== this.chatSessionId) {\n this.chatSessionId = newId\n }\n }\n\n /** 채팅 패널의 \"+\" 버튼 — 새 세션 생성 후 활성으로 전환. */\n private onChatSessionCreate = async () => {\n if (!this.boardId) return\n try {\n const created = await client.mutate({\n mutation: gql`\n mutation CreateChatSession($boardId: String!, $name: String) {\n createChatSession(boardId: $boardId, name: $name) {\n id\n name\n createdAt\n }\n }\n `,\n variables: { boardId: this.boardId, name: null },\n context: gqlContext()\n })\n const s = created.data?.createChatSession\n if (s) {\n this.chatSessions = [...this.chatSessions, s]\n this.chatSessionId = s.id\n }\n } catch (ex) {\n notifyError(ex)\n }\n }\n\n /**\n * 모델러의 selected (things-scene Component 인스턴스 배열) 에서 refid 추출.\n *\n * refid 는 things-scene 이 모든 컴포넌트에 자동 발급하는 universal numeric handle.\n * AI 의 selection 표현은 항상 refid 기반 (id 는 데이터 바인딩 이름이며 unique 가\n * 아니라 targeting 에 부적합).\n */\n private extractSelectedRefids(): number[] {\n const sel = this.selected || []\n return sel\n .map((c: any) => {\n if (!c || typeof c.get !== 'function') return null\n const refid = c.get('refid')\n return typeof refid === 'number' && Number.isFinite(refid) ? refid : null\n })\n .filter((x: number | null): x is number => x !== null)\n }\n\n /**\n * AI 채팅 mention popup (`#`) 검색 위임 — things-scene SceneSearchEngine 으로 wrap.\n *\n * F-key Finder 와 동일한 검색 룰 (id → name → tag → type → class → text → data → property)\n * 을 그대로 mention popup 에 재사용. 컴포넌트 트리를 직접 순회하므로 model JSON 으로\n * 변환할 필요 없음 — 일관성 + 성능 이점.\n *\n * 반환 구조 (FilterResult) 는 chat 의 mention-popup 표시 모델. host 는 things-scene\n * Component 에서 refid/id/type/tag/class/text 를 뽑아 candidate 로 채워 넘긴다.\n */\n private searchSceneForMention = (() => {\n const engine = new SceneSearchEngine()\n return (query: string): any[] => {\n const root = (this.scene as any)?.root\n if (!root) return []\n const results = engine.search(root, query)\n return results.map(r => {\n const c: any = r.component\n const refid = typeof c.get === 'function' ? c.get('refid') : undefined\n const id = typeof c.get === 'function' ? c.get('id') : undefined\n const type = typeof c.get === 'function' ? c.get('type') : undefined\n const tag = typeof c.get === 'function' ? c.get('tag') : undefined\n const cls = typeof c.get === 'function' ? c.get('class') : undefined\n const text = typeof c.get === 'function' ? c.get('text') : undefined\n return {\n candidate: {\n refid: typeof refid === 'number' ? refid : undefined,\n label: id || type || 'unknown',\n type: typeof type === 'string' ? type : 'unknown',\n id: typeof id === 'string' && id ? id : undefined,\n tag: typeof tag === 'string' && tag ? tag : undefined,\n class: typeof cls === 'string' && cls ? cls : undefined,\n text: typeof text === 'string' && text ? text : undefined\n },\n matchType: r.matchType,\n matchValue: r.matchValue\n }\n })\n }\n })()\n\n /**\n * 라이브 보드 모델 — things-scene 캔버스가 들고 있는 가장 최신 상태의 deep clone.\n *\n * 사용자 수작업 편집은 scene.model 에만 반영되고 this.model property 에는 자동\n * 동기화되지 않는다. AI 한테 보드 상태를 넘기거나 patch 를 적용할 때는 반드시\n * 이쪽을 쓴다.\n *\n * id 와 refid 는 별개 필드 — 컴포넌트는 model.id (선택, 사용자/AI 가 명시 설정)\n * 와 model.refid (필수, things-scene 자동 발급) 를 둘 다 가질 수 있고 AI 가\n * 양쪽을 구별해서 다룬다 (tool 인자가 분리됨). 따라서 여기서는 model 을 mutate\n * 하지 않고 raw deep clone 만 반환.\n */\n private getLiveBoard(): any {\n const sceneModel = (this.scene as any)?.model\n if (sceneModel) {\n return JSON.parse(JSON.stringify(sceneModel))\n }\n return this.model\n }\n\n /**\n * 채팅에서 patch 도착 — things-scene 에 in-place 적용.\n *\n * 핵심: `this.model = next` 로 모델을 통째 교체하면 ox-scene-viewer 가 scene 을\n * dispose → 재생성 → undo 히스토리/dirty 플래그가 모두 초기화된다.\n * 따라서 일반 add/modify/remove 는 scene 의 in-place mutation API 로 적용 →\n * commander 가 자동으로 snapshot push → CMD+Z 와 저장 안내 팝업이 정상 동작.\n *\n * 'replace' op (보드 전체 교체) 만은 부득이 wholesale 경로로 빠진다 — 이 경우는\n * 사용자에게 명시적으로 안내 후 진행.\n */\n private onBoardEditPatch = (e: CustomEvent) => {\n const { patch, patchId } = e.detail || {}\n if (!patch) return\n const scene = this.scene as any\n if (!scene) {\n notifyError(new Error('Scene not initialized'))\n return\n }\n\n const hasReplace = (patch.ops || []).some((op: BoardEditOp) => op.op === 'replace')\n if (hasReplace) {\n this.applyPatchWholesale(patch, patchId)\n return\n }\n\n try {\n const missed: BoardEditOp[] = []\n // 각 op 의 inverse — dispatcher 가 적용 직전/직후 scene 상태로 계산해 반환.\n // 누적했다가 onBoardEditRevert 가 역순 in-place 적용으로 복원.\n const inverseOps: BoardEditOp[] = []\n const normalize = (c: any) => this.normalizeComponent(c)\n\n for (const op of patch.ops as BoardEditOp[]) {\n const r = dispatchBoardEditOp(scene, op, { normalize })\n if (r.applied) {\n inverseOps.push(...r.inverseOps)\n } else {\n missed.push(op)\n }\n }\n\n // patch 단위로 inverse 보관 — Revert 시 역순 적용\n if (patchId && inverseOps.length > 0) {\n this.patchInverses.set(patchId, inverseOps)\n }\n\n if (missed.length > 0) {\n const missedDesc = missed\n .map((op: any) => {\n if (op.op === 'modify' || op.op === 'remove') return `refid=${op.refid}`\n if (op.op === 'modifyBoard') return 'modifyBoard (scene root unavailable)'\n return op.op\n })\n .join(', ')\n notify(\n 'warn',\n `AI 가 ${missed.length}개의 변경을 시도했지만 적용되지 않았습니다: ${missedDesc}`\n )\n }\n } catch (ex) {\n notifyError(ex)\n }\n }\n\n /**\n * 'replace' op 적용 — scene 통째로 재생성 (undo 히스토리 초기화).\n *\n * AI 가 보드 전체 재구성을 요청한 경우. 사용자에게 history 손실 안내.\n * board-import 데이터 wholesale 적용 등 드문 케이스에 한정.\n *\n * Revert 지원 — replace 직전 보드 model 을 inverse 로 보관 (또 다른 replace).\n */\n private applyPatchWholesale(patch: BoardEditPatch, patchId?: string) {\n try {\n const baseBoard = this.getLiveBoard()\n const inverseOps: BoardEditOp[] = baseBoard\n ? [{ op: 'replace', board: JSON.parse(JSON.stringify(baseBoard)) }]\n : []\n const report = applyBoardEditPatchVerbose(baseBoard, patch)\n const next = report.board as any\n if (report.applied.length === 0) return\n const normalized = (next.components || []).map((c: any) => this.normalizeComponent(c))\n this.model = {\n ...(baseBoard || {}),\n ...(next.width !== undefined && { width: next.width }),\n ...(next.height !== undefined && { height: next.height }),\n components: normalized\n }\n if (patchId && inverseOps.length > 0) {\n this.patchInverses.set(patchId, inverseOps)\n }\n notify('info', '보드 전체가 교체되어 실행 취소(undo) 히스토리가 초기화되었습니다.')\n } catch (ex) {\n notifyError(ex)\n }\n }\n\n /**\n * 채팅에서 board-edit-revert 도착 — 누적된 inverse 를 역순 in-place 적용.\n *\n * - 메모리상 patchInverses Map 만 본다. 페이지 reload 후엔 inverse 가 없어\n * 되돌릴 수 없음 (사용자에게 안내).\n * - 성공 시 서버 revertPatch mutation 호출 → PatchEntry.reverted=true 영속.\n * - in-place 적용이라 undo/dirty/selection 보존 (chatViaTools 의 모든 강점 그대로).\n */\n private onBoardEditRevert = async (e: CustomEvent) => {\n const { patchId } = e.detail || {}\n if (!patchId) return\n const inverses = this.patchInverses.get(patchId)\n if (!inverses || inverses.length === 0) {\n notify(\n 'warn',\n '이 변경은 자동 되돌릴 수 없습니다 (페이지 새로고침 이후 적용된 변경은 메모리에 inverse 가 없습니다).'\n )\n return\n }\n\n const scene = this.scene as any\n if (!scene) {\n notifyError(new Error('Scene not initialized'))\n return\n }\n\n try {\n // 역순 적용 — 마지막 op 부터 되돌림. 단일 patch 안에서도 순서 의미 있음\n // (예: add 후 modify 한 시퀀스를 되돌리려면 modify(원본) 후 remove)\n const reversed = [...inverses].reverse()\n const reverseFakePatch: BoardEditPatch = {\n ops: reversed,\n summary: 'revert',\n confidence: 1\n }\n // hasReplace 면 wholesale, 그렇지 않으면 in-place — 일반 onBoardEditPatch 와 동일\n const hasReplace = reversed.some(op => op.op === 'replace')\n if (hasReplace) {\n this.applyPatchWholesale(reverseFakePatch)\n } else {\n // in-place 직접 적용 — 새 inverse 누적은 안 함 (revert 의 revert 는 redo, 별개 기능)\n this.applyInverseOpsInPlace(scene, reversed)\n }\n\n this.patchInverses.delete(patchId)\n await this.recordRevertOnServer(patchId)\n } catch (ex) {\n notifyError(ex)\n }\n }\n\n /**\n * inverse op 시퀀스를 things-scene 위에 in-place 적용 (snapshot/undo 보존).\n * dispatcher 재사용 — inverse 누적은 무시 (revert 의 revert 는 redo, 별개 기능).\n */\n private applyInverseOpsInPlace(scene: any, ops: BoardEditOp[]): void {\n const normalize = (c: any) => this.normalizeComponent(c)\n for (const op of ops) {\n dispatchBoardEditOp(scene, op, { normalize })\n }\n }\n\n /** 서버 PatchEntry.reverted=true 영속 — fail 해도 UI 는 이미 되돌렸으니 warn 만. */\n private async recordRevertOnServer(patchId: string): Promise<void> {\n try {\n await client.mutate({ mutation: REVERT_PATCH_MUTATION, variables: { patchId } })\n } catch (e: any) {\n console.warn('[board-modeller] revertPatch server persist failed:', e?.message ?? e)\n }\n }\n\n /**\n * AI 가 ephemeral scene action (selection / view / mode) 을 실행하라고 요청.\n *\n * board-edit-patch 와 별개 채널 — 모델 변경 X, undo 영향 X, dirty flag 무관.\n * things-scene API 직접 호출. 잘못된 action / 누락된 컴포넌트는 console.warn 만.\n */\n private onBoardActionExecute = (e: CustomEvent) => {\n const { actions } = e.detail || {}\n if (!Array.isArray(actions) || actions.length === 0) return\n const scene = this.scene as any\n if (!scene) return\n\n let modeChanged = false\n for (const action of actions) {\n try {\n dispatchBoardAction(scene, action)\n if (action?.action === 'setSceneMode') modeChanged = true\n } catch (ex: any) {\n console.warn('[board-modeller] action execute failed:', action, ex?.message ?? ex)\n }\n }\n // setSceneMode 가 있었으면 page 의 reactive mode 도 동기 (Lit re-render 트리거)\n if (modeChanged) this.mode = scene.mode\n }\n\n /**\n * AI 가 만든 컴포넌트를 things-scene 이 기대하는 완전한 형태로 normalize.\n * BoardModeller 에 등록된 template.model (default 값) 위에 AI 응답을 덮어쓴다.\n * 이 단계 없이 things-scene 에 넣으면 필수 속성 누락 시 render 에서 crash 가능\n * (예: gauge-circle 의 value 가 undefined → toString() TypeError).\n */\n private normalizeComponent(c: any): any {\n if (!c || typeof c !== 'object' || typeof c.type !== 'string') return c\n const template = this.findTemplateByType(c.type)\n if (!template || !template.model) return c\n return mergeDefaultsDeep(template.model, c)\n }\n\n private findTemplateByType(type: string): any {\n for (const group of BoardModeller.groups || []) {\n for (const t of group.templates || []) {\n if (t && t.type === type) return t\n }\n }\n return undefined\n }\n\n async updated(changes) {\n if (changes.has('boardId')) {\n try {\n if (await hasPrivilege({ privilege: 'mutation', category: 'board' })) {\n this.refresh()\n } else {\n this.boardId = null\n this.model = null\n this.modeller?.close()\n }\n } catch {\n this.boardId = null\n this.model = null\n this.modeller?.close()\n }\n }\n }\n\n pageUpdated(changes, lifecycle) {\n if (this.active) {\n this.boardId = lifecycle.resourceId\n } else {\n this.boardId = null\n this.board = null\n this.model = null\n this.preparing = false\n this.modeller?.close()\n this.updateContext()\n }\n }\n\n render() {\n const oops = !this.preparing && !this.model && this.oopsNote\n\n return html`\n <div class=\"modeller-area\">\n ${oops\n ? html`<ox-oops-note icon=${oops.icon} title=${oops.title} description=${oops.description}></ox-oops-note>`\n : html`\n <ox-board-modeller\n .mode=${this.mode}\n .loadTracker=${this._loadTracker}\n @mode-changed=${e => {\n this.mode = e.detail.value\n }}\n .model=${this.model}\n @model-changed=${e => {\n this.model = e.detail.value\n }}\n .scene=${this.scene}\n @scene-changed=${e => {\n this.scene = e.detail.value\n }}\n .selected=${this.selected}\n @selected-changed=${e => {\n this.selected = e.detail.value\n }}\n .provider=${provider}\n @save-model=${e => this.saveBoard()}\n .componentGroupList=${this.componentGroupList}\n .fonts=${this._fontCtrl.fonts}\n .hideProperty=${this.hideProperty}\n >\n </ox-board-modeller>\n `}\n ${this.boardId\n ? html`\n <button\n class=\"ai-toggle ${this.aiPanelOpen ? 'open' : ''}\"\n title=\"${i18next.t('text.ai_assistant') || 'AI Assistant'}\"\n @click=${this.toggleAIPanel}>\n ${this.aiPanelOpen ? '✕ AI' : '✨ AI'}\n </button>\n `\n : nothing}\n </div>\n ${this.aiPanelOpen\n ? html`\n <div class=\"ai-panel\">\n <ox-board-ai-chat\n .sessionId=${this.chatSessionId}\n .sessions=${this.chatSessions}\n .currentBoard=${this.model}\n .boardProvider=${() => this.getLiveBoard()}\n .searchProvider=${this.searchSceneForMention}\n .userProvider=${this.fetchDomainUsersForMention}\n .knownTypes=${this.knownTypes}\n .categories=${this.knownCategories}\n .componentSchemas=${this.componentSchemas}\n .selectedRefids=${this.extractSelectedRefids()}\n @session-switch=${this.onChatSessionSwitch}\n @session-create=${this.onChatSessionCreate}\n @board-edit-patch=${this.onBoardEditPatch}\n @board-edit-revert=${this.onBoardEditRevert}\n @board-action-execute=${this.onBoardActionExecute}>\n </ox-board-ai-chat>\n </div>\n `\n : nothing}\n `\n }\n\n async updateBoard() {\n try {\n this.preparing = true\n\n const { id, name, description, groupId } = this.board\n const model = JSON.stringify(this.scene.model)\n\n await client.mutate({\n mutation: gql`\n mutation UpdateBoard($id: String!, $patch: BoardPatch!) {\n updateBoard(id: $id, patch: $patch) {\n id\n }\n }\n `,\n variables: {\n id,\n patch: { name, description, model, groupId }\n },\n context: gqlContext()\n })\n\n notify('info', i18next.t('text.saved'))\n } catch (ex) {\n notifyError(ex)\n } finally {\n this.preparing = false\n }\n\n this.updateContext()\n }\n\n async saveBoard() {\n await this.updateBoard()\n this.modeller?.preserve()\n }\n\n async canDeactivate(): Promise<boolean> {\n if (this.modeller?.hasUnpreservedChanges()) {\n return await OxPrompt.open({\n title: i18next.t('text.are_you_sure'),\n text: i18next.t('prompt.sure to navigate away?'),\n confirmButton: { text: i18next.t('button.confirm') },\n cancelButton: { text: i18next.t('button.cancel') }\n })\n }\n\n return true\n }\n}\n"]}
1
+ {"version":3,"file":"board-modeller-page.js","sourceRoot":"","sources":["../../client/pages/board-modeller-page.ts"],"names":[],"mappings":"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","sourcesContent":["import './things-scene-components.import'\nimport '@operato/board/ox-board-modeller.js'\nimport '@operato/oops'\nimport '@things-factory/board-ai'\n\nimport gql from 'graphql-tag'\nimport { css, html, nothing } from 'lit'\nimport { customElement, property, query, state } from 'lit/decorators.js'\nimport { BoardModeller } from '@operato/board/ox-board-modeller.js'\nimport { LoadTracker, SceneSearchEngine } from '@hatiolab/things-scene'\nimport { OxPropertyEditor } from '@operato/property-editor'\nimport { PageView, FontController } from '@operato/shell'\nimport { hasPrivilege } from '@things-factory/auth-base/dist-client/index.js'\nimport { applyBoardEditPatchVerbose } from '@things-factory/board-ai'\nimport type { BoardEditOp, BoardEditPatch } from '@things-factory/board-ai'\nimport { client, gqlContext } from '@operato/graphql'\nimport { i18next } from '@operato/i18n'\nimport { OxPrompt } from '@operato/popup/ox-prompt.js'\n\nimport { provider } from '../board-provider.js'\nimport components from './things-scene-components-with-tools.import'\nimport { notify, notifyError } from '../utils/notify-helper.js'\nimport { findSceneComponent, dispatchBoardAction } from './board-action-dispatch.js'\nimport { dispatchBoardEditOp } from './board-edit-dispatch.js'\n\nexport { findSceneComponent }\n\n/** 서버 PatchEntry.reverted=true 플래그 영속용 mutation. */\nconst REVERT_PATCH_MUTATION = gql`\n mutation RevertPatch($patchId: String!) {\n revertPatch(patchId: $patchId)\n }\n`\n\n/**\n * default 위에 override 를 깊게 merge.\n * - override 의 키 값이 undefined 가 아닌 경우 우선 적용\n * - override 의 nested object 는 default 와 deep merge\n * - array / primitive 는 override 가 그대로 우선\n */\nfunction mergeDefaultsDeep(defaults: any, override: any): any {\n if (defaults === null || defaults === undefined) return override\n if (override === null || override === undefined) return defaults\n if (typeof defaults !== 'object' || typeof override !== 'object') return override\n if (Array.isArray(defaults) || Array.isArray(override)) return override\n\n const out: any = { ...defaults }\n for (const key of Object.keys(override)) {\n const dv = (defaults as any)[key]\n const ov = (override as any)[key]\n if (\n dv !== null &&\n ov !== null &&\n typeof dv === 'object' &&\n typeof ov === 'object' &&\n !Array.isArray(dv) &&\n !Array.isArray(ov)\n ) {\n out[key] = mergeDefaultsDeep(dv, ov)\n } else if (ov !== undefined) {\n out[key] = ov\n }\n }\n return out\n}\n\n@customElement('board-modeller-page')\nexport class BoardModellerPage extends PageView {\n constructor() {\n super()\n\n components.forEach(({ templates = [], groups = [] }) => {\n groups.forEach(group => BoardModeller.registerGroup(group))\n BoardModeller.registerTemplate(templates)\n })\n\n /* 컴포넌트에서 정의된 에디터들을 MODELLER_EDITORS에 등록 */\n const addedEditors = {}\n for (const component in components) {\n let { editors } = components[component]\n\n editors &&\n editors.forEach(editor => {\n let { type, element } = editor\n\n addedEditors[type] = element\n })\n }\n\n OxPropertyEditor.register(addedEditors)\n }\n\n static styles = [\n css`\n :host {\n display: flex;\n flex-direction: row;\n\n overflow: hidden;\n position: relative;\n }\n\n .modeller-area {\n flex: 1;\n min-width: 0;\n display: flex;\n flex-direction: column;\n position: relative;\n }\n\n ox-board-modeller {\n flex: 1;\n }\n\n .ai-toggle {\n position: absolute;\n right: 12px;\n bottom: 12px;\n z-index: 10;\n padding: 8px 14px;\n border: 0;\n border-radius: 24px;\n background: #2563eb;\n color: #fff;\n font: 500 13px/1.4 system-ui, -apple-system, sans-serif;\n cursor: pointer;\n box-shadow: 0 4px 12px rgba(37, 99, 235, 0.35);\n transition: background 0.15s, transform 0.1s;\n }\n .ai-toggle:hover { background: #1d4ed8; }\n .ai-toggle:active { transform: scale(0.97); }\n .ai-toggle.open { background: #475569; }\n .ai-toggle.open:hover { background: #334155; }\n\n .ai-panel {\n flex: 0 0 320px;\n display: flex;\n border-left: 1px solid #e2e8f0;\n background: #ffffff;\n }\n\n ox-board-ai-chat { flex: 1; }\n\n ox-oops-note {\n display: block;\n position: absolute;\n left: 50%;\n top: 50%;\n transform: translate(-50%, -50%);\n }\n\n @media (max-width: 900px) {\n .ai-panel {\n flex-basis: 100%;\n position: absolute;\n inset: 0;\n z-index: 20;\n }\n }\n `\n ]\n\n @property({ type: String }) boardId?: string | null\n @property({ type: String }) boardName?: string = ''\n @property({ type: Object }) model: any = null\n @property({ type: Array }) selected: any = []\n @property({ type: Number }) mode?: number = 1\n @property({ type: Boolean }) hideProperty?: boolean = false\n @property({ type: String }) overlay?: string | null = null\n @property({ type: Object }) scene: any = null\n @property({ type: Array }) componentGroupList?: any[] = BoardModeller.groups\n @property({ type: Array }) propertyEditor: any\n @property({ type: Boolean }) preparing?: boolean\n\n /** AI 협력 세션 id — 현재 활성 세션. boardId 변경 시 자동 ensure. 다중 세션 시 활성 1개. */\n @state() chatSessionId?: string\n\n /** 보드의 모든 ChatSession (탭으로 표시). */\n @state() chatSessions: Array<{ id: string; name?: string; createdAt?: string }> = []\n\n /** AI 채팅 패널 열림 상태 (기본 닫힘) */\n @state() aiPanelOpen = false\n\n /**\n * 적용된 patch 별 inverse op 시퀀스 — Revert 기능용.\n *\n * Map<patchId, inverseOps[]>: 각 patch 적용 시점에 계산해 저장. revert 클릭\n * 시 역순 in-place 적용. 페이지 reload 시 손실 — reload 후 revert 시도는\n * notify 로 안내 (메모리 외 영속은 별개 작업).\n */\n private patchInverses = new Map<string, BoardEditOp[]>()\n\n /** 모델러에 등록된 컴포넌트 type 목록 — chat 호출 시 LLM 에 전달 (lazy cache) */\n private _knownTypesCache?: string[]\n /** 등록된 type 의 group 분류 — categories 로 LLM 에 전달 */\n private _knownCategoriesCache?: string[]\n\n private get knownTypes(): string[] {\n if (!this._knownTypesCache) {\n const types = new Set<string>()\n for (const group of BoardModeller.groups || []) {\n for (const t of group.templates || []) {\n if (t && typeof t.type === 'string') types.add(t.type)\n }\n }\n this._knownTypesCache = Array.from(types).sort()\n }\n return this._knownTypesCache\n }\n\n private get knownCategories(): string[] {\n if (!this._knownCategoriesCache) {\n const cats = new Set<string>()\n for (const group of BoardModeller.groups || []) {\n for (const t of group.templates || []) {\n if (t && typeof t.group === 'string') cats.add(t.group)\n }\n }\n this._knownCategoriesCache = Array.from(cats).sort()\n }\n return this._knownCategoriesCache\n }\n\n /**\n * type 별 유효 속성 스킴 — LLM 이 정확한 컴포넌트를 만들도록 (lazy cache).\n *\n * 핵심: 좌표 표현은 type 마다 다르다 (left/top vs cx/cy vs points 등).\n * 따라서 OMIT 하지 않고 `geometryKeys` 로 별도 노출 → LLM 이 type 별 좌표 키 사용 가능.\n */\n private _componentSchemasCache?: Array<{\n type: string\n description?: string\n group?: string\n /** template.model 의 좌표/크기 관련 키들 — type 별 다름 */\n geometryKeys?: string[]\n /** 좌표 외 type 특화 속성 (default 값 또는 nested 키 목록) */\n properties?: Record<string, any>\n }>\n\n private get componentSchemas() {\n if (!this._componentSchemasCache) {\n const out: Array<{\n type: string\n description?: string\n group?: string\n geometryKeys?: string[]\n properties?: Record<string, any>\n }> = []\n const NON_PROPS = new Set(['type', 'id', 'name', 'class'])\n // 좌표/크기/경로 관련 키들 — type 마다 다른 조합으로 사용\n const GEOMETRY_KEYS = new Set([\n 'left',\n 'top',\n 'right',\n 'bottom',\n 'x',\n 'y',\n 'cx',\n 'cy',\n 'width',\n 'height',\n 'rx',\n 'ry',\n 'radius',\n 'rotation',\n 'translate',\n 'scale',\n 'points',\n 'path',\n 'vertices',\n 'verticles',\n 'startX',\n 'startY',\n 'endX',\n 'endY',\n 'startPoint',\n 'endPoint'\n ])\n for (const group of BoardModeller.groups || []) {\n for (const t of group.templates || []) {\n if (!t || typeof t.type !== 'string') continue\n const properties: Record<string, any> = {}\n const geometryKeys: string[] = []\n const model: any = t.model || {}\n for (const key of Object.keys(model)) {\n if (NON_PROPS.has(key)) continue\n if (GEOMETRY_KEYS.has(key)) {\n geometryKeys.push(key)\n continue\n }\n const v = model[key]\n if (v && typeof v === 'object' && !Array.isArray(v)) {\n properties[key] = Object.fromEntries(\n Object.keys(v)\n .slice(0, 8)\n .map(k => [k, null])\n )\n } else if (Array.isArray(v)) {\n properties[key] = []\n } else {\n properties[key] = v\n }\n }\n out.push({\n type: t.type,\n description: t.description || undefined,\n group: t.group || undefined,\n geometryKeys: geometryKeys.length > 0 ? geometryKeys : undefined,\n properties: Object.keys(properties).length > 0 ? properties : undefined\n })\n }\n }\n this._componentSchemasCache = out\n }\n return this._componentSchemasCache\n }\n\n private board: any = null\n private _loadTracker?: LoadTracker\n private _fontCtrl = new FontController(this)\n @query('ox-board-modeller') modeller?: BoardModeller\n\n get context() {\n return {\n title: this.board ? this.boardName : this.preparing ? 'Fetching board...' : '',\n help: 'board-modeller/modeller',\n widebleed: true\n }\n }\n\n get oopsNote() {\n return {\n icon: 'color_lens',\n title: 'EMPTY BOARD',\n description: 'There are no board to be designed'\n }\n }\n\n async refresh() {\n if (!this.boardId) {\n this.board = null\n this.model = null\n\n return\n }\n try {\n // 이전 보드 완전히 클리어\n this.board = null\n this.model = null\n this._loadTracker = new LoadTracker()\n this._loadTracker.setPhase('fetch')\n this.preparing = true\n this.updateContext()\n\n const response = await client.query({\n query: gql`\n query FetchBoardById($id: String!) {\n board(id: $id) {\n id\n name\n model\n }\n }\n `,\n variables: { id: this.boardId },\n context: gqlContext()\n })\n\n this._loadTracker?.setPhase('parse')\n\n const board = response.data.board\n\n if (!board) {\n this.board = null\n throw new Error('board not found')\n }\n\n this.board = {\n ...board,\n model: JSON.parse(board.model)\n }\n\n this.boardName = this.board.name\n this.model = {\n ...this.board.model\n }\n\n // AI 협력 세션 — 보드 전환 시 이전 보드의 세션 / 세션 리스트 정리 (chat panel 의\n // sessionId/sessions 변화 감지 → history 비우고 새로 로드). 패널 열린 채 보드 옮겼다면\n // 새 보드의 세션 리스트를 즉시 ensure (fire-and-forget — 에러는 ensureChatSession 처리).\n this.chatSessionId = undefined\n this.chatSessions = []\n if (this.aiPanelOpen && this.boardId) {\n this.ensureChatSession()\n }\n } catch (ex) {\n notifyError(ex)\n } finally {\n this.preparing = false\n this.updateContext()\n }\n }\n\n /** 패널 토글 시 호출 — 열 때만 chatSession 보장 (세션 리스트 로드 + 활성 세션 결정) */\n private async toggleAIPanel() {\n const willOpen = !this.aiPanelOpen\n if (willOpen && this.boardId) {\n await this.ensureChatSession()\n }\n this.aiPanelOpen = willOpen\n }\n\n /** 보드의 모든 ChatSession 목록을 로드 + 활성 세션 보장.\n * - 세션이 0 개면 startBoardAISession 으로 첫 세션 생성 (단일 세션 UX 호환)\n * - 활성 세션 미설정 시 첫 번째 세션을 활성으로\n * - 이미 활성 세션 set 돼있고 list 에 포함되면 유지 */\n private async ensureChatSession() {\n if (!this.boardId) return\n try {\n const result = await client.query({\n query: gql`\n query ChatSessionsByBoard($boardId: String!) {\n chatSessionsByBoard(boardId: $boardId) {\n id\n name\n createdAt\n }\n }\n `,\n variables: { boardId: this.boardId },\n context: gqlContext(),\n fetchPolicy: 'network-only'\n })\n let sessions: Array<{ id: string; name?: string; createdAt?: string }> =\n result.data?.chatSessionsByBoard ?? []\n\n if (sessions.length === 0) {\n // 첫 세션 ensure (idempotent)\n const created = await client.mutate({\n mutation: gql`\n mutation StartBoardAISession($boardId: String!) {\n startBoardAISession(boardId: $boardId) {\n id\n name\n createdAt\n }\n }\n `,\n variables: { boardId: this.boardId },\n context: gqlContext()\n })\n const s = created.data?.startBoardAISession\n if (s) sessions = [s]\n }\n\n this.chatSessions = sessions\n // 활성 세션 결정 — 기존 active 가 list 에 있으면 유지, 아니면 첫 번째\n if (!this.chatSessionId || !sessions.some(s => s.id === this.chatSessionId)) {\n this.chatSessionId = sessions[0]?.id\n }\n } catch (ex) {\n notifyError(ex)\n }\n }\n\n /** `@` mention popup 의 도메인 사용자 후보 — chat 의 userProvider prop 으로 wire.\n * query 변할 때마다 호출되므로 server query (`domainUsersForMention`) 를 그대로 위임. */\n private fetchDomainUsersForMention = async (\n query: string\n ): Promise<Array<{ id: string; name?: string; email?: string }>> => {\n try {\n const result = await client.query({\n query: gql`\n query DomainUsersForMention($query: String, $limit: Int) {\n domainUsersForMention(query: $query, limit: $limit) {\n id\n name\n email\n }\n }\n `,\n variables: { query: query || null, limit: 50 },\n context: gqlContext(),\n fetchPolicy: 'cache-first'\n })\n return result.data?.domainUsersForMention ?? []\n } catch (ex) {\n // 조용히 — chat panel 이 빈 결과로 처리\n console.warn('[board-modeller-page] fetchDomainUsersForMention failed:', ex)\n return []\n }\n }\n\n /** 채팅 패널의 탭 클릭 — 활성 세션 전환. */\n private onChatSessionSwitch = (e: CustomEvent) => {\n const newId = e.detail?.sessionId\n if (typeof newId === 'string' && newId !== this.chatSessionId) {\n this.chatSessionId = newId\n }\n }\n\n /** 채팅 패널의 \"+\" 버튼 — 새 세션 생성 후 활성으로 전환. */\n private onChatSessionCreate = async () => {\n if (!this.boardId) return\n try {\n const created = await client.mutate({\n mutation: gql`\n mutation CreateChatSession($boardId: String!, $name: String) {\n createChatSession(boardId: $boardId, name: $name) {\n id\n name\n createdAt\n }\n }\n `,\n variables: { boardId: this.boardId, name: null },\n context: gqlContext()\n })\n const s = created.data?.createChatSession\n if (s) {\n this.chatSessions = [...this.chatSessions, s]\n this.chatSessionId = s.id\n }\n } catch (ex) {\n notifyError(ex)\n }\n }\n\n /**\n * 모델러의 selected (things-scene Component 인스턴스 배열) 에서 refid 추출.\n *\n * refid 는 things-scene 이 모든 컴포넌트에 자동 발급하는 universal numeric handle.\n * AI 의 selection 표현은 항상 refid 기반 (id 는 데이터 바인딩 이름이며 unique 가\n * 아니라 targeting 에 부적합).\n */\n private extractSelectedRefids(): number[] {\n const sel = this.selected || []\n return sel\n .map((c: any) => {\n if (!c || typeof c.get !== 'function') return null\n const refid = c.get('refid')\n return typeof refid === 'number' && Number.isFinite(refid) ? refid : null\n })\n .filter((x: number | null): x is number => x !== null)\n }\n\n /**\n * AI 채팅 mention popup (`#`) 검색 위임 — things-scene SceneSearchEngine 으로 wrap.\n *\n * F-key Finder 와 동일한 검색 룰 (id → name → tag → type → class → text → data → property)\n * 을 그대로 mention popup 에 재사용. 컴포넌트 트리를 직접 순회하므로 model JSON 으로\n * 변환할 필요 없음 — 일관성 + 성능 이점.\n *\n * 반환 구조 (FilterResult) 는 chat 의 mention-popup 표시 모델. host 는 things-scene\n * Component 에서 refid/id/type/tag/class/text 를 뽑아 candidate 로 채워 넘긴다.\n */\n private searchSceneForMention = (() => {\n const engine = new SceneSearchEngine()\n return (query: string): any[] => {\n const root = (this.scene as any)?.root\n if (!root) return []\n const results = engine.search(root, query)\n return results.map(r => {\n const c: any = r.component\n const refid = typeof c.get === 'function' ? c.get('refid') : undefined\n const id = typeof c.get === 'function' ? c.get('id') : undefined\n const type = typeof c.get === 'function' ? c.get('type') : undefined\n const tag = typeof c.get === 'function' ? c.get('tag') : undefined\n const cls = typeof c.get === 'function' ? c.get('class') : undefined\n const text = typeof c.get === 'function' ? c.get('text') : undefined\n return {\n candidate: {\n refid: typeof refid === 'number' ? refid : undefined,\n label: id || type || 'unknown',\n type: typeof type === 'string' ? type : 'unknown',\n id: typeof id === 'string' && id ? id : undefined,\n tag: typeof tag === 'string' && tag ? tag : undefined,\n class: typeof cls === 'string' && cls ? cls : undefined,\n text: typeof text === 'string' && text ? text : undefined\n },\n matchType: r.matchType,\n matchValue: r.matchValue\n }\n })\n }\n })()\n\n /**\n * 라이브 보드 모델 — things-scene 캔버스가 들고 있는 가장 최신 상태의 deep clone.\n *\n * 사용자 수작업 편집은 scene.model 에만 반영되고 this.model property 에는 자동\n * 동기화되지 않는다. AI 한테 보드 상태를 넘기거나 patch 를 적용할 때는 반드시\n * 이쪽을 쓴다.\n *\n * id 와 refid 는 별개 필드 — 컴포넌트는 model.id (선택, 사용자/AI 가 명시 설정)\n * 와 model.refid (필수, things-scene 자동 발급) 를 둘 다 가질 수 있고 AI 가\n * 양쪽을 구별해서 다룬다 (tool 인자가 분리됨). 따라서 여기서는 model 을 mutate\n * 하지 않고 raw deep clone 만 반환.\n */\n private getLiveBoard(): any {\n const sceneModel = (this.scene as any)?.model\n if (sceneModel) {\n return JSON.parse(JSON.stringify(sceneModel))\n }\n return this.model\n }\n\n /**\n * 채팅에서 patch 도착 — things-scene 에 in-place 적용.\n *\n * 핵심: `this.model = next` 로 모델을 통째 교체하면 ox-scene-viewer 가 scene 을\n * dispose → 재생성 → undo 히스토리/dirty 플래그가 모두 초기화된다.\n * 따라서 일반 add/modify/remove 는 scene 의 in-place mutation API 로 적용 →\n * commander 가 자동으로 snapshot push → CMD+Z 와 저장 안내 팝업이 정상 동작.\n *\n * 'replace' op (보드 전체 교체) 만은 부득이 wholesale 경로로 빠진다 — 이 경우는\n * 사용자에게 명시적으로 안내 후 진행.\n */\n private onBoardEditPatch = (e: CustomEvent) => {\n const { patch, patchId } = e.detail || {}\n if (!patch) return\n const scene = this.scene as any\n if (!scene) {\n notifyError(new Error('Scene not initialized'))\n return\n }\n\n const hasReplace = (patch.ops || []).some((op: BoardEditOp) => op.op === 'replace')\n if (hasReplace) {\n this.applyPatchWholesale(patch, patchId)\n return\n }\n\n try {\n const missed: BoardEditOp[] = []\n // 각 op 의 inverse — dispatcher 가 적용 직전/직후 scene 상태로 계산해 반환.\n // 누적했다가 onBoardEditRevert 가 역순 in-place 적용으로 복원.\n const inverseOps: BoardEditOp[] = []\n const normalize = (c: any) => this.normalizeComponent(c)\n\n for (const op of patch.ops as BoardEditOp[]) {\n const r = dispatchBoardEditOp(scene, op, { normalize })\n if (r.applied) {\n inverseOps.push(...r.inverseOps)\n } else {\n missed.push(op)\n }\n }\n\n // patch 단위로 inverse 보관 — Revert 시 역순 적용\n if (patchId && inverseOps.length > 0) {\n this.patchInverses.set(patchId, inverseOps)\n }\n\n if (missed.length > 0) {\n const missedDesc = missed\n .map((op: any) => {\n if (op.op === 'modify' || op.op === 'remove') return `refid=${op.refid}`\n if (op.op === 'modifyBoard') return 'modifyBoard (scene root unavailable)'\n return op.op\n })\n .join(', ')\n notify(\n 'warn',\n `AI 가 ${missed.length}개의 변경을 시도했지만 적용되지 않았습니다: ${missedDesc}`\n )\n }\n } catch (ex) {\n notifyError(ex)\n }\n }\n\n /**\n * 'replace' op 적용 — scene 통째로 재생성 (undo 히스토리 초기화).\n *\n * AI 가 보드 전체 재구성을 요청한 경우. 사용자에게 history 손실 안내.\n * board-import 데이터 wholesale 적용 등 드문 케이스에 한정.\n *\n * Revert 지원 — replace 직전 보드 model 을 inverse 로 보관 (또 다른 replace).\n */\n private applyPatchWholesale(patch: BoardEditPatch, patchId?: string) {\n try {\n const baseBoard = this.getLiveBoard()\n const inverseOps: BoardEditOp[] = baseBoard\n ? [{ op: 'replace', board: JSON.parse(JSON.stringify(baseBoard)) }]\n : []\n const report = applyBoardEditPatchVerbose(baseBoard, patch)\n const next = report.board as any\n if (report.applied.length === 0) return\n const normalized = (next.components || []).map((c: any) => this.normalizeComponent(c))\n this.model = {\n ...(baseBoard || {}),\n ...(next.width !== undefined && { width: next.width }),\n ...(next.height !== undefined && { height: next.height }),\n components: normalized\n }\n if (patchId && inverseOps.length > 0) {\n this.patchInverses.set(patchId, inverseOps)\n }\n notify('info', '보드 전체가 교체되어 실행 취소(undo) 히스토리가 초기화되었습니다.')\n } catch (ex) {\n notifyError(ex)\n }\n }\n\n /**\n * 채팅에서 board-edit-revert 도착 — 누적된 inverse 를 역순 in-place 적용.\n *\n * - 메모리상 patchInverses Map 만 본다. 페이지 reload 후엔 inverse 가 없어\n * 되돌릴 수 없음 (사용자에게 안내).\n * - 성공 시 서버 revertPatch mutation 호출 → PatchEntry.reverted=true 영속.\n * - in-place 적용이라 undo/dirty/selection 보존 (chatViaTools 의 모든 강점 그대로).\n */\n private onBoardEditRevert = async (e: CustomEvent) => {\n const { patchId } = e.detail || {}\n if (!patchId) return\n const inverses = this.patchInverses.get(patchId)\n if (!inverses || inverses.length === 0) {\n notify(\n 'warn',\n '이 변경은 자동 되돌릴 수 없습니다 (페이지 새로고침 이후 적용된 변경은 메모리에 inverse 가 없습니다).'\n )\n return\n }\n\n const scene = this.scene as any\n if (!scene) {\n notifyError(new Error('Scene not initialized'))\n return\n }\n\n try {\n // 역순 적용 — 마지막 op 부터 되돌림. 단일 patch 안에서도 순서 의미 있음\n // (예: add 후 modify 한 시퀀스를 되돌리려면 modify(원본) 후 remove)\n const reversed = [...inverses].reverse()\n const reverseFakePatch: BoardEditPatch = {\n ops: reversed,\n summary: 'revert',\n confidence: 1\n }\n // hasReplace 면 wholesale, 그렇지 않으면 in-place — 일반 onBoardEditPatch 와 동일\n const hasReplace = reversed.some(op => op.op === 'replace')\n if (hasReplace) {\n this.applyPatchWholesale(reverseFakePatch)\n } else {\n // in-place 직접 적용 — 새 inverse 누적은 안 함 (revert 의 revert 는 redo, 별개 기능)\n this.applyInverseOpsInPlace(scene, reversed)\n }\n\n this.patchInverses.delete(patchId)\n await this.recordRevertOnServer(patchId)\n } catch (ex) {\n notifyError(ex)\n }\n }\n\n /**\n * inverse op 시퀀스를 things-scene 위에 in-place 적용 (snapshot/undo 보존).\n * dispatcher 재사용 — inverse 누적은 무시 (revert 의 revert 는 redo, 별개 기능).\n */\n private applyInverseOpsInPlace(scene: any, ops: BoardEditOp[]): void {\n const normalize = (c: any) => this.normalizeComponent(c)\n for (const op of ops) {\n dispatchBoardEditOp(scene, op, { normalize })\n }\n }\n\n /** 서버 PatchEntry.reverted=true 영속 — fail 해도 UI 는 이미 되돌렸으니 warn 만. */\n private async recordRevertOnServer(patchId: string): Promise<void> {\n try {\n await client.mutate({ mutation: REVERT_PATCH_MUTATION, variables: { patchId } })\n } catch (e: any) {\n console.warn('[board-modeller] revertPatch server persist failed:', e?.message ?? e)\n }\n }\n\n /**\n * AI 가 ephemeral scene action (selection / view / mode) 을 실행하라고 요청.\n *\n * board-edit-patch 와 별개 채널 — 모델 변경 X, undo 영향 X, dirty flag 무관.\n * things-scene API 직접 호출. 잘못된 action / 누락된 컴포넌트는 console.warn 만.\n */\n private onBoardActionExecute = (e: CustomEvent) => {\n const { actions } = e.detail || {}\n if (!Array.isArray(actions) || actions.length === 0) return\n const scene = this.scene as any\n if (!scene) return\n\n let modeChanged = false\n for (const action of actions) {\n try {\n dispatchBoardAction(scene, action)\n if (action?.action === 'setSceneMode') modeChanged = true\n } catch (ex: any) {\n console.warn('[board-modeller] action execute failed:', action, ex?.message ?? ex)\n }\n }\n // setSceneMode 가 있었으면 page 의 reactive mode 도 동기 (Lit re-render 트리거)\n if (modeChanged) this.mode = scene.mode\n }\n\n /**\n * AI 가 만든 컴포넌트를 things-scene 이 기대하는 완전한 형태로 normalize.\n * BoardModeller 에 등록된 template.model (default 값) 위에 AI 응답을 덮어쓴다.\n * 이 단계 없이 things-scene 에 넣으면 필수 속성 누락 시 render 에서 crash 가능\n * (예: gauge-circle 의 value 가 undefined → toString() TypeError).\n */\n private normalizeComponent(c: any): any {\n if (!c || typeof c !== 'object' || typeof c.type !== 'string') return c\n const template = this.findTemplateByType(c.type)\n if (!template || !template.model) return c\n return mergeDefaultsDeep(template.model, c)\n }\n\n private findTemplateByType(type: string): any {\n for (const group of BoardModeller.groups || []) {\n for (const t of group.templates || []) {\n if (t && t.type === type) return t\n }\n }\n return undefined\n }\n\n async updated(changes) {\n if (changes.has('boardId')) {\n try {\n if (await hasPrivilege({ privilege: 'mutation', category: 'board' })) {\n this.refresh()\n } else {\n this.boardId = null\n this.model = null\n this.modeller?.close()\n }\n } catch {\n this.boardId = null\n this.model = null\n this.modeller?.close()\n }\n }\n }\n\n pageUpdated(changes, lifecycle) {\n if (this.active) {\n this.boardId = lifecycle.resourceId\n } else {\n this.boardId = null\n this.board = null\n this.model = null\n this.preparing = false\n this.modeller?.close()\n this.updateContext()\n }\n }\n\n render() {\n const oops = !this.preparing && !this.model && this.oopsNote\n\n return html`\n <div class=\"modeller-area\">\n ${oops\n ? html`<ox-oops-note icon=${oops.icon} title=${oops.title} description=${oops.description}></ox-oops-note>`\n : html`\n <ox-board-modeller\n .mode=${this.mode}\n .loadTracker=${this._loadTracker}\n @mode-changed=${e => {\n this.mode = e.detail.value\n }}\n .model=${this.model}\n @model-changed=${e => {\n this.model = e.detail.value\n }}\n .scene=${this.scene}\n @scene-changed=${e => {\n this.scene = e.detail.value\n }}\n .selected=${this.selected}\n @selected-changed=${e => {\n this.selected = e.detail.value\n }}\n .provider=${provider}\n @save-model=${e => this.saveBoard()}\n .componentGroupList=${this.componentGroupList}\n .fonts=${this._fontCtrl.fonts}\n .hideProperty=${this.hideProperty}\n >\n </ox-board-modeller>\n `}\n ${this.boardId\n ? html`\n <button\n class=\"ai-toggle ${this.aiPanelOpen ? 'open' : ''}\"\n title=\"${i18next.t('text.ai_assistant') || 'AI Assistant'}\"\n @click=${this.toggleAIPanel}>\n ${this.aiPanelOpen ? '✕ AI' : '✨ AI'}\n </button>\n `\n : nothing}\n </div>\n ${this.aiPanelOpen\n ? html`\n <div class=\"ai-panel\">\n <ox-board-ai-chat\n .sessionId=${this.chatSessionId}\n .sessions=${this.chatSessions}\n .currentBoard=${this.model}\n .boardProvider=${() => this.getLiveBoard()}\n .searchProvider=${this.searchSceneForMention}\n .userProvider=${this.fetchDomainUsersForMention}\n .knownTypes=${this.knownTypes}\n .categories=${this.knownCategories}\n .componentSchemas=${this.componentSchemas}\n .selectedRefids=${this.extractSelectedRefids()}\n @session-switch=${this.onChatSessionSwitch}\n @session-create=${this.onChatSessionCreate}\n @board-edit-patch=${this.onBoardEditPatch}\n @board-edit-revert=${this.onBoardEditRevert}\n @board-action-execute=${this.onBoardActionExecute}>\n </ox-board-ai-chat>\n </div>\n `\n : nothing}\n `\n }\n\n async updateBoard() {\n try {\n this.preparing = true\n\n const { id, name, description, groupId } = this.board\n const model = JSON.stringify(this.scene.model)\n\n await client.mutate({\n mutation: gql`\n mutation UpdateBoard($id: String!, $patch: BoardPatch!) {\n updateBoard(id: $id, patch: $patch) {\n id\n }\n }\n `,\n variables: {\n id,\n patch: { name, description, model, groupId }\n },\n context: gqlContext()\n })\n\n notify('info', i18next.t('text.saved'))\n } catch (ex) {\n notifyError(ex)\n } finally {\n this.preparing = false\n }\n\n this.updateContext()\n }\n\n async saveBoard() {\n await this.updateBoard()\n this.modeller?.preserve()\n }\n\n async canDeactivate(): Promise<boolean> {\n if (this.modeller?.hasUnpreservedChanges()) {\n return await OxPrompt.open({\n title: i18next.t('text.are_you_sure'),\n text: i18next.t('prompt.sure to navigate away?'),\n confirmButton: { text: i18next.t('button.confirm') },\n cancelButton: { text: i18next.t('button.cancel') }\n })\n }\n\n return true\n }\n}\n"]}