@operato/board 10.0.0-beta.46 → 10.0.0-beta.48

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.
package/CHANGELOG.md CHANGED
@@ -3,6 +3,24 @@
3
3
  All notable changes to this project will be documented in this file.
4
4
  See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
5
5
 
6
+ ## [10.0.0-beta.48](https://github.com/hatiolab/operato/compare/v10.0.0-beta.47...v10.0.0-beta.48) (2026-05-05)
7
+
8
+
9
+ ### :rocket: New Features
10
+
11
+ * **property-panel:** generic-object 타입 변환 UI를 specifics 탭 전용 에디터로 이동 ([4b90f57](https://github.com/hatiolab/operato/commit/4b90f574c29f1677eb0dacc73b05e9fd10c9d2b2))
12
+
13
+
14
+
15
+ ## [10.0.0-beta.47](https://github.com/hatiolab/operato/compare/v10.0.0-beta.46...v10.0.0-beta.47) (2026-05-03)
16
+
17
+
18
+ ### :rocket: New Features
19
+
20
+ * **modeller:** popup-container palette + inline-popup tap actions ([737b081](https://github.com/hatiolab/operato/commit/737b0819c287d7703f404db266e79b12281070d3))
21
+
22
+
23
+
6
24
  ## [10.0.0-beta.46](https://github.com/hatiolab/operato/compare/v10.0.0-beta.45...v10.0.0-beta.46) (2026-05-01)
7
25
 
8
26
  **Note:** Version bump only for package @operato/board
@@ -52,6 +52,23 @@ export const container = {
52
52
  strokeStyle: 'DarkGoldenRod',
53
53
  hidden: true
54
54
  }
55
+ },
56
+ {
57
+ type: 'popup-container',
58
+ description: 'popup template — modeled in board, fresh instance per popup show',
59
+ icon: popup,
60
+ group: 'container',
61
+ model: {
62
+ type: 'popup-container',
63
+ left: 100,
64
+ top: 100,
65
+ width: 320,
66
+ height: 200,
67
+ fillStyle: '#fff',
68
+ strokeStyle: '#999',
69
+ lineWidth: 1,
70
+ position: 'popup'
71
+ }
55
72
  }
56
73
  ]
57
74
  };
@@ -1 +1 @@
1
- {"version":3,"file":"container.js","sourceRoot":"","sources":["../../../src/component/container.ts"],"names":[],"mappings":"AAEA,MAAM,aAAa,GAAG,IAAI,GAAG,CAAC,yCAAyC,EAAE,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAA;AAC9F,MAAM,KAAK,GAAG,IAAI,GAAG,CAAC,qCAAqC,EAAE,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAA;AAElF,MAAM,IAAI,GAAG;;;;;;;;;;;;;CAaZ,CAAA;AAED,MAAM,CAAC,MAAM,SAAS,GAAmB;IACvC,IAAI,EAAE,WAAW;IACjB,WAAW,EAAE,+BAA+B;IAC5C,IAAI;IACJ,SAAS,EAAE;QACT;YACE,IAAI,EAAE,WAAW;YACjB,WAAW,EAAE,mBAAmB;YAChC,IAAI,EAAE,aAAa;YACnB,KAAK,EAAE,WAAW;YAClB,KAAK,EAAE;gBACL,IAAI,EAAE,WAAW;gBACjB,IAAI,EAAE,GAAG;gBACT,GAAG,EAAE,GAAG;gBACR,KAAK,EAAE,GAAG;gBACV,MAAM,EAAE,GAAG;gBACX,SAAS,EAAE,MAAM;gBACjB,WAAW,EAAE,MAAM;gBACnB,KAAK,EAAE,CAAC;gBACR,MAAM,EAAE,KAAK;gBACb,SAAS,EAAE,CAAC;aACb;SACF;QACD;YACE,IAAI,EAAE,OAAO;YACb,WAAW,EAAE,cAAc;YAC3B,IAAI,EAAE,KAAK;YACX,KAAK,EAAE,WAAW;YAClB,KAAK,EAAE;gBACL,IAAI,EAAE,OAAO;gBACb,IAAI,EAAE,GAAG;gBACT,GAAG,EAAE,GAAG;gBACR,KAAK,EAAE,GAAG;gBACV,MAAM,EAAE,GAAG;gBACX,SAAS,EAAE,MAAM;gBACjB,WAAW,EAAE,eAAe;gBAC5B,MAAM,EAAE,IAAI;aACb;SACF;KACF;CACF,CAAA","sourcesContent":["import { ComponentGroup } from '../types.js'\n\nconst iconContainer = new URL('../../../icons/components/container.png', import.meta.url).href\nconst popup = new URL('../../../icons/components/popup.png', import.meta.url).href\n\nconst icon = `\n<svg version=\"1.1\" id=\"Layer_1\" xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\" x=\"0px\" y=\"0px\" viewBox=\"0 0 30 30\" style=\"enable-background:new 0 0 30 30;\" xml:space=\"preserve\">\n <style type=\"text/css\">\n .st0{fill:{{strokeColor}};}\n .st3{fill:none;stroke:{{strokeColor}};stroke-width:2;stroke-linecap:round;stroke-miterlimit:10;}\n .st9{fill:none;stroke:{{strokeColor}};stroke-miterlimit:10;}\n </style>\n <g>\n <rect x=\"2.8\" y=\"10.3\" class=\"st9\" width=\"24.4\" height=\"13.8\"/>\n <polygon class=\"st0\" points=\"27.7,9.8 2.3,9.8 7.8,5.9 22.2,5.9\"/>\n <line class=\"st3\" x1=\"10.9\" y1=\"15.5\" x2=\"19.1\" y2=\"15.5\"/>\n </g>\n</svg>\n`\n\nexport const container: ComponentGroup = {\n name: 'container',\n description: 'a group of various containers',\n icon,\n templates: [\n {\n type: 'container',\n description: 'general container',\n icon: iconContainer,\n group: 'container',\n model: {\n type: 'container',\n left: 100,\n top: 100,\n width: 100,\n height: 100,\n fillStyle: '#fff',\n strokeStyle: '#999',\n alpha: 1,\n hidden: false,\n lineWidth: 1\n }\n },\n {\n type: 'popup',\n description: 'popup window',\n icon: popup,\n group: 'container',\n model: {\n type: 'popup',\n left: 100,\n top: 100,\n width: 100,\n height: 100,\n fillStyle: '#fff',\n strokeStyle: 'DarkGoldenRod',\n hidden: true\n }\n }\n ]\n}\n"]}
1
+ {"version":3,"file":"container.js","sourceRoot":"","sources":["../../../src/component/container.ts"],"names":[],"mappings":"AAEA,MAAM,aAAa,GAAG,IAAI,GAAG,CAAC,yCAAyC,EAAE,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAA;AAC9F,MAAM,KAAK,GAAG,IAAI,GAAG,CAAC,qCAAqC,EAAE,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAA;AAElF,MAAM,IAAI,GAAG;;;;;;;;;;;;;CAaZ,CAAA;AAED,MAAM,CAAC,MAAM,SAAS,GAAmB;IACvC,IAAI,EAAE,WAAW;IACjB,WAAW,EAAE,+BAA+B;IAC5C,IAAI;IACJ,SAAS,EAAE;QACT;YACE,IAAI,EAAE,WAAW;YACjB,WAAW,EAAE,mBAAmB;YAChC,IAAI,EAAE,aAAa;YACnB,KAAK,EAAE,WAAW;YAClB,KAAK,EAAE;gBACL,IAAI,EAAE,WAAW;gBACjB,IAAI,EAAE,GAAG;gBACT,GAAG,EAAE,GAAG;gBACR,KAAK,EAAE,GAAG;gBACV,MAAM,EAAE,GAAG;gBACX,SAAS,EAAE,MAAM;gBACjB,WAAW,EAAE,MAAM;gBACnB,KAAK,EAAE,CAAC;gBACR,MAAM,EAAE,KAAK;gBACb,SAAS,EAAE,CAAC;aACb;SACF;QACD;YACE,IAAI,EAAE,OAAO;YACb,WAAW,EAAE,cAAc;YAC3B,IAAI,EAAE,KAAK;YACX,KAAK,EAAE,WAAW;YAClB,KAAK,EAAE;gBACL,IAAI,EAAE,OAAO;gBACb,IAAI,EAAE,GAAG;gBACT,GAAG,EAAE,GAAG;gBACR,KAAK,EAAE,GAAG;gBACV,MAAM,EAAE,GAAG;gBACX,SAAS,EAAE,MAAM;gBACjB,WAAW,EAAE,eAAe;gBAC5B,MAAM,EAAE,IAAI;aACb;SACF;QACD;YACE,IAAI,EAAE,iBAAiB;YACvB,WAAW,EAAE,kEAAkE;YAC/E,IAAI,EAAE,KAAK;YACX,KAAK,EAAE,WAAW;YAClB,KAAK,EAAE;gBACL,IAAI,EAAE,iBAAiB;gBACvB,IAAI,EAAE,GAAG;gBACT,GAAG,EAAE,GAAG;gBACR,KAAK,EAAE,GAAG;gBACV,MAAM,EAAE,GAAG;gBACX,SAAS,EAAE,MAAM;gBACjB,WAAW,EAAE,MAAM;gBACnB,SAAS,EAAE,CAAC;gBACZ,QAAQ,EAAE,OAAO;aAClB;SACF;KACF;CACF,CAAA","sourcesContent":["import { ComponentGroup } from '../types.js'\n\nconst iconContainer = new URL('../../../icons/components/container.png', import.meta.url).href\nconst popup = new URL('../../../icons/components/popup.png', import.meta.url).href\n\nconst icon = `\n<svg version=\"1.1\" id=\"Layer_1\" xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\" x=\"0px\" y=\"0px\" viewBox=\"0 0 30 30\" style=\"enable-background:new 0 0 30 30;\" xml:space=\"preserve\">\n <style type=\"text/css\">\n .st0{fill:{{strokeColor}};}\n .st3{fill:none;stroke:{{strokeColor}};stroke-width:2;stroke-linecap:round;stroke-miterlimit:10;}\n .st9{fill:none;stroke:{{strokeColor}};stroke-miterlimit:10;}\n </style>\n <g>\n <rect x=\"2.8\" y=\"10.3\" class=\"st9\" width=\"24.4\" height=\"13.8\"/>\n <polygon class=\"st0\" points=\"27.7,9.8 2.3,9.8 7.8,5.9 22.2,5.9\"/>\n <line class=\"st3\" x1=\"10.9\" y1=\"15.5\" x2=\"19.1\" y2=\"15.5\"/>\n </g>\n</svg>\n`\n\nexport const container: ComponentGroup = {\n name: 'container',\n description: 'a group of various containers',\n icon,\n templates: [\n {\n type: 'container',\n description: 'general container',\n icon: iconContainer,\n group: 'container',\n model: {\n type: 'container',\n left: 100,\n top: 100,\n width: 100,\n height: 100,\n fillStyle: '#fff',\n strokeStyle: '#999',\n alpha: 1,\n hidden: false,\n lineWidth: 1\n }\n },\n {\n type: 'popup',\n description: 'popup window',\n icon: popup,\n group: 'container',\n model: {\n type: 'popup',\n left: 100,\n top: 100,\n width: 100,\n height: 100,\n fillStyle: '#fff',\n strokeStyle: 'DarkGoldenRod',\n hidden: true\n }\n },\n {\n type: 'popup-container',\n description: 'popup template — modeled in board, fresh instance per popup show',\n icon: popup,\n group: 'container',\n model: {\n type: 'popup-container',\n left: 100,\n top: 100,\n width: 320,\n height: 200,\n fillStyle: '#fff',\n strokeStyle: '#999',\n lineWidth: 1,\n position: 'popup'\n }\n }\n ]\n}\n"]}
@@ -4,6 +4,7 @@ export { BoardDataStorage, PlaylistStorage } from './data-storage/data-storage.j
4
4
  export { BoardModelCache } from './data-storage/board-model-cache.js';
5
5
  export type { CachedBoard } from './data-storage/board-model-cache.js';
6
6
  export * from './ox-board-viewer.js';
7
+ export * from './ox-board-preview.js';
7
8
  export * from './ox-board-player.js';
8
9
  export * from './ox-board-modeller.js';
9
10
  export * from './ox-board-list.js';
package/dist/src/index.js CHANGED
@@ -3,6 +3,7 @@ export { registerDefaultGroups } from './component/register-default-groups.js';
3
3
  export { BoardDataStorage, PlaylistStorage } from './data-storage/data-storage.js';
4
4
  export { BoardModelCache } from './data-storage/board-model-cache.js';
5
5
  export * from './ox-board-viewer.js';
6
+ export * from './ox-board-preview.js';
6
7
  export * from './ox-board-player.js';
7
8
  export * from './ox-board-modeller.js';
8
9
  export * from './ox-board-list.js';
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,YAAY,CAAA;AAE1B,OAAO,EAAE,qBAAqB,EAAE,MAAM,wCAAwC,CAAA;AAC9E,OAAO,EAAE,gBAAgB,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAA;AAClF,OAAO,EAAE,eAAe,EAAE,MAAM,qCAAqC,CAAA;AAGrE,cAAc,sBAAsB,CAAA;AACpC,cAAc,sBAAsB,CAAA;AACpC,cAAc,wBAAwB,CAAA;AACtC,cAAc,oBAAoB,CAAA;AAClC,cAAc,+BAA+B,CAAA","sourcesContent":["export * from './types.js'\n\nexport { registerDefaultGroups } from './component/register-default-groups.js'\nexport { BoardDataStorage, PlaylistStorage } from './data-storage/data-storage.js'\nexport { BoardModelCache } from './data-storage/board-model-cache.js'\nexport type { CachedBoard } from './data-storage/board-model-cache.js'\n\nexport * from './ox-board-viewer.js'\nexport * from './ox-board-player.js'\nexport * from './ox-board-modeller.js'\nexport * from './ox-board-list.js'\nexport * from './ox-board-template-viewer.js'\n"]}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,YAAY,CAAA;AAE1B,OAAO,EAAE,qBAAqB,EAAE,MAAM,wCAAwC,CAAA;AAC9E,OAAO,EAAE,gBAAgB,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAA;AAClF,OAAO,EAAE,eAAe,EAAE,MAAM,qCAAqC,CAAA;AAGrE,cAAc,sBAAsB,CAAA;AACpC,cAAc,uBAAuB,CAAA;AACrC,cAAc,sBAAsB,CAAA;AACpC,cAAc,wBAAwB,CAAA;AACtC,cAAc,oBAAoB,CAAA;AAClC,cAAc,+BAA+B,CAAA","sourcesContent":["export * from './types.js'\n\nexport { registerDefaultGroups } from './component/register-default-groups.js'\nexport { BoardDataStorage, PlaylistStorage } from './data-storage/data-storage.js'\nexport { BoardModelCache } from './data-storage/board-model-cache.js'\nexport type { CachedBoard } from './data-storage/board-model-cache.js'\n\nexport * from './ox-board-viewer.js'\nexport * from './ox-board-preview.js'\nexport * from './ox-board-player.js'\nexport * from './ox-board-modeller.js'\nexport * from './ox-board-list.js'\nexport * from './ox-board-template-viewer.js'\n"]}
@@ -0,0 +1,36 @@
1
+ /**
2
+ * @license Copyright © HatioLab Inc. All rights reserved.
3
+ *
4
+ * `<ox-board-preview>` — boardModel JSON 을 read-only 시각으로 띄우는 lightweight 컴포넌트.
5
+ *
6
+ * 용도:
7
+ * - board-import wizard 의 review 단계에서 시안 (variant) 별 썸네일 + 큰 미리보기
8
+ * - 그 외 인터랙션 없이 보드를 살짝 보여주는 경우 (board card hover preview 등)
9
+ *
10
+ * 차이 (`<ox-board-viewer>` 대비):
11
+ * - 데이터 바인딩 / playback / fullscreen / 검색 / popup / scenario 모두 X
12
+ * - boardModel prop 직접 받음 (Board entity 가 필요 없음 — 미저장 model 도 지원)
13
+ * - dataSubscription 미연결 — 정적 시각만
14
+ * - 사용자 입력 미수용 (selectable, dragable 모두 비활성)
15
+ */
16
+ import { LitElement, type PropertyValues } from 'lit';
17
+ export declare class BoardPreview extends LitElement {
18
+ static styles: import("lit").CSSResult[];
19
+ /**
20
+ * 보드 모델 JSON. 변경 시 Scene 재생성 (cheap operation).
21
+ * Object reference 가 같아도 deep 변경은 감지 못 하므로, 호출자가 새 객체로 갱신.
22
+ */
23
+ boardModel: any;
24
+ /** 사용자 클릭 / 휠 등 인터랙션 차단 (썸네일 모드용). */
25
+ interactive: boolean;
26
+ private _target;
27
+ private _scene;
28
+ render(): import("lit-html").TemplateResult<1>;
29
+ firstUpdated(): void;
30
+ updated(changes: PropertyValues<this>): void;
31
+ disconnectedCallback(): void;
32
+ /** Scene 새로 만들고 target 에 부착. 기존 Scene 은 dispose. */
33
+ private _rebuild;
34
+ /** Scene release + target 분리. */
35
+ private _dispose;
36
+ }
@@ -0,0 +1,114 @@
1
+ import { __decorate } from "tslib";
2
+ /**
3
+ * @license Copyright © HatioLab Inc. All rights reserved.
4
+ *
5
+ * `<ox-board-preview>` — boardModel JSON 을 read-only 시각으로 띄우는 lightweight 컴포넌트.
6
+ *
7
+ * 용도:
8
+ * - board-import wizard 의 review 단계에서 시안 (variant) 별 썸네일 + 큰 미리보기
9
+ * - 그 외 인터랙션 없이 보드를 살짝 보여주는 경우 (board card hover preview 등)
10
+ *
11
+ * 차이 (`<ox-board-viewer>` 대비):
12
+ * - 데이터 바인딩 / playback / fullscreen / 검색 / popup / scenario 모두 X
13
+ * - boardModel prop 직접 받음 (Board entity 가 필요 없음 — 미저장 model 도 지원)
14
+ * - dataSubscription 미연결 — 정적 시각만
15
+ * - 사용자 입력 미수용 (selectable, dragable 모두 비활성)
16
+ */
17
+ import { css, html, LitElement } from 'lit';
18
+ import { customElement, property, query } from 'lit/decorators.js';
19
+ import { create, SCENE_MODE } from '@hatiolab/things-scene';
20
+ let BoardPreview = class BoardPreview extends LitElement {
21
+ constructor() {
22
+ super(...arguments);
23
+ /**
24
+ * 보드 모델 JSON. 변경 시 Scene 재생성 (cheap operation).
25
+ * Object reference 가 같아도 deep 변경은 감지 못 하므로, 호출자가 새 객체로 갱신.
26
+ */
27
+ this.boardModel = null;
28
+ /** 사용자 클릭 / 휠 등 인터랙션 차단 (썸네일 모드용). */
29
+ this.interactive = false;
30
+ this._scene = null;
31
+ }
32
+ render() {
33
+ return html `<div
34
+ id="target"
35
+ style=${this.interactive ? '' : 'pointer-events: none;'}
36
+ ></div>`;
37
+ }
38
+ firstUpdated() {
39
+ this._rebuild();
40
+ }
41
+ updated(changes) {
42
+ if (changes.has('boardModel') && this._target) {
43
+ this._rebuild();
44
+ }
45
+ }
46
+ disconnectedCallback() {
47
+ this._dispose();
48
+ super.disconnectedCallback();
49
+ }
50
+ /** Scene 새로 만들고 target 에 부착. 기존 Scene 은 dispose. */
51
+ _rebuild() {
52
+ var _a, _b;
53
+ this._dispose();
54
+ if (!this.boardModel || !this._target)
55
+ return;
56
+ try {
57
+ this._scene = create({
58
+ model: { ...this.boardModel },
59
+ mode: SCENE_MODE.VIEW,
60
+ handlers: []
61
+ });
62
+ // 컴포넌트 모두 보이도록 fit
63
+ (_b = (_a = this._scene).fit) === null || _b === void 0 ? void 0 : _b.call(_a, this._scene.fitMode);
64
+ this._scene.target = this._target;
65
+ }
66
+ catch (e) {
67
+ console.warn('[ox-board-preview] failed to render boardModel:', e);
68
+ }
69
+ }
70
+ /** Scene release + target 분리. */
71
+ _dispose() {
72
+ var _a, _b;
73
+ if (!this._scene)
74
+ return;
75
+ try {
76
+ this._scene.target = null;
77
+ (_b = (_a = this._scene).release) === null || _b === void 0 ? void 0 : _b.call(_a);
78
+ }
79
+ catch (_c) {
80
+ /* swallow — preview 정리는 best effort */
81
+ }
82
+ this._scene = null;
83
+ }
84
+ };
85
+ BoardPreview.styles = [
86
+ css `
87
+ :host {
88
+ display: block;
89
+ position: relative;
90
+ width: 100%;
91
+ height: 100%;
92
+ overflow: hidden;
93
+ }
94
+ #target {
95
+ position: absolute;
96
+ inset: 0;
97
+ outline: 0;
98
+ }
99
+ `
100
+ ];
101
+ __decorate([
102
+ property({ type: Object })
103
+ ], BoardPreview.prototype, "boardModel", void 0);
104
+ __decorate([
105
+ property({ type: Boolean })
106
+ ], BoardPreview.prototype, "interactive", void 0);
107
+ __decorate([
108
+ query('#target')
109
+ ], BoardPreview.prototype, "_target", void 0);
110
+ BoardPreview = __decorate([
111
+ customElement('ox-board-preview')
112
+ ], BoardPreview);
113
+ export { BoardPreview };
114
+ //# sourceMappingURL=ox-board-preview.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ox-board-preview.js","sourceRoot":"","sources":["../../src/ox-board-preview.ts"],"names":[],"mappings":";AAAA;;;;;;;;;;;;;;GAcG;AACH,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,UAAU,EAAuB,MAAM,KAAK,CAAA;AAChE,OAAO,EAAE,aAAa,EAAE,QAAQ,EAAE,KAAK,EAAE,MAAM,mBAAmB,CAAA;AAElE,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,wBAAwB,CAAA;AAGpD,IAAM,YAAY,GAAlB,MAAM,YAAa,SAAQ,UAAU;IAArC;;QAkBL;;;WAGG;QACyB,eAAU,GAAQ,IAAI,CAAA;QAElD,sCAAsC;QACT,gBAAW,GAAY,KAAK,CAAA;QAGjD,WAAM,GAAQ,IAAI,CAAA;IAqD5B,CAAC;IAnDC,MAAM;QACJ,OAAO,IAAI,CAAA;;cAED,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,uBAAuB;YACjD,CAAA;IACV,CAAC;IAED,YAAY;QACV,IAAI,CAAC,QAAQ,EAAE,CAAA;IACjB,CAAC;IAED,OAAO,CAAC,OAA6B;QACnC,IAAI,OAAO,CAAC,GAAG,CAAC,YAAY,CAAC,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;YAC9C,IAAI,CAAC,QAAQ,EAAE,CAAA;QACjB,CAAC;IACH,CAAC;IAED,oBAAoB;QAClB,IAAI,CAAC,QAAQ,EAAE,CAAA;QACf,KAAK,CAAC,oBAAoB,EAAE,CAAA;IAC9B,CAAC;IAED,oDAAoD;IAC5C,QAAQ;;QACd,IAAI,CAAC,QAAQ,EAAE,CAAA;QACf,IAAI,CAAC,IAAI,CAAC,UAAU,IAAI,CAAC,IAAI,CAAC,OAAO;YAAE,OAAM;QAC7C,IAAI,CAAC;YACH,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;gBACnB,KAAK,EAAE,EAAE,GAAG,IAAI,CAAC,UAAU,EAAE;gBAC7B,IAAI,EAAE,UAAU,CAAC,IAAI;gBACrB,QAAQ,EAAE,EAAE;aACN,CAAC,CAAA;YACT,mBAAmB;YACnB,MAAA,MAAA,IAAI,CAAC,MAAM,EAAC,GAAG,mDAAG,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAA;YACtC,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,IAAI,CAAC,OAAO,CAAA;QACnC,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,OAAO,CAAC,IAAI,CAAC,iDAAiD,EAAE,CAAC,CAAC,CAAA;QACpE,CAAC;IACH,CAAC;IAED,iCAAiC;IACzB,QAAQ;;QACd,IAAI,CAAC,IAAI,CAAC,MAAM;YAAE,OAAM;QACxB,IAAI,CAAC;YACH,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,IAAI,CAAA;YACzB,MAAA,MAAA,IAAI,CAAC,MAAM,EAAC,OAAO,kDAAI,CAAA;QACzB,CAAC;QAAC,WAAM,CAAC;YACP,uCAAuC;QACzC,CAAC;QACD,IAAI,CAAC,MAAM,GAAG,IAAI,CAAA;IACpB,CAAC;;AA/EM,mBAAM,GAAG;IACd,GAAG,CAAA;;;;;;;;;;;;;KAaF;CACF,AAfY,CAeZ;AAM2B;IAA3B,QAAQ,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC;gDAAuB;AAGrB;IAA5B,QAAQ,CAAC,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC;iDAA6B;AAE/B;IAAzB,KAAK,CAAC,SAAS,CAAC;6CAA8B;AA3BpC,YAAY;IADxB,aAAa,CAAC,kBAAkB,CAAC;GACrB,YAAY,CAiFxB","sourcesContent":["/**\n * @license Copyright © HatioLab Inc. All rights reserved.\n *\n * `<ox-board-preview>` — boardModel JSON 을 read-only 시각으로 띄우는 lightweight 컴포넌트.\n *\n * 용도:\n * - board-import wizard 의 review 단계에서 시안 (variant) 별 썸네일 + 큰 미리보기\n * - 그 외 인터랙션 없이 보드를 살짝 보여주는 경우 (board card hover preview 등)\n *\n * 차이 (`<ox-board-viewer>` 대비):\n * - 데이터 바인딩 / playback / fullscreen / 검색 / popup / scenario 모두 X\n * - boardModel prop 직접 받음 (Board entity 가 필요 없음 — 미저장 model 도 지원)\n * - dataSubscription 미연결 — 정적 시각만\n * - 사용자 입력 미수용 (selectable, dragable 모두 비활성)\n */\nimport { css, html, LitElement, type PropertyValues } from 'lit'\nimport { customElement, property, query } from 'lit/decorators.js'\n\nimport { create, SCENE_MODE } from '@hatiolab/things-scene'\n\n@customElement('ox-board-preview')\nexport class BoardPreview extends LitElement {\n static styles = [\n css`\n :host {\n display: block;\n position: relative;\n width: 100%;\n height: 100%;\n overflow: hidden;\n }\n #target {\n position: absolute;\n inset: 0;\n outline: 0;\n }\n `\n ]\n\n /**\n * 보드 모델 JSON. 변경 시 Scene 재생성 (cheap operation).\n * Object reference 가 같아도 deep 변경은 감지 못 하므로, 호출자가 새 객체로 갱신.\n */\n @property({ type: Object }) boardModel: any = null\n\n /** 사용자 클릭 / 휠 등 인터랙션 차단 (썸네일 모드용). */\n @property({ type: Boolean }) interactive: boolean = false\n\n @query('#target') private _target!: HTMLElement\n private _scene: any = null\n\n render() {\n return html`<div\n id=\"target\"\n style=${this.interactive ? '' : 'pointer-events: none;'}\n ></div>`\n }\n\n firstUpdated() {\n this._rebuild()\n }\n\n updated(changes: PropertyValues<this>) {\n if (changes.has('boardModel') && this._target) {\n this._rebuild()\n }\n }\n\n disconnectedCallback() {\n this._dispose()\n super.disconnectedCallback()\n }\n\n /** Scene 새로 만들고 target 에 부착. 기존 Scene 은 dispose. */\n private _rebuild() {\n this._dispose()\n if (!this.boardModel || !this._target) return\n try {\n this._scene = create({\n model: { ...this.boardModel },\n mode: SCENE_MODE.VIEW,\n handlers: []\n } as any)\n // 컴포넌트 모두 보이도록 fit\n this._scene.fit?.(this._scene.fitMode)\n this._scene.target = this._target\n } catch (e) {\n console.warn('[ox-board-preview] failed to render boardModel:', e)\n }\n }\n\n /** Scene release + target 분리. */\n private _dispose() {\n if (!this._scene) return\n try {\n this._scene.target = null\n this._scene.release?.()\n } catch {\n /* swallow — preview 정리는 best effort */\n }\n this._scene = null\n }\n}\n"]}
@@ -6,6 +6,7 @@ export declare class OxBoardTemplateList extends LitElement {
6
6
  static styles: import("lit").CSSResult[];
7
7
  withoutSearch: boolean;
8
8
  grist: DataGrist;
9
+ constructor();
9
10
  render(): import("lit-html").TemplateResult<1>;
10
11
  fetchHandler({ page, limit, sortings, filters }: FetchOption): Promise<{
11
12
  total: any;
@@ -39,8 +39,26 @@ const FETCH_BOARD_TEMPLATE_GQL = (id) => {
39
39
  };
40
40
  let OxBoardTemplateList = class OxBoardTemplateList extends LitElement {
41
41
  constructor() {
42
- super(...arguments);
42
+ super();
43
43
  this.withoutSearch = false;
44
+ this.addEventListener('select-record-change', async (e) => {
45
+ var _a, _b;
46
+ const { added = [], removed = [] } = e.detail;
47
+ if (added.length > 0) {
48
+ const templateId = (_a = added[0]) === null || _a === void 0 ? void 0 : _a.id;
49
+ if (!templateId)
50
+ return;
51
+ const response = await client.query({ query: FETCH_BOARD_TEMPLATE_GQL(templateId) });
52
+ let template = (_b = response.data) === null || _b === void 0 ? void 0 : _b.boardTemplate;
53
+ if (template) {
54
+ template = { ...template, model: JSON.parse(template.model) };
55
+ }
56
+ this.dispatchEvent(new CustomEvent('selection-changed', { bubbles: true, composed: true, detail: template || null }));
57
+ }
58
+ else if (removed.length > 0) {
59
+ this.dispatchEvent(new CustomEvent('selection-changed', { bubbles: true, composed: true, detail: null }));
60
+ }
61
+ });
44
62
  }
45
63
  render() {
46
64
  return html `
@@ -1 +1 @@
1
- {"version":3,"file":"ox-board-template-list.js","sourceRoot":"","sources":["../../src/ox-board-template-list.ts"],"names":[],"mappings":"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","sourcesContent":["import '@operato/data-grist'\nimport '@operato/input/ox-input-file.js'\n\nimport gql from 'graphql-tag'\nimport { css, html, LitElement } from 'lit'\nimport { customElement, property, query } from 'lit/decorators.js'\n\nimport { DataGrist, FetchOption, FilterValue, PaginationConfig, SortersConfig } from '@operato/data-grist'\nimport { buildArgs, client } from '@operato/graphql'\nimport { ScrollbarStyles } from '@operato/styles'\nimport { i18next } from '@operato/i18n'\n\nconst FETCH_BOARD_TEMPLATE_LIST_GQL = (listParam: any) => {\n return gql`\n {\n boardTemplates(${buildArgs(listParam)}) {\n items {\n id\n name\n description\n visibility\n thumbnail\n }\n total\n }\n }\n`\n}\n\nconst FETCH_BOARD_TEMPLATE_GQL = (id: string) => {\n return gql`\n {\n boardTemplate(id: \"${id}\") {\n id\n name\n description\n visibility\n model\n thumbnail\n }\n }\n`\n}\n\n@customElement('ox-board-template-list')\nexport class OxBoardTemplateList extends LitElement {\n static styles = [\n ScrollbarStyles,\n css`\n :host {\n display: flex;\n\n width: 100%;\n }\n\n ox-grist {\n flex: 1;\n overflow-y: auto;\n }\n\n #headroom {\n align-items: center;\n padding: var(--padding-default) var(--spacing-large);\n border-top: 2px solid rgba(0, 0, 0, 0.2);\n color: var(--md-sys-color-on-surface-variant);\n background-color: var(--md-sys-color-surface-variant);\n box-shadow: var(--box-shadow);\n }\n\n #filters {\n display: flex;\n flex-direction: row;\n place-content: space-between;\n margin: var(--margin-default) 0;\n }\n\n select {\n border: 0;\n outline: none;\n text-align: right;\n }\n\n @media only screen and (max-width: 460px) {\n #filters {\n flex-direction: column;\n }\n }\n `\n ]\n\n @property({ type: Boolean, attribute: 'without-search' }) withoutSearch: boolean = false\n\n @query('ox-grist') grist!: DataGrist\n\n render() {\n return html`\n <ox-grist .config=${this.gristConfig} .mode=${'CARD'} auto-fetch .fetchHandler=${this.fetchHandler.bind(this)}>\n <div slot=\"headroom\" id=\"headroom\">\n <div id=\"filters\">\n <ox-filters-form autofocus .withoutSearch=${this.withoutSearch}></ox-filters-form>\n </div>\n </div>\n </ox-grist>\n `\n }\n\n async fetchHandler({ page = 1, limit = 100, sortings = [], filters = [] }: FetchOption) {\n const { items: records, total } = await this.getBoardTemplates({ page, limit, filters, sortings })\n\n return {\n total,\n records\n }\n }\n\n get gristConfig() {\n const PRIVATE = i18next.t('label.visibility-private')\n const DOMAIN = i18next.t('label.visibility-domain')\n const PUBLIC = i18next.t('label.visibility-public')\n\n const visibilities = window.location.pathname.startsWith('/domain/')\n ? [\n { display: PRIVATE, value: 'private' },\n { display: DOMAIN, value: 'domain' },\n { display: PUBLIC, value: 'public' }\n ]\n : [\n { display: PRIVATE, value: 'private' },\n { display: PUBLIC, value: 'public' }\n ]\n\n return {\n list: {\n thumbnail: 'thumbnail',\n fields: ['name'],\n details: ['description']\n },\n columns: [\n {\n type: 'string',\n name: 'id',\n hidden: true\n },\n {\n type: 'string',\n name: 'name',\n header: 'name',\n record: {\n editable: true,\n align: 'left'\n },\n width: 200,\n filter: 'search',\n sortable: true\n },\n {\n type: 'string',\n name: 'description',\n header: 'description',\n record: {\n editable: true,\n align: 'left'\n },\n width: 200,\n filter: 'search'\n },\n {\n type: 'select-buttons',\n name: 'visibility',\n header: 'visibility',\n record: {\n editable: false\n },\n hidden: true,\n filter: {\n operator: 'in',\n options: visibilities,\n label: ''\n }\n },\n {\n type: 'image',\n name: 'thumbnail',\n hidden: true,\n record: {\n editable: false\n },\n width: 120\n }\n ],\n rows: {\n appendable: false,\n selectable: {\n multiple: false\n },\n handlers: {\n click: 'select-row-toggle'\n }\n },\n sorters: [\n {\n name: 'name',\n desc: false\n }\n ],\n pagination: {\n pages: [20, 30, 50, 100, 200]\n }\n }\n }\n\n async firstUpdated() {\n this.refreshBoardTemplates()\n }\n\n async getSelected() {\n const selected = this.grist.selected\n const templateId = selected && selected[0]?.id\n\n if (!templateId) {\n return\n }\n\n var boardTemplateResponse = await client.query({\n query: FETCH_BOARD_TEMPLATE_GQL(templateId)\n })\n\n var template = boardTemplateResponse.data?.boardTemplate\n\n if (template) {\n template = {\n ...template,\n model: JSON.parse(template.model)\n }\n }\n\n return template\n }\n\n async refreshBoardTemplates() {\n this.grist.fetch()\n }\n\n async getBoardTemplates({\n page = 1,\n limit = 30,\n filters = [],\n sortings = []\n }: { page?: number; limit?: number; filters?: FilterValue[]; sortings?: SortersConfig } = {}) {\n var pagination: PaginationConfig = {\n limit,\n page\n }\n\n var params = {\n filters,\n sortings,\n pagination\n }\n\n var boardTemplateListResponse = await client.query({\n query: FETCH_BOARD_TEMPLATE_LIST_GQL(params)\n })\n\n return boardTemplateListResponse?.data?.boardTemplates || {}\n }\n}\n"]}
1
+ {"version":3,"file":"ox-board-template-list.js","sourceRoot":"","sources":["../../src/ox-board-template-list.ts"],"names":[],"mappings":"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","sourcesContent":["import '@operato/data-grist'\nimport '@operato/input/ox-input-file.js'\n\nimport gql from 'graphql-tag'\nimport { css, html, LitElement } from 'lit'\nimport { customElement, property, query } from 'lit/decorators.js'\n\nimport { DataGrist, FetchOption, FilterValue, PaginationConfig, SortersConfig } from '@operato/data-grist'\nimport { buildArgs, client } from '@operato/graphql'\nimport { ScrollbarStyles } from '@operato/styles'\nimport { i18next } from '@operato/i18n'\n\nconst FETCH_BOARD_TEMPLATE_LIST_GQL = (listParam: any) => {\n return gql`\n {\n boardTemplates(${buildArgs(listParam)}) {\n items {\n id\n name\n description\n visibility\n thumbnail\n }\n total\n }\n }\n`\n}\n\nconst FETCH_BOARD_TEMPLATE_GQL = (id: string) => {\n return gql`\n {\n boardTemplate(id: \"${id}\") {\n id\n name\n description\n visibility\n model\n thumbnail\n }\n }\n`\n}\n\n@customElement('ox-board-template-list')\nexport class OxBoardTemplateList extends LitElement {\n static styles = [\n ScrollbarStyles,\n css`\n :host {\n display: flex;\n\n width: 100%;\n }\n\n ox-grist {\n flex: 1;\n overflow-y: auto;\n }\n\n #headroom {\n align-items: center;\n padding: var(--padding-default) var(--spacing-large);\n border-top: 2px solid rgba(0, 0, 0, 0.2);\n color: var(--md-sys-color-on-surface-variant);\n background-color: var(--md-sys-color-surface-variant);\n box-shadow: var(--box-shadow);\n }\n\n #filters {\n display: flex;\n flex-direction: row;\n place-content: space-between;\n margin: var(--margin-default) 0;\n }\n\n select {\n border: 0;\n outline: none;\n text-align: right;\n }\n\n @media only screen and (max-width: 460px) {\n #filters {\n flex-direction: column;\n }\n }\n `\n ]\n\n @property({ type: Boolean, attribute: 'without-search' }) withoutSearch: boolean = false\n\n @query('ox-grist') grist!: DataGrist\n\n constructor() {\n super()\n this.addEventListener('select-record-change', async (e: Event) => {\n const { added = [], removed = [] } = (e as CustomEvent).detail as { added?: any[]; removed?: any[] }\n if (added.length > 0) {\n const templateId = added[0]?.id\n if (!templateId) return\n const response = await client.query({ query: FETCH_BOARD_TEMPLATE_GQL(templateId) })\n let template = response.data?.boardTemplate\n if (template) {\n template = { ...template, model: JSON.parse(template.model) }\n }\n this.dispatchEvent(new CustomEvent('selection-changed', { bubbles: true, composed: true, detail: template || null }))\n } else if (removed.length > 0) {\n this.dispatchEvent(new CustomEvent('selection-changed', { bubbles: true, composed: true, detail: null }))\n }\n })\n }\n\n render() {\n return html`\n <ox-grist .config=${this.gristConfig} .mode=${'CARD'} auto-fetch .fetchHandler=${this.fetchHandler.bind(this)}>\n <div slot=\"headroom\" id=\"headroom\">\n <div id=\"filters\">\n <ox-filters-form autofocus .withoutSearch=${this.withoutSearch}></ox-filters-form>\n </div>\n </div>\n </ox-grist>\n `\n }\n\n async fetchHandler({ page = 1, limit = 100, sortings = [], filters = [] }: FetchOption) {\n const { items: records, total } = await this.getBoardTemplates({ page, limit, filters, sortings })\n\n return {\n total,\n records\n }\n }\n\n get gristConfig() {\n const PRIVATE = i18next.t('label.visibility-private')\n const DOMAIN = i18next.t('label.visibility-domain')\n const PUBLIC = i18next.t('label.visibility-public')\n\n const visibilities = window.location.pathname.startsWith('/domain/')\n ? [\n { display: PRIVATE, value: 'private' },\n { display: DOMAIN, value: 'domain' },\n { display: PUBLIC, value: 'public' }\n ]\n : [\n { display: PRIVATE, value: 'private' },\n { display: PUBLIC, value: 'public' }\n ]\n\n return {\n list: {\n thumbnail: 'thumbnail',\n fields: ['name'],\n details: ['description']\n },\n columns: [\n {\n type: 'string',\n name: 'id',\n hidden: true\n },\n {\n type: 'string',\n name: 'name',\n header: 'name',\n record: {\n editable: true,\n align: 'left'\n },\n width: 200,\n filter: 'search',\n sortable: true\n },\n {\n type: 'string',\n name: 'description',\n header: 'description',\n record: {\n editable: true,\n align: 'left'\n },\n width: 200,\n filter: 'search'\n },\n {\n type: 'select-buttons',\n name: 'visibility',\n header: 'visibility',\n record: {\n editable: false\n },\n hidden: true,\n filter: {\n operator: 'in',\n options: visibilities,\n label: ''\n }\n },\n {\n type: 'image',\n name: 'thumbnail',\n hidden: true,\n record: {\n editable: false\n },\n width: 120\n }\n ],\n rows: {\n appendable: false,\n selectable: {\n multiple: false\n },\n handlers: {\n click: 'select-row-toggle'\n }\n },\n sorters: [\n {\n name: 'name',\n desc: false\n }\n ],\n pagination: {\n pages: [20, 30, 50, 100, 200]\n }\n }\n }\n\n async firstUpdated() {\n this.refreshBoardTemplates()\n }\n\n async getSelected() {\n const selected = this.grist.selected\n const templateId = selected && selected[0]?.id\n\n if (!templateId) {\n return\n }\n\n var boardTemplateResponse = await client.query({\n query: FETCH_BOARD_TEMPLATE_GQL(templateId)\n })\n\n var template = boardTemplateResponse.data?.boardTemplate\n\n if (template) {\n template = {\n ...template,\n model: JSON.parse(template.model)\n }\n }\n\n return template\n }\n\n async refreshBoardTemplates() {\n this.grist.fetch()\n }\n\n async getBoardTemplates({\n page = 1,\n limit = 30,\n filters = [],\n sortings = []\n }: { page?: number; limit?: number; filters?: FilterValue[]; sortings?: SortersConfig } = {}) {\n var pagination: PaginationConfig = {\n limit,\n page\n }\n\n var params = {\n filters,\n sortings,\n pagination\n }\n\n var boardTemplateListResponse = await client.query({\n query: FETCH_BOARD_TEMPLATE_LIST_GQL(params)\n })\n\n return boardTemplateListResponse?.data?.boardTemplates || {}\n }\n}\n"]}