@lumiscaphe/ng-viewer 14.0.0 → 14.2.0

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/README.md CHANGED
@@ -7,203 +7,221 @@ Official Angular component for Lumiscaphe 3D Viewer
7
7
  ```
8
8
  $ npm i @lumiscaphe/ng-viewer
9
9
  ```
10
+
10
11
  or
12
+
11
13
  ```
12
14
  $ yarn add @lumiscaphe/ng-viewer
13
15
  ```
14
16
 
15
- ## The basics
17
+ ## Usage
16
18
 
17
- Import Lumiscaphe Viewer module in your module:
19
+ ### Import
18
20
 
19
21
  ```javascript
20
22
  import { NgViewerModule } from '@lumiscaphe/ng-viewer';
23
+ ```
24
+
25
+ ### Basic
21
26
 
22
- @NgModule({
23
- ...,
24
- imports: [
25
- ...,
26
- NgViewerModule
27
- ],
28
- ...
29
- })
27
+ ```html
28
+ <NgViewer [server]="server" [scene]="scene" [view]="view"></NgViewer>
30
29
  ```
31
30
 
32
- Prepare Lumiscaphe Viewer data in your component:
33
31
  ```javascript
34
32
  server = 'https://wr.lumiscaphe.com';
35
33
 
36
- scene = [{
37
- database: 'ee294840-5689-49b0-9edb-527598602df0',
38
- configuration: 'Bin.Blue/Cabin.Yellow/Style.Design1/Wheels.Red',
39
- animations: ['LeftDoor'],
40
- }];
34
+ scene = [
35
+ {
36
+ database: 'ee294840-5689-49b0-9edb-527598602df0',
37
+ configuration: 'Bin.Blue/Cabin.Yellow/Style.Design1/Wheels.Red'
38
+ }
39
+ ];
41
40
 
42
41
  view = {
43
42
  mode: 'image',
44
- camera: 'EXTER/1',
45
- background: 'product',
43
+ camera: 'EXTER/1'
46
44
  };
45
+ ```
47
46
 
48
- // Optional
49
- transition = {
50
- fromPosition: 0,
51
- toPosition: 2,
52
- animation: {
53
- name: 'Porte Gauche',
54
- camera: '/Camera_Ext',
55
- duration: 1,
56
- reverse: false,
57
- },
58
- };
47
+ ### Hotspot
59
48
 
60
- // Optional - default values
61
- encoder = {
62
- format: 'jpeg',
63
- quality: 80,
64
- };
49
+ Hotspots are interactive elements that can be selected to engage in additional content or media.\
50
+ Hotspots are overlaid on top of the viewer to provide additional media for informational purposes.
65
51
 
66
- // Optional - default values
67
- parameters = {
68
- antialiasing: false,
69
- superSampling: 2,
70
- };
52
+ ```html
53
+ <NgViewer ... [hotspots]="hotspots">
54
+ <ng-template #hotspotTemplate let-hotspot let-index="index">
55
+ <ng-container *ngIf="hotspot.visibility === 'Visible'">
56
+ <button>{{ index + 1 }}</button>
57
+ </ng-container>
58
+ </ng-template>
59
+ </NgViewer>
60
+ ```
61
+
62
+ ```typescript
63
+ hotspots = [{ x: 0, y: 0, z: 0 }];
64
+ ```
65
+
66
+ ### Pick
71
67
 
72
- // Optional - default values
73
- onLoadStart() {
74
- }
68
+ Pick allow to retrieve information of a 3D point from a 2D point.
75
69
 
76
- onLoadProgress() {
77
- }
70
+ ```html
71
+ <NgViewer ...></NgViewer>
78
72
 
79
- onLoadEnd() {
80
- }
73
+ <button type="button" (click)="pick()">Pick</button>
74
+ ```
81
75
 
82
- onLoadError() {
83
- }
76
+ ```typescript
77
+ import { NgViewerComponent } from '@lumiscaphe/ng-viewer';
84
78
 
85
- onVrcubeInteraction() {
86
- }
79
+ ...
87
80
 
88
- onVrobjectInteraction() {
89
- }
81
+ @ViewChild(NgViewerComponent) viewer: NgViewerComponent;
82
+ @ViewChild(ngViewerComponent, { read: ElementRef }) viewerElement: ElementRef<HTMLDivElement>;
90
83
 
91
- ```
84
+ ...
92
85
 
93
- Add Lumiscaphe Viewer component in your HTML:
94
- ```html
95
- <NgViewer [server]="server" [scene]="scene" [view]="view" [parameters]="parameters" [encoder]="encoder" (onLoadStart)="onLoadStart()" (onLoadProgress)="onLoadProgress()" (onLoadEnd)="onLoadEnd()" (onLoadError)="onLoadError()"></NgViewer>
86
+ pick(): void {
87
+ this.viewerElement.nativeElement.addEventListener('click', e => {
88
+ const position = { x: e.clientX, y: e.clientY };
89
+ this.viewer.pick(position).then(pickResult => console.log(pickResult))
90
+ }, { once: true });
91
+ };
96
92
  ```
97
93
 
98
- ## Scene
94
+ ## API
99
95
 
100
- A scene is an array of products defined with:
101
- * `database` : product 3D model guid string
102
- * `configuration` : product configuration as a string of concatenated values separated by a slash
103
- * `animations` : product animations as an array of string
96
+ ### Properties
104
97
 
105
- Basic scenes should contain only one product.
98
+ | Name | Type | Default | Description |
99
+ | -------------- | ------------ | ------------------------------------------- | -------------------------------- |
100
+ | **server** | `string` | `undefined` | WebRender URL to use. |
101
+ | **scene** | `Scene` | `undefined` | Scene to render. |
102
+ | **view** | `View` | `undefined` | View to render. |
103
+ | **api** | `'v1'\|'v2'` | `'v1'` | API version to use. |
104
+ | **encoder** | `Encoder` | `{ format: 'jpeg', quality: 80 }` | Image encoder to use. |
105
+ | **parameters** | `Parameters` | `{ antialiasing: false, superSampling: 2 }` | Rendering parameters to use. |
106
+ | **hotspots** | `Vector3D[]` | `undefined` | Hotspot array to display. |
107
+ | **vrcube** | `VRCube` | `undefined` | VRCube point of view to display. |
108
+ | **vrobject** | `VRObject` | `undefined` | VRObject position to display. |
106
109
 
107
- ## Decor
110
+ ### Emitters
108
111
 
109
- Two ways to drive scene decor.
112
+ | Name | Parameters | Description |
113
+ | ------------------------- | -------------------- | ------------------------------------------------------ |
114
+ | **onLoadStart** | `progress: number` | Callback to invoke when load starts. |
115
+ | **onLoadProgress** | `progress: number` | Callback to invoke when load makes progress. |
116
+ | **onLoadEnd** | `progress: number` | Callback to invoke when load ends. |
117
+ | **onLoadError** | `Error` | Callback to invoke when load failed. |
118
+ | **onInteraction** | `VRCube \| VRObject` | Callback to invoke when interaction happened. |
119
+ | **onVrcubeInteraction** | `VRCube` | Callback to invoke when VRCube interaction happened. |
120
+ | **onVrobjectInteraction** | `VRObject` | Callback to invoke when VRObject interaction happened. |
110
121
 
111
- **Integrated**
122
+ ### Methods
112
123
 
113
- When a 3D model contains integrated decors, it can be change through product configuration. For example: add `ENV.STUDIO` or `ENV.ALPES` to product configuration string.
124
+ | Name | Parameters | Description | Returns |
125
+ | ------------ | --------------------------------- | ------------------------------------ | -------------------- |
126
+ | **pick** | `position: Position` | Get the 3D point from a 2D position. | `PickResult` |
127
+ | **snapshot** | `type: string`, `quality: number` | Get a snapshot of the viewer. | `string` - Image URL |
114
128
 
115
- ```javascript
116
- const scene = [{
117
- database: 'ee294840-5689-49b0-9edb-527598602df0',
118
- configuration: 'Bin.Blue/Cabin.Yellow/Style.Design1/Wheels.Red/ENV.ALPES',
119
- animations: ['LeftDoor'],
120
- }];
121
- ```
129
+ ### Templates
122
130
 
123
- **External**
131
+ | Name | Parameters | Description |
132
+ | ------------------- | ------------------------------------------------ | ------------------------ |
133
+ | **hotspotTemplate** | `context: { $implicit: Hotspot, index: number }` | Custom hotspot template. |
124
134
 
125
- When a 3D model does not contain integrated decors, it can be added with an external 3D model. For example:
135
+ ### Interfaces
126
136
 
127
- ```javascript
128
- const decorProduct = {
129
- database: '82c1f7e8-9ae4-4f00-b45c-c857e21a954f',
130
- translation: { x: 0, y: -0.12, z: 0 },
131
- };
137
+ #### Scene
132
138
 
133
- const product = {
134
- database: 'ee294840-5689-49b0-9edb-527598602df0',
135
- configuration: 'Bin.Blue/Cabin.Yellow/Style.Design1/Wheels.Red',
136
- animations: ['LeftDoor'],
137
- };
139
+ | Name | Type | Description |
140
+ | ----------------- | ---------- | -------------------------------------- |
141
+ | **database** | `string` | Database id of the 3D model. |
142
+ | **configuration** | `string` | Product configuration of the 3D model. |
143
+ | **animations** | `string[]` | Animations of the 3D model |
138
144
 
139
- const scene = [decorProduct, product];
140
- ```
145
+ #### View
146
+
147
+ | Name | Type | Description |
148
+ | ---------- | ------------------------------- | ---------------------------------------- |
149
+ | **mode** | `'image'\|'vrcube'\|'vrobject'` | Mode of the view. |
150
+ | **camera** | `string` | Camera or camera group name of the view. |
151
+
152
+ #### Encoder
153
+
154
+ | Name | Type | Description |
155
+ | ----------- | -------- | --------------------------------- |
156
+ | **format** | `'jpeg'` | Format of the encoder. |
157
+ | **quality** | `number` | Quality of the encoder (0 - 100). |
141
158
 
142
- Decor product is defined with:
143
- * `database` : decor product 3D model guid string
144
- * `translation` : decor product 3D position (e.g. decorDeltaAltitude)
159
+ #### Parameters
145
160
 
146
- Note: decor product should always be the first product in the scene.
161
+ | Name | Type | Description |
162
+ | ----------------- | ------ | ---------------------------------------------- |
163
+ | **superSampling** | number | Super sampling to use for antialiasing (1 -2). |
147
164
 
148
- ## View
165
+ #### Hotspot
149
166
 
150
- A view can be defined in the following modes:
167
+ | Name | Type | Description |
168
+ | ------------ | ----------- | --------------------------------------------------------------- |
169
+ | `id` | `string` | Id of the hotspot (when hotspot is provided by the 3D model). |
170
+ | `name` | `string` | Name of the hotspot (when hotspot is provided by the 3D model). |
171
+ | `position2D` | `Vector2D` | 2D position of the hotspot in the viewer. |
172
+ | `position3D` | `Vector3D` | 3D position of the hotspot. |
173
+ | `visibility` | `Visiblity` | Visiblity of the hotspot. |
151
174
 
152
- **Image**
153
- * `mode` : `image`
154
- * `camera` : camera path
155
- * `background` : view background mode (`product`, `transparent`, `gradient`)
175
+ #### VRCube
156
176
 
157
- **VRCube**
158
- * `mode` : `vrcube`
159
- * `camera` : camera path
160
- * `background` : view background mode (`product`, `transparent`, `gradient`)
177
+ | Name | Type | Description |
178
+ | ---- | ------------- | ---------------------------------------------- |
179
+ | pov | `PointOfView` | Point of view of the VRCube widget. |
180
+ | fov | `number` | Field of view in degrees of the VRCube widget. |
161
181
 
162
- **VRObject - animation**
163
- * `mode` : `vrobject`
164
- * `animation` : animation name
165
- * `camera` : camera path
166
- * `frames` : number of frames
167
- * `loop` : whether animation loops or not
168
- * `background` : view background mode (`product`, `transparent`, `gradient`)
182
+ #### VRObject
169
183
 
170
- **VRObject - bookmark set**
171
- * `mode` : `vrobject`
172
- * `camera` : camera group path
173
- * `background` : view background mode (`product`, `transparent`, `gradient`)
184
+ | Name | Type | Description |
185
+ | ---------- | -------- | ---------------------------------- |
186
+ | `position` | `number` | Image inde of the VRObject widget. |
174
187
 
175
- ## Transition
188
+ #### PickResult
176
189
 
177
- A transition can be set when loading a scene or setting a view with:
178
- * `fromPosition` : go to specific vrobject image at the beginning - optional
179
- * `toPosition` : go to specific vrobject image at the end - optional
180
- * `animation` :
181
- * `name`: animation name
182
- * `camera`: camera path to use for animation
183
- * `duration`: animation duration in seconds
184
- * `reverse`: whether to reverse animation direction
190
+ | Name | Type | Description |
191
+ | ----------- | ---------- | --------------------------------------- |
192
+ | **product** | `string` | Product id of the picked point. |
193
+ | **surface** | `Surface` | Surface of the picked point. |
194
+ | **normal** | `Vector3D` | Normal coordinates of the picked point. |
195
+ | **point** | `Vector3D` | Point coordinates of the picked point. |
196
+ | **uvw** | `UVW` | UV coordinates of the picked point. |
185
197
 
186
- ## Encoder
198
+ #### Surface
199
+
200
+ | Name | Type | Description |
201
+ | -------- | ---------- | ------------------------ |
202
+ | **id** | `string` | Id of the surface. |
203
+ | **name** | `string` | Name of the surface. |
204
+ | **tags** | `string[]` | Tag list of the surface. |
187
205
 
188
- An encoder is defined with:
189
- * `format` : encoder format (`jpeg`, `png`, `webp`)
190
- * `quality` : encoder quality (0-100 for `jpeg` and `webp` and 0-9 for `png`)
206
+ #### UVW
191
207
 
192
- ## Parameters
208
+ | Name | Type | Description |
209
+ | ----- | -------- | ------------- |
210
+ | **u** | `number` | U coordinate. |
211
+ | **v** | `number` | V coordinate. |
212
+ | **w** | `number` | W coordinate. |
193
213
 
194
- Render parameters are defined with:
195
- * `antialiasing` : whether software antialiasing is enabled or not
196
- * `superSampling` : super sampling coefficient 1-4
214
+ #### Vector2D
197
215
 
198
- WebRender is always optimized for the following default values (false, 2). Change this with precaution !
216
+ | Name | Type | Description |
217
+ | ----- | -------- | ------------- |
218
+ | **x** | `number` | X coordinate. |
219
+ | **y** | `number` | Y coordinate. |
199
220
 
200
- ## Events
221
+ #### Vector3D
201
222
 
202
- Events are defined with:
203
- * `onLoadStart` : event sent on load start
204
- * `onLoadProgress` : event sent on load progress
205
- * `onLoadEnd` : event sent on load end
206
- * `onLoadError` : event sent on load error
207
- * `onInteraction` : event sent on vrcube or vrobject interaction
208
- * `onVrcubeInteraction` : event sent on vrcube interaction
209
- * `onVrobjectInteraction` : event sent on vrobject interaction
223
+ | Name | Type | Description |
224
+ | ----- | -------- | ------------- |
225
+ | **x** | `number` | X coordinate. |
226
+ | **y** | `number` | Y coordinate. |
227
+ | **z** | `number` | Z coordinate. |
@@ -1,12 +1,10 @@
1
- import { Component, EventEmitter, Input, ViewChild, Output, HostListener } from '@angular/core';
1
+ import { Component, ContentChild, EventEmitter, Input, Output, ViewChild } from '@angular/core';
2
2
  import { Viewer } from '@lumiscaphe/viewer';
3
- import Viewport from './viewport';
4
3
  import * as i0 from "@angular/core";
5
4
  import * as i1 from "@angular/common";
6
5
  export class NgViewerComponent {
7
6
  constructor() {
8
- this.hotspots = [];
9
- this.hotspotTemplateRef = null;
7
+ this.hotspots2D = [];
10
8
  this.onLoadStart = new EventEmitter();
11
9
  this.onLoadProgress = new EventEmitter();
12
10
  this.onLoadEnd = new EventEmitter();
@@ -14,9 +12,7 @@ export class NgViewerComponent {
14
12
  this.onInteraction = new EventEmitter();
15
13
  this.onVrcubeInteraction = new EventEmitter();
16
14
  this.onVrobjectInteraction = new EventEmitter();
17
- this.fitPosition = { width: 1, height: 1, top: 1, left: 1 };
18
- this.scaleX = 1;
19
- this.scaleY = 1;
15
+ this.hotspotTemplateRef = null;
20
16
  }
21
17
  ngAfterViewInit() {
22
18
  this.viewer = new Viewer(this.container.nativeElement, {
@@ -24,22 +20,39 @@ export class NgViewerComponent {
24
20
  api: this.api || 'v1',
25
21
  fit: this.fit || 'cover',
26
22
  events: {
27
- onLoadStart: (...args) => { this.onLoadStart.emit(...args); },
28
- onLoadProgress: (...args) => { this.onLoadProgress.emit(...args); },
29
- onLoadEnd: (...args) => { this.onLoadEnd.emit(...args); },
30
- onLoadError: (...args) => { this.onLoadError.emit(...args); },
31
- onInteraction: (...args) => { this.onInteraction.emit(...args); },
32
- onVrcubeInteraction: (...args) => { this.onVrcubeInteraction.emit(...args); },
33
- onVrobjectInteraction: (...args) => { this.onVrobjectInteraction.emit(...args); },
34
- onHotspotsChange: this.onHotspotsChange.bind(this)
35
- },
23
+ onLoadStart: (...args) => {
24
+ this.onLoadStart.emit(...args);
25
+ },
26
+ onLoadProgress: (...args) => {
27
+ this.onLoadProgress.emit(...args);
28
+ },
29
+ onLoadEnd: (...args) => {
30
+ this.onLoadEnd.emit(...args);
31
+ },
32
+ onLoadError: (...args) => {
33
+ this.onLoadError.emit(...args);
34
+ },
35
+ onHotspotsChange: this.onHotspotsChange.bind(this),
36
+ onInteraction: (...args) => {
37
+ this.onInteraction.emit(...args);
38
+ },
39
+ onVrcubeInteraction: (...args) => {
40
+ this.onVrcubeInteraction.emit(...args);
41
+ },
42
+ onVrobjectInteraction: (...args) => {
43
+ this.onVrobjectInteraction.emit(...args);
44
+ }
45
+ }
36
46
  });
37
- if (this['scene']) {
47
+ if (this.scene) {
38
48
  this.viewer.load(this.scene).catch(() => { });
39
49
  }
40
- if (this['encoder']) {
50
+ if (this.encoder) {
41
51
  this.viewer.setEncoder(this.encoder).catch(() => { });
42
52
  }
53
+ if (this.hotspots) {
54
+ this.viewer.setHotspots(this.hotspots);
55
+ }
43
56
  if (this.parameters) {
44
57
  this.viewer.setParameters(this.parameters).catch(() => { });
45
58
  }
@@ -52,10 +65,6 @@ export class NgViewerComponent {
52
65
  if (this.vrobject) {
53
66
  this.viewer.setVrobject(this.vrobject);
54
67
  }
55
- if (this.tags) {
56
- this.viewer.setTags(this.tags);
57
- }
58
- this.onResize();
59
68
  }
60
69
  ngOnDestroy() {
61
70
  this.viewer?.destroy();
@@ -70,12 +79,12 @@ export class NgViewerComponent {
70
79
  if (changes['encoder'] && changes['encoder'].currentValue) {
71
80
  this.viewer.setEncoder(changes['encoder'].currentValue).catch(() => { });
72
81
  }
82
+ if (changes['hotspots'] && changes['hotspots'].currentValue) {
83
+ this.viewer.setHotspots(changes['hotspots'].currentValue);
84
+ }
73
85
  if (changes['parameters'] && changes['parameters'].currentValue) {
74
86
  this.viewer.setParameters(changes['parameters'].currentValue).catch(() => { });
75
87
  }
76
- if (changes['tags'] && changes['tags'].currentValue) {
77
- this.viewer.setTags(changes['tags'].currentValue);
78
- }
79
88
  if (changes['view'] && changes['view'].currentValue) {
80
89
  this.viewer.setView(changes['view'].currentValue).catch(() => { });
81
90
  }
@@ -86,43 +95,24 @@ export class NgViewerComponent {
86
95
  this.viewer.setVrobject(changes['vrobject'].currentValue);
87
96
  }
88
97
  }
89
- onHotspotsChange(hotspots) {
90
- this.hotspots = hotspots;
91
- if (this.hotspotsFilter !== undefined) {
92
- this.hotspots = this.hotspotsFilter(this.hotspots);
93
- }
98
+ async pick(position) {
99
+ return this.viewer?.pick(position);
94
100
  }
95
101
  snapshot(type, quality) {
96
102
  return this.viewer?.snapshot(type, quality);
97
103
  }
98
- onResize() {
99
- this.resolution = Viewport.getStandardResolution(window.innerWidth, window.innerHeight);
100
- this.fitPosition = Viewport.fit(this.resolution, window.innerWidth, window.innerHeight, 'cover');
101
- this.scaleX = this.fitPosition.width / this.resolution.width;
102
- this.scaleY = this.fitPosition.height / this.resolution.height;
103
- }
104
- hotspotStyle(hotspot) {
105
- if (this.view?.mode === 'vrcube') {
106
- return {
107
- position: 'absolute',
108
- top: hotspot.position2D.y + 'px',
109
- left: hotspot.position2D.x + 'px'
110
- };
111
- }
112
- else {
113
- return {
114
- position: 'absolute',
115
- top: (hotspot.position2D.y * this.scaleY + this.fitPosition.top) + 'px',
116
- left: (hotspot.position2D.x * this.scaleX + this.fitPosition.left) + 'px'
117
- };
104
+ onHotspotsChange(hotspots) {
105
+ this.hotspots2D = hotspots;
106
+ if (this.hotspotsFilter !== undefined) {
107
+ this.hotspots2D = this.hotspotsFilter(this.hotspots2D);
118
108
  }
119
109
  }
120
110
  }
121
- NgViewerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.0", ngImport: i0, type: NgViewerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
122
- NgViewerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.0", type: NgViewerComponent, selector: "NgViewer", inputs: { server: "server", api: "api", fit: "fit", scene: "scene", encoder: "encoder", parameters: "parameters", view: "view", vrcube: "vrcube", vrobject: "vrobject", tags: "tags", hotspotTemplateRef: "hotspotTemplateRef", hotspotsFilter: "hotspotsFilter" }, outputs: { onLoadStart: "onLoadStart", onLoadProgress: "onLoadProgress", onLoadEnd: "onLoadEnd", onLoadError: "onLoadError", onInteraction: "onInteraction", onVrcubeInteraction: "onVrcubeInteraction", onVrobjectInteraction: "onVrobjectInteraction" }, host: { listeners: { "window:resize": "onResize($event)" } }, viewQueries: [{ propertyName: "container", first: true, predicate: ["container"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div #container></div>\n\n<div class=\"position-absolute w-100 h-100\" style=\"pointer-events: none;\" *ngIf=\"hotspots.length\">\n <div [ngStyle]=\"hotspotStyle(hotspot)\" *ngFor=\"let hotspot of hotspots\">\n <ng-container *ngTemplateOutlet=\"hotspotTemplateRef; context:{hotspot: hotspot}\"></ng-container>\n </div>\n</div>", styles: [":host{display:flex;flex:1 0 auto}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] });
123
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.0", ngImport: i0, type: NgViewerComponent, decorators: [{
111
+ NgViewerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: NgViewerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
112
+ NgViewerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.12", type: NgViewerComponent, selector: "NgViewer", inputs: { server: "server", api: "api", fit: "fit", scene: "scene", encoder: "encoder", parameters: "parameters", view: "view", vrcube: "vrcube", vrobject: "vrobject", hotspots: "hotspots", hotspotsFilter: "hotspotsFilter" }, outputs: { onLoadStart: "onLoadStart", onLoadProgress: "onLoadProgress", onLoadEnd: "onLoadEnd", onLoadError: "onLoadError", onInteraction: "onInteraction", onVrcubeInteraction: "onVrcubeInteraction", onVrobjectInteraction: "onVrobjectInteraction" }, queries: [{ propertyName: "hotspotTemplateRef", first: true, predicate: ["hotspotTemplate"], descendants: true }], viewQueries: [{ propertyName: "container", first: true, predicate: ["container"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div #container></div>\n\n<div class=\"position-absolute w-100 h-100\" style=\"pointer-events: none\" *ngIf=\"hotspots2D.length\">\n <ng-container *ngFor=\"let hotspot of hotspots2D; index as i\">\n <div [ngStyle]=\"{ position: 'absolute', top: hotspot.position2D.y + 'px', left: hotspot.position2D.x + 'px' }\">\n <ng-container *ngTemplateOutlet=\"hotspotTemplateRef; context: { $implicit: hotspot, index: i }\"></ng-container>\n </div>\n </ng-container>\n</div>\n", styles: [":host{display:flex;flex:1 0 auto}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] });
113
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: NgViewerComponent, decorators: [{
124
114
  type: Component,
125
- args: [{ selector: 'NgViewer', template: "<div #container></div>\n\n<div class=\"position-absolute w-100 h-100\" style=\"pointer-events: none;\" *ngIf=\"hotspots.length\">\n <div [ngStyle]=\"hotspotStyle(hotspot)\" *ngFor=\"let hotspot of hotspots\">\n <ng-container *ngTemplateOutlet=\"hotspotTemplateRef; context:{hotspot: hotspot}\"></ng-container>\n </div>\n</div>", styles: [":host{display:flex;flex:1 0 auto}\n"] }]
115
+ args: [{ selector: 'NgViewer', template: "<div #container></div>\n\n<div class=\"position-absolute w-100 h-100\" style=\"pointer-events: none\" *ngIf=\"hotspots2D.length\">\n <ng-container *ngFor=\"let hotspot of hotspots2D; index as i\">\n <div [ngStyle]=\"{ position: 'absolute', top: hotspot.position2D.y + 'px', left: hotspot.position2D.x + 'px' }\">\n <ng-container *ngTemplateOutlet=\"hotspotTemplateRef; context: { $implicit: hotspot, index: i }\"></ng-container>\n </div>\n </ng-container>\n</div>\n", styles: [":host{display:flex;flex:1 0 auto}\n"] }]
126
116
  }], propDecorators: { server: [{
127
117
  type: Input
128
118
  }], api: [{
@@ -141,9 +131,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.0", ngImpor
141
131
  type: Input
142
132
  }], vrobject: [{
143
133
  type: Input
144
- }], tags: [{
145
- type: Input
146
- }], hotspotTemplateRef: [{
134
+ }], hotspots: [{
147
135
  type: Input
148
136
  }], hotspotsFilter: [{
149
137
  type: Input
@@ -161,11 +149,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.0", ngImpor
161
149
  type: Output
162
150
  }], onVrobjectInteraction: [{
163
151
  type: Output
152
+ }], hotspotTemplateRef: [{
153
+ type: ContentChild,
154
+ args: ['hotspotTemplate']
164
155
  }], container: [{
165
156
  type: ViewChild,
166
157
  args: ['container']
167
- }], onResize: [{
168
- type: HostListener,
169
- args: ['window:resize', ['$event']]
170
158
  }] } });
171
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng-viewer.component.js","sourceRoot":"","sources":["../../../../projects/ng-viewer/src/lib/ng-viewer.component.ts","../../../../projects/ng-viewer/src/lib/ng-viewer.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, ElementRef, EventEmitter, Input, OnChanges, OnDestroy, SimpleChanges, ViewChild, Output, TemplateRef, HostListener } from '@angular/core';\n\nimport { Encoder, Hotspot, Parameters, Scene, View, Viewer, WidgetVRCube, WidgetVRObject } from '@lumiscaphe/viewer';\nimport Viewport from './viewport';\n\n@Component({\n  selector: 'NgViewer',\n  templateUrl: './ng-viewer.component.html',\n  styleUrls: ['./ng-viewer.component.scss']\n})\n\nexport class NgViewerComponent implements AfterViewInit, OnChanges, OnDestroy {\n\n  public viewer: Viewer | undefined;\n\n  public hotspots: Hotspot[] = [];\n\n  @Input() server: string | undefined;\n  @Input() api: string | undefined;\n  @Input() fit: 'contain' | 'cover' | 'fill' | undefined;\n  @Input() scene: Scene | undefined;\n  @Input() encoder: Encoder | undefined;\n  @Input() parameters: Parameters | undefined;\n  @Input() view: View | undefined;\n  @Input() vrcube: Partial<WidgetVRCube> | undefined;\n  @Input() vrobject: Partial<WidgetVRObject> | undefined;\n  @Input() tags: string[] | undefined;\n  @Input() hotspotTemplateRef: TemplateRef<any> | null = null;\n  @Input() hotspotsFilter: ((hotspots: Hotspot[]) => Hotspot[]) | undefined;\n\n  @Output() onLoadStart = new EventEmitter();\n  @Output() onLoadProgress = new EventEmitter();\n  @Output() onLoadEnd = new EventEmitter();\n  @Output() onLoadError = new EventEmitter();\n  @Output() onInteraction = new EventEmitter();\n  @Output() onVrcubeInteraction = new EventEmitter();\n  @Output() onVrobjectInteraction = new EventEmitter();\n\n  @ViewChild('container') container!: ElementRef;\n\n  ngAfterViewInit() {\n    this.viewer = new Viewer(this.container.nativeElement, {\n      server: this.server,\n      api: this.api || 'v1',\n      fit: this.fit || 'cover',\n      events: {\n        onLoadStart: (...args: any[]) => { this.onLoadStart.emit(...args); },\n        onLoadProgress: (...args: any[]) => { this.onLoadProgress.emit(...args); },\n        onLoadEnd: (...args: any[]) => { this.onLoadEnd.emit(...args); },\n        onLoadError: (...args: any[]) => { this.onLoadError.emit(...args); },\n        onInteraction: (...args: any[]) => { this.onInteraction.emit(...args); },\n        onVrcubeInteraction: (...args: any[]) => { this.onVrcubeInteraction.emit(...args); },\n        onVrobjectInteraction: (...args: any[]) => { this.onVrobjectInteraction.emit(...args); },\n        onHotspotsChange: this.onHotspotsChange.bind(this)\n      },\n    });\n\n    if (this['scene']) {\n      this.viewer.load(this.scene).catch(() => { });\n    }\n\n    if (this['encoder']) {\n      this.viewer.setEncoder(this.encoder).catch(() => { });\n    }\n\n    if (this.parameters) {\n      this.viewer.setParameters(this.parameters).catch(() => { });\n    }\n\n    if (this.view) {\n      this.viewer.setView(this.view).catch(() => { });\n    }\n\n    if (this.vrcube) {\n      this.viewer.setVrcube(this.vrcube);\n    }\n\n    if (this.vrobject) {\n      this.viewer.setVrobject(this.vrobject);\n    }\n\n    if (this.tags) {\n      this.viewer.setTags(this.tags);\n    }\n\n    this.onResize();\n  }\n\n  ngOnDestroy(): void {\n    this.viewer?.destroy();\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (!this.viewer) {\n      return;\n    }\n\n    if (changes['scene'] && changes['scene'].currentValue) {\n      this.viewer.load(changes['scene'].currentValue).catch(() => { });\n    }\n\n    if (changes['encoder'] && changes['encoder'].currentValue) {\n      this.viewer.setEncoder(changes['encoder'].currentValue).catch(() => { });\n    }\n\n    if (changes['parameters'] && changes['parameters'].currentValue) {\n      this.viewer.setParameters(changes['parameters'].currentValue).catch(() => { });\n    }\n\n    if (changes['tags'] && changes['tags'].currentValue) {\n      this.viewer.setTags(changes['tags'].currentValue);\n    }\n\n    if (changes['view'] && changes['view'].currentValue) {\n      this.viewer.setView(changes['view'].currentValue).catch(() => { });\n    }\n\n    if (changes['vrcube'] && changes['vrcube'].currentValue) {\n      this.viewer.setVrcube(changes['vrcube'].currentValue);\n    }\n\n    if (changes['vrobject'] && changes['vrobject'].currentValue) {\n      this.viewer.setVrobject(changes['vrobject'].currentValue);\n    }\n  }\n\n  onHotspotsChange(hotspots: Hotspot[]): void {\n    this.hotspots = hotspots;\n\n    if (this.hotspotsFilter !== undefined) {\n      this.hotspots = this.hotspotsFilter(this.hotspots);\n    }\n  }\n\n  snapshot(type: string, quality: number): string | undefined {\n    return this.viewer?.snapshot(type, quality);\n  }\n\n  public fitPosition: { width: number, height: number, top: number, left: number } = { width: 1, height: 1, top: 1, left: 1 };\n  public resolution: { width: number, height: number } | undefined;\n  public scaleX: number = 1;\n  public scaleY: number = 1;\n\n  @HostListener('window:resize', ['$event'])\n  public onResize(): void {\n    this.resolution = Viewport.getStandardResolution(window.innerWidth, window.innerHeight);\n    this.fitPosition = Viewport.fit(this.resolution, window.innerWidth, window.innerHeight, 'cover');\n    this.scaleX = this.fitPosition.width / this.resolution.width;\n    this.scaleY = this.fitPosition.height / this.resolution.height;\n  }\n\n  public hotspotStyle(hotspot: Hotspot): { left: string, top: string, position: string } {\n    if (this.view?.mode === 'vrcube') {\n      return {\n        position: 'absolute',\n        top: hotspot.position2D.y + 'px',\n        left: hotspot.position2D.x + 'px'\n      };\n    } else {\n      return {\n        position: 'absolute',\n        top: (hotspot.position2D.y * this.scaleY + this.fitPosition.top) + 'px',\n        left: (hotspot.position2D.x * this.scaleX + this.fitPosition.left) + 'px'\n      };\n    }\n  }\n}\n","<div #container></div>\n\n<div class=\"position-absolute w-100 h-100\" style=\"pointer-events: none;\" *ngIf=\"hotspots.length\">\n  <div [ngStyle]=\"hotspotStyle(hotspot)\" *ngFor=\"let hotspot of hotspots\">\n    <ng-container *ngTemplateOutlet=\"hotspotTemplateRef; context:{hotspot: hotspot}\"></ng-container>\n  </div>\n</div>"]}
159
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng-viewer.component.js","sourceRoot":"","sources":["../../../../projects/ng-viewer/src/lib/ng-viewer.component.ts","../../../../projects/ng-viewer/src/lib/ng-viewer.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, ContentChild, ElementRef, EventEmitter, Input, OnChanges, OnDestroy, Output, SimpleChanges, TemplateRef, ViewChild } from '@angular/core';\nimport { Encoder, Hotspot, Parameters, Scene, View, Viewer, WRAPIv2, WidgetVRCube, WidgetVRObject } from '@lumiscaphe/viewer';\n\n@Component({\n  selector: 'NgViewer',\n  templateUrl: './ng-viewer.component.html',\n  styleUrls: ['./ng-viewer.component.scss']\n})\nexport class NgViewerComponent implements AfterViewInit, OnChanges, OnDestroy {\n  public viewer: Viewer | undefined;\n\n  public hotspots2D: Hotspot[] = [];\n\n  @Input() server: string | undefined;\n  @Input() api: string | undefined;\n  @Input() fit: 'contain' | 'cover' | 'fill' | undefined;\n  @Input() scene: Scene | undefined;\n  @Input() encoder: Encoder | undefined;\n  @Input() parameters: Parameters | undefined;\n  @Input() view: View | undefined;\n  @Input() vrcube: Partial<WidgetVRCube> | undefined;\n  @Input() vrobject: Partial<WidgetVRObject> | undefined;\n  @Input() hotspots: Array<string | WRAPIv2.Vector3D> | undefined;\n  @Input() hotspotsFilter: ((hotspots: Hotspot[]) => Hotspot[]) | undefined;\n\n  @Output() onLoadStart = new EventEmitter();\n  @Output() onLoadProgress = new EventEmitter();\n  @Output() onLoadEnd = new EventEmitter();\n  @Output() onLoadError = new EventEmitter();\n  @Output() onInteraction = new EventEmitter();\n  @Output() onVrcubeInteraction = new EventEmitter();\n  @Output() onVrobjectInteraction = new EventEmitter();\n\n  @ContentChild('hotspotTemplate') hotspotTemplateRef: TemplateRef<any> | null = null;\n\n  @ViewChild('container') container!: ElementRef;\n\n  ngAfterViewInit() {\n    this.viewer = new Viewer(this.container.nativeElement, {\n      server: this.server,\n      api: this.api || 'v1',\n      fit: this.fit || 'cover',\n      events: {\n        onLoadStart: (...args: any[]) => {\n          this.onLoadStart.emit(...args);\n        },\n        onLoadProgress: (...args: any[]) => {\n          this.onLoadProgress.emit(...args);\n        },\n        onLoadEnd: (...args: any[]) => {\n          this.onLoadEnd.emit(...args);\n        },\n        onLoadError: (...args: any[]) => {\n          this.onLoadError.emit(...args);\n        },\n        onHotspotsChange: this.onHotspotsChange.bind(this),\n        onInteraction: (...args: any[]) => {\n          this.onInteraction.emit(...args);\n        },\n        onVrcubeInteraction: (...args: any[]) => {\n          this.onVrcubeInteraction.emit(...args);\n        },\n        onVrobjectInteraction: (...args: any[]) => {\n          this.onVrobjectInteraction.emit(...args);\n        }\n      }\n    });\n\n    if (this.scene) {\n      this.viewer.load(this.scene).catch(() => {});\n    }\n\n    if (this.encoder) {\n      this.viewer.setEncoder(this.encoder).catch(() => {});\n    }\n\n    if (this.hotspots) {\n      this.viewer.setHotspots(this.hotspots);\n    }\n\n    if (this.parameters) {\n      this.viewer.setParameters(this.parameters).catch(() => {});\n    }\n\n    if (this.view) {\n      this.viewer.setView(this.view).catch(() => {});\n    }\n\n    if (this.vrcube) {\n      this.viewer.setVrcube(this.vrcube);\n    }\n\n    if (this.vrobject) {\n      this.viewer.setVrobject(this.vrobject);\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.viewer?.destroy();\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (!this.viewer) {\n      return;\n    }\n\n    if (changes['scene'] && changes['scene'].currentValue) {\n      this.viewer.load(changes['scene'].currentValue).catch(() => {});\n    }\n\n    if (changes['encoder'] && changes['encoder'].currentValue) {\n      this.viewer.setEncoder(changes['encoder'].currentValue).catch(() => {});\n    }\n\n    if (changes['hotspots'] && changes['hotspots'].currentValue) {\n      this.viewer.setHotspots(changes['hotspots'].currentValue);\n    }\n\n    if (changes['parameters'] && changes['parameters'].currentValue) {\n      this.viewer.setParameters(changes['parameters'].currentValue).catch(() => {});\n    }\n\n    if (changes['view'] && changes['view'].currentValue) {\n      this.viewer.setView(changes['view'].currentValue).catch(() => {});\n    }\n\n    if (changes['vrcube'] && changes['vrcube'].currentValue) {\n      this.viewer.setVrcube(changes['vrcube'].currentValue);\n    }\n\n    if (changes['vrobject'] && changes['vrobject'].currentValue) {\n      this.viewer.setVrobject(changes['vrobject'].currentValue);\n    }\n  }\n\n  async pick(position: WRAPIv2.Vector2D): Promise<WRAPIv2.PickResult | undefined> {\n    return this.viewer?.pick(position);\n  }\n\n  snapshot(type: string, quality: number): string | undefined {\n    return this.viewer?.snapshot(type, quality);\n  }\n\n  onHotspotsChange(hotspots: Hotspot[]): void {\n    this.hotspots2D = hotspots;\n\n    if (this.hotspotsFilter !== undefined) {\n      this.hotspots2D = this.hotspotsFilter(this.hotspots2D);\n    }\n  }\n}\n","<div #container></div>\n\n<div class=\"position-absolute w-100 h-100\" style=\"pointer-events: none\" *ngIf=\"hotspots2D.length\">\n  <ng-container *ngFor=\"let hotspot of hotspots2D; index as i\">\n    <div [ngStyle]=\"{ position: 'absolute', top: hotspot.position2D.y + 'px', left: hotspot.position2D.x + 'px' }\">\n      <ng-container *ngTemplateOutlet=\"hotspotTemplateRef; context: { $implicit: hotspot, index: i }\"></ng-container>\n    </div>\n  </ng-container>\n</div>\n"]}