@operato/board 8.0.0-alpha.7 → 8.0.0-alpha.9

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,23 @@
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
+ ## [8.0.0-alpha.9](https://github.com/hatiolab/operato/compare/v8.0.0-alpha.8...v8.0.0-alpha.9) (2024-09-15)
7
+
8
+ **Note:** Version bump only for package @operato/board
9
+
10
+
11
+
12
+
13
+
14
+ ## [8.0.0-alpha.8](https://github.com/hatiolab/operato/compare/v8.0.0-alpha.7...v8.0.0-alpha.8) (2024-09-15)
15
+
16
+
17
+ ### :bug: Bug Fix
18
+
19
+ * template viewer for board and process ([535433c](https://github.com/hatiolab/operato/commit/535433c7227190714a14cc0d1089a99e1075a381))
20
+
21
+
22
+
6
23
  ## [8.0.0-alpha.7](https://github.com/hatiolab/operato/compare/v8.0.0-alpha.6...v8.0.0-alpha.7) (2024-09-14)
7
24
 
8
25
 
@@ -5,3 +5,4 @@ export * from './ox-board-viewer.js';
5
5
  export * from './ox-board-player.js';
6
6
  export * from './ox-board-modeller.js';
7
7
  export * from './ox-board-list.js';
8
+ export * from './ox-board-template-viewer.js';
package/dist/src/index.js CHANGED
@@ -5,4 +5,5 @@ export * from './ox-board-viewer.js';
5
5
  export * from './ox-board-player.js';
6
6
  export * from './ox-board-modeller.js';
7
7
  export * from './ox-board-list.js';
8
+ export * from './ox-board-template-viewer.js';
8
9
  //# sourceMappingURL=index.js.map
@@ -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;AAElF,cAAc,sBAAsB,CAAA;AACpC,cAAc,sBAAsB,CAAA;AACpC,cAAc,wBAAwB,CAAA;AACtC,cAAc,oBAAoB,CAAA","sourcesContent":["export * from './types.js'\n\nexport { registerDefaultGroups } from './component/register-default-groups.js'\nexport { BoardDataStorage, PlaylistStorage } from './data-storage/data-storage.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'\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;AAElF,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'\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"]}
@@ -0,0 +1,27 @@
1
+ import '@material/web/icon/icon.js';
2
+ import '@material/web/fab/fab.js';
3
+ import { LitElement, PropertyValues } from 'lit';
4
+ export declare class BoardTemplateViewer extends LitElement {
5
+ static styles: import("lit").CSSResult[];
6
+ boardTemplate: any;
7
+ _scene: any;
8
+ _target: HTMLElement;
9
+ render(): import("lit-html").TemplateResult<1>;
10
+ private resizeHandler;
11
+ connectedCallback(): void;
12
+ disconnectedCallback(): void;
13
+ updated(changes: PropertyValues<this>): void;
14
+ initScene(): void;
15
+ closeScene(): void;
16
+ releaseScene(): void;
17
+ setupScene({ id, scene }: {
18
+ id: string;
19
+ scene: any;
20
+ }): void;
21
+ getSceneImageData(base64?: boolean): Promise<{
22
+ width: any;
23
+ height: any;
24
+ data: string | Uint8ClampedArray;
25
+ } | undefined>;
26
+ printTrick(image: string): Promise<void>;
27
+ }
@@ -0,0 +1,178 @@
1
+ import { __decorate } from "tslib";
2
+ import '@material/web/icon/icon.js';
3
+ import '@material/web/fab/fab.js';
4
+ import { css, html, LitElement } from 'lit';
5
+ import { customElement, property, query, state } from 'lit/decorators.js';
6
+ import { create, SCENE_MODE } from '@hatiolab/things-scene';
7
+ import { ScrollbarStyles } from '@operato/styles';
8
+ let BoardTemplateViewer = class BoardTemplateViewer extends LitElement {
9
+ constructor() {
10
+ super(...arguments);
11
+ this.boardTemplate = {};
12
+ this._scene = null;
13
+ this.resizeHandler = () => {
14
+ this._scene && this._scene.fit();
15
+ };
16
+ }
17
+ render() {
18
+ return html ` <div id="target"></div> `;
19
+ }
20
+ connectedCallback() {
21
+ super.connectedCallback();
22
+ window.addEventListener('resize', this.resizeHandler);
23
+ window.addEventListener('orientationchange', this.resizeHandler);
24
+ }
25
+ disconnectedCallback() {
26
+ window.removeEventListener('resize', this.resizeHandler);
27
+ window.removeEventListener('orientationchange', this.resizeHandler);
28
+ super.disconnectedCallback();
29
+ this.closeScene();
30
+ }
31
+ updated(changes) {
32
+ if (changes.has('boardTemplate')) {
33
+ this.closeScene();
34
+ if (this.boardTemplate && this.boardTemplate.id) {
35
+ this.initScene();
36
+ }
37
+ }
38
+ }
39
+ initScene() {
40
+ if (!this.boardTemplate || !this.boardTemplate.id)
41
+ return;
42
+ this._scene = create({
43
+ model: {
44
+ ...this.boardTemplate.model
45
+ },
46
+ mode: SCENE_MODE.VIEW
47
+ });
48
+ this.setupScene({ id: this.boardTemplate.id, scene: this._scene });
49
+ }
50
+ closeScene() {
51
+ if (this._scene) {
52
+ this._scene.target = null;
53
+ this._scene.release && this._scene.release();
54
+ this._scene = null;
55
+ }
56
+ }
57
+ releaseScene() {
58
+ this.closeScene();
59
+ }
60
+ setupScene({ id, scene }) {
61
+ var _a;
62
+ this._scene = scene;
63
+ const backgroundColor = (_a = this._scene) === null || _a === void 0 ? void 0 : _a.root.state.fillStyle;
64
+ if (typeof backgroundColor === 'string') {
65
+ this.style.backgroundColor = backgroundColor;
66
+ }
67
+ else {
68
+ this.style.backgroundColor = 'initial';
69
+ }
70
+ /* scene의 기존 target을 보관한다. */
71
+ this._scene.fit(this._scene.fitMode);
72
+ this._scene.target = this._target;
73
+ }
74
+ /* event handlers */
75
+ async getSceneImageData(base64 = false) {
76
+ if (!this._scene) {
77
+ return;
78
+ }
79
+ var { width, height } = this._scene.model;
80
+ var pixelRatio = window.devicePixelRatio;
81
+ // 1. Scene의 바운드에 근거하여, 오프스크린 캔바스를 만든다.
82
+ var canvas = document.createElement('canvas');
83
+ canvas.width = Number(width);
84
+ canvas.height = Number(height);
85
+ var root = this._scene.root;
86
+ // 2. 모델레이어의 원래 위치와 스케일을 저장한다.
87
+ var translate = root.get('translate');
88
+ var scale = root.get('scale');
89
+ // 3. 위치와 스케일 기본 설정.
90
+ root.set('translate', { x: 0, y: 0 });
91
+ root.set('scale', { x: 1 / pixelRatio, y: 1 / pixelRatio });
92
+ // 4. 오프스크린 캔바스의 Context2D를 구한뒤, 모델레이어를 그 위에 그린다.
93
+ var context = canvas.getContext('2d');
94
+ root.draw(context);
95
+ root.set('translate', translate);
96
+ root.set('scale', scale);
97
+ var data = base64 ? canvas.toDataURL() : context.getImageData(0, 0, width, height).data;
98
+ return {
99
+ width,
100
+ height,
101
+ data
102
+ };
103
+ }
104
+ async printTrick(image) {
105
+ var _a;
106
+ var viewTarget = null;
107
+ var printTarget = null;
108
+ if (!image) {
109
+ image = (_a = (await this.getSceneImageData(true))) === null || _a === void 0 ? void 0 : _a.data;
110
+ }
111
+ printTarget = document.createElement('img');
112
+ printTarget.id = 'target';
113
+ printTarget.src = image;
114
+ printTarget.style.width = '100%';
115
+ printTarget.style.height = '100%';
116
+ const x = (mql) => {
117
+ if (mql.matches) {
118
+ if (!viewTarget) {
119
+ viewTarget = this.renderRoot.getElementById('target');
120
+ this.renderRoot.replaceChild(printTarget, viewTarget);
121
+ }
122
+ }
123
+ else {
124
+ this.renderRoot.replaceChild(viewTarget, printTarget);
125
+ printTarget.remove();
126
+ mediaQueryList.removeEventListener('change', x);
127
+ }
128
+ };
129
+ if (window.matchMedia) {
130
+ var mediaQueryList = window.matchMedia('print');
131
+ mediaQueryList.addEventListener('change', x);
132
+ }
133
+ }
134
+ };
135
+ BoardTemplateViewer.styles = [
136
+ ScrollbarStyles,
137
+ css `
138
+ :host {
139
+ display: flex;
140
+ flex-direction: column;
141
+
142
+ position: relative;
143
+
144
+ width: 100%; /* 전체화면보기를 위해서 필요함. */
145
+ overflow: hidden;
146
+ }
147
+
148
+ #target {
149
+ flex: 1;
150
+
151
+ width: 100%; /* 전체화면보기를 위해서 필요함. */
152
+ height: 100%;
153
+
154
+ outline: 0;
155
+ }
156
+
157
+ @media print {
158
+ md-fab,
159
+ md-icon {
160
+ display: none;
161
+ }
162
+ }
163
+ `
164
+ ];
165
+ __decorate([
166
+ property({ type: Object })
167
+ ], BoardTemplateViewer.prototype, "boardTemplate", void 0);
168
+ __decorate([
169
+ state()
170
+ ], BoardTemplateViewer.prototype, "_scene", void 0);
171
+ __decorate([
172
+ query('#target')
173
+ ], BoardTemplateViewer.prototype, "_target", void 0);
174
+ BoardTemplateViewer = __decorate([
175
+ customElement('ox-board-template-viewer')
176
+ ], BoardTemplateViewer);
177
+ export { BoardTemplateViewer };
178
+ //# sourceMappingURL=ox-board-template-viewer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ox-board-template-viewer.js","sourceRoot":"","sources":["../../src/ox-board-template-viewer.ts"],"names":[],"mappings":"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","sourcesContent":["import '@material/web/icon/icon.js'\nimport '@material/web/fab/fab.js'\n\nimport { css, html, LitElement, PropertyValues } from 'lit'\nimport { customElement, property, query, state } from 'lit/decorators.js'\n\nimport { create, SCENE_MODE } from '@hatiolab/things-scene'\nimport { ScrollbarStyles } from '@operato/styles'\n\n@customElement('ox-board-template-viewer')\nexport class BoardTemplateViewer extends LitElement {\n static styles = [\n ScrollbarStyles,\n css`\n :host {\n display: flex;\n flex-direction: column;\n\n position: relative;\n\n width: 100%; /* 전체화면보기를 위해서 필요함. */\n overflow: hidden;\n }\n\n #target {\n flex: 1;\n\n width: 100%; /* 전체화면보기를 위해서 필요함. */\n height: 100%;\n\n outline: 0;\n }\n\n @media print {\n md-fab,\n md-icon {\n display: none;\n }\n }\n `\n ]\n\n @property({ type: Object }) boardTemplate: any = {}\n\n @state() _scene: any = null\n\n @query('#target') _target!: HTMLElement\n\n render() {\n return html` <div id=\"target\"></div> `\n }\n\n private resizeHandler = () => {\n this._scene && this._scene.fit()\n }\n\n connectedCallback() {\n super.connectedCallback()\n\n window.addEventListener('resize', this.resizeHandler)\n\n window.addEventListener('orientationchange', this.resizeHandler)\n }\n\n disconnectedCallback() {\n window.removeEventListener('resize', this.resizeHandler)\n window.removeEventListener('orientationchange', this.resizeHandler)\n\n super.disconnectedCallback()\n\n this.closeScene()\n }\n\n updated(changes: PropertyValues<this>) {\n if (changes.has('boardTemplate')) {\n this.closeScene()\n\n if (this.boardTemplate && this.boardTemplate.id) {\n this.initScene()\n }\n }\n }\n\n initScene() {\n if (!this.boardTemplate || !this.boardTemplate.id) return\n\n this._scene = create({\n model: {\n ...this.boardTemplate.model\n },\n mode: SCENE_MODE.VIEW\n })\n\n this.setupScene({ id: this.boardTemplate.id, scene: this._scene })\n }\n\n closeScene() {\n if (this._scene) {\n this._scene.target = null\n this._scene.release && this._scene.release()\n\n this._scene = null\n }\n }\n\n releaseScene() {\n this.closeScene()\n }\n\n setupScene({ id, scene }: { id: string; scene: any }) {\n this._scene = scene\n\n const backgroundColor = this._scene?.root.state.fillStyle\n if (typeof backgroundColor === 'string') {\n this.style.backgroundColor = backgroundColor\n } else {\n this.style.backgroundColor = 'initial'\n }\n\n /* scene의 기존 target을 보관한다. */\n\n this._scene.fit(this._scene.fitMode)\n this._scene.target = this._target\n }\n\n /* event handlers */\n async getSceneImageData(base64 = false) {\n if (!this._scene) {\n return\n }\n\n var { width, height } = this._scene.model\n var pixelRatio = window.devicePixelRatio\n\n // 1. Scene의 바운드에 근거하여, 오프스크린 캔바스를 만든다.\n var canvas = document.createElement('canvas')\n canvas.width = Number(width)\n canvas.height = Number(height)\n\n var root = this._scene.root\n // 2. 모델레이어의 원래 위치와 스케일을 저장한다.\n var translate = root.get('translate')\n var scale = root.get('scale')\n\n // 3. 위치와 스케일 기본 설정.\n root.set('translate', { x: 0, y: 0 })\n root.set('scale', { x: 1 / pixelRatio, y: 1 / pixelRatio })\n\n // 4. 오프스크린 캔바스의 Context2D를 구한뒤, 모델레이어를 그 위에 그린다.\n var context = canvas.getContext('2d')\n\n root.draw(context)\n\n root.set('translate', translate)\n root.set('scale', scale)\n\n var data = base64 ? canvas.toDataURL() : context!.getImageData(0, 0, width, height).data\n\n return {\n width,\n height,\n data\n }\n }\n\n async printTrick(image: string) {\n var viewTarget: HTMLElement | null = null\n var printTarget: HTMLImageElement | null = null\n\n if (!image) {\n image = (await this.getSceneImageData(true))?.data as string\n }\n\n printTarget = document.createElement('img')\n printTarget.id = 'target'\n printTarget.src = image\n printTarget.style.width = '100%'\n printTarget.style.height = '100%'\n\n const x = (mql: MediaQueryListEvent) => {\n if (mql.matches) {\n if (!viewTarget) {\n viewTarget = (this.renderRoot as ShadowRoot)!.getElementById('target')\n this.renderRoot.replaceChild(printTarget!, viewTarget!)\n }\n } else {\n this.renderRoot.replaceChild(viewTarget!, printTarget!)\n printTarget!.remove()\n mediaQueryList.removeEventListener('change', x)\n }\n }\n\n if (window.matchMedia) {\n var mediaQueryList = window.matchMedia('print')\n mediaQueryList.addEventListener('change', x)\n }\n }\n}\n"]}