@libs-ui/components-audio 0.2.355-9 → 0.2.356-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
@@ -1,347 +1,210 @@
1
- # Audio Component
1
+ # @libs-ui/components-audio
2
2
 
3
- ## Giới thiệu
3
+ > Component phát âm thanh với đầy đủ chức năng điều khiển cho Angular.
4
4
 
5
- `audio` một component mạnh mẽ dùng để phát âm thanh trong ứng dụng Angular. Component này cung cấp giao diện đơn giản để phát, tạm dừng và điều khiển các tệp âm thanh với nhiều tính năng.
5
+ ## Giới thiệu
6
6
 
7
- ## Tính năng
7
+ `LibsUiComponentsAudioComponent` một standalone Angular component cung cấp giao diện phát audio với các tính năng:
8
8
 
9
- - Phát/tạm dừng âm thanh
10
- - Điều chỉnh âm lượng với chức năng tắt/bật tiếng
11
- - Hiển thị thời gian theo định dạng HH:MM:SS
12
- - Thanh tiến độ với chức năng tua nhanh/tua lại
13
- - Tải xuống audio với kiểm soát quyền
14
- - Function Control API cho phép điều khiển từ bên ngoài
15
- - Thiết kế responsive
16
- - Source audio thể cấu hình
9
+ - Phát/tạm dừng âm thanh với giao diện trực quan
10
+ - Điều chỉnh âm lượng với thanh trượt ẩn/hiện
11
+ - Hiển thị thời gian theo định dạng HH:MM:SS
12
+ - Thanh tiến độ với chức năng tua nhanh/tua lại
13
+ - Tải xuống audio với kiểm soát quyền
14
+ - Function Control API cho phép điều khiển từ bên ngoài
15
+ - Hỗ trợ OnPush Change Detection cho hiệu năng cao
16
+ - Sử dụng Angular Signals cho state management
17
17
 
18
18
  ## Cài đặt
19
19
 
20
- Để cài đặt component `audio`, sử dụng npm hoặc yarn:
21
-
22
20
  ```bash
21
+ # npm
23
22
  npm install @libs-ui/components-audio
24
- ```
25
-
26
- hoặc
27
23
 
28
- ```bash
24
+ # yarn
29
25
  yarn add @libs-ui/components-audio
30
26
  ```
31
27
 
32
28
  ## Sử dụng
33
29
 
34
- ### Import module
35
-
36
- ```typescript
37
- import { LibsUiComponentsAudioComponent } from '@libs-ui/components-audio';
38
-
39
- @NgModule({
40
- declarations: [AppComponent],
41
- imports: [BrowserModule, LibsUiComponentsAudioComponent],
42
- bootstrap: [AppComponent],
43
- })
44
- export class AppModule {}
45
- ```
46
-
47
- Hoặc trong component standalone:
30
+ ### Import Component
48
31
 
49
32
  ```typescript
50
33
  import { Component } from '@angular/core';
51
- import { CommonModule } from '@angular/common';
52
- import { LibsUiComponentsAudioComponent } from '@libs-ui/components-audio';
53
- import { IAudioFunctionControlEvent } from '@libs-ui/components-audio';
34
+ import { LibsUiComponentsAudioComponent, IAudioFunctionControlEvent } from '@libs-ui/components-audio';
54
35
 
55
36
  @Component({
56
37
  selector: 'app-example',
57
38
  standalone: true,
58
- imports: [CommonModule, LibsUiComponentsAudioComponent],
39
+ imports: [LibsUiComponentsAudioComponent],
59
40
  template: `
60
41
  <libs_ui-components-audio
61
- [fileAudio]="audioSource"
62
- [checkPermissionDownloadAudio]="checkDownloadPermission"
63
- (outFunctionsControl)="registerFunctions($event)"></libs_ui-components-audio>
64
-
65
- <div class="controls">
66
- <button (click)="playAudio()">Phát/Tạm dừng</button>
67
- <button (click)="toggleMute()">Bật/Tắt tiếng</button>
68
- </div>
42
+ [fileAudio]="audioUrl"
43
+ [checkPermissionDownloadAudio]="checkPermission"
44
+ (outFunctionsControl)="onFunctionsControl($event)"
45
+ (outPlay)="onPlayChange($event)"
46
+ (outVolumeControl)="onVolumeChange($event)"
47
+ (outTimeUpdate)="onTimeUpdate($event)"
48
+ (outMute)="onMuteChange($event)"
49
+ (outEnded)="onEnded()"></libs_ui-components-audio>
69
50
  `,
70
51
  })
71
52
  export class ExampleComponent {
72
- audioSource = 'path/to/audio/file.mp3';
73
- functionControls: IAudioFunctionControlEvent | null = null;
53
+ audioUrl = 'https://example.com/audio.mp3';
54
+ controls: IAudioFunctionControlEvent | null = null;
55
+
56
+ checkPermission = (): Promise<boolean> => Promise.resolve(true);
57
+
58
+ onFunctionsControl(event: IAudioFunctionControlEvent) {
59
+ this.controls = event;
60
+ }
61
+
62
+ onPlayChange(isPlaying: boolean) {
63
+ console.log('Đang phát:', isPlaying);
64
+ }
74
65
 
75
- checkDownloadPermission(): Promise<boolean> {
76
- // Kiểm tra quyền download
77
- return Promise.resolve(true);
66
+ onVolumeChange(volume: number) {
67
+ console.log('Âm lượng:', volume);
78
68
  }
79
69
 
80
- registerFunctions(event: IAudioFunctionControlEvent) {
81
- this.functionControls = event;
70
+ onTimeUpdate(time: { currentTime: string; duration: string }) {
71
+ console.log('Thời gian:', time.currentTime, '/', time.duration);
82
72
  }
83
73
 
84
- playAudio() {
85
- if (this.functionControls) {
86
- this.functionControls.playPause();
87
- }
74
+ onMuteChange(isMuted: boolean) {
75
+ console.log('Tắt tiếng:', isMuted);
88
76
  }
89
77
 
90
- toggleMute() {
91
- if (this.functionControls) {
92
- this.functionControls.toggleMute();
93
- }
78
+ onEnded() {
79
+ console.log('Audio đã phát xong');
94
80
  }
95
81
  }
96
82
  ```
97
83
 
84
+ ### Điều khiển từ bên ngoài
85
+
86
+ Sử dụng `outFunctionsControl` để lấy reference các hàm điều khiển:
87
+
88
+ ```typescript
89
+ // Phát/Tạm dừng
90
+ this.controls.playPause();
91
+
92
+ // Bật/Tắt tiếng
93
+ this.controls.toggleMute();
94
+
95
+ // Set âm lượng (0-100)
96
+ this.controls.setVolume(50);
97
+
98
+ // Tua đến vị trí (0-100%)
99
+ this.controls.seekTo(25);
100
+
101
+ // Download audio
102
+ this.controls.download();
103
+
104
+ // Kiểm tra trạng thái
105
+ const isPlaying = this.controls.isPlaying();
106
+ const isMuted = this.controls.isMuted();
107
+ ```
108
+
98
109
  ## API Reference
99
110
 
100
111
  ### Inputs
101
112
 
102
- | Tên | Kiểu | Mặc định | Mô tả |
103
- | ---------------------------- | ------------------------ | -------- | ---------------------------------------------------------------------------- |
104
- | fileAudio | `string` | required | URL của file audio cần phát. |
105
- | checkPermissionDownloadAudio | `() => Promise<boolean>` | required | Function trả về promise với kết quả boolean cho biết nếu được phép download. |
113
+ | Tên | Kiểu | Mặc định | Mô tả |
114
+ | ------------------------------ | ------------------------ | ---------- | --------------------------------------------------------------------- |
115
+ | `fileAudio` | `string` | _required_ | URL của file audio cần phát. Hỗ trợ MP3, WAV, OGG. |
116
+ | `checkPermissionDownloadAudio` | `() => Promise<boolean>` | _required_ | Callback kiểm tra quyền download. Trả về Promise với kết quả boolean. |
106
117
 
107
118
  ### Outputs
108
119
 
109
- | Tên | Kiểu | Mô tả |
110
- | ------------------- | ------------------------------------------- | ------------------------------------------------------- |
111
- | outFunctionsControl | `IAudioFunctionControlEvent` | Event chứa các hàm điều khiển của audio component. |
112
- | outVolumeControl | `number` | Phát ra giá trị âm lượng hiện tại (0-100). |
113
- | outTimeUpdate | `{ currentTime: string, duration: string }` | Phát ra thông tin thời gian hiện tại tổng thời gian. |
114
- | outEnded | `void` | Phát ra khi audio kết thúc phát. |
115
- | outMute | `boolean` | Phát ra trạng thái tắt/bật tiếng. |
116
- | outPlay | `boolean` | Phát ra trạng thái phát/tạm dừng. |
117
-
118
- ### Các phương thức (qua outFunctionsControl)
119
-
120
- | Tên phương thức | Tham số | Kiểu trả về | Mô tả |
121
- | --------------- | --------------- | ----------- | ------------------------------------------------ |
122
- | playPause | `event?: Event` | `void` | Bắt đầu/tạm dừng phát audio. |
123
- | toggleMute | `event?: Event` | `void` | Bật/tắt âm thanh. |
124
- | setVolume | `value: number` | `void` | Điều chỉnh âm lượng (0-100). |
125
- | seekTo | `value: number` | `void` | Di chuyển đến vị trí cụ thể trong audio (0-100). |
126
- | download | `event?: Event` | `void` | Tải xuống file audio. |
127
- | isPlaying | `-` | `boolean` | Kiểm tra trạng thái đang phát audio. |
128
- | isMuted | `-` | `boolean` | Kiểm tra trạng thái tắt tiếng. |
129
-
130
- ## Interfaces
120
+ | Tên | Kiểu | Mô tả |
121
+ | --------------------- | ------------------------------------------- | ------------------------------------------------------------------------------------------------------------ |
122
+ | `outFunctionsControl` | `IAudioFunctionControlEvent` | Emit object chứa các hàm điều khiển: playPause, toggleMute, seekTo, setVolume, download, isPlaying, isMuted. |
123
+ | `outVolumeControl` | `number` | Emit giá trị âm lượng hiện tại từ 0-100. |
124
+ | `outTimeUpdate` | `{ currentTime: string, duration: string }` | Emit thông tin thời gian theo định dạng HH:MM:SS. |
125
+ | `outEnded` | `void` | Emit khi audio phát xong. |
126
+ | `outMute` | `boolean` | Emit trạng thái tắt/bật tiếng. |
127
+ | `outPlay` | `boolean` | Emit trạng thái đang phát/tạm dừng. |
128
+
129
+ ### Interface: IAudioFunctionControlEvent
131
130
 
132
131
  ```typescript
133
- // Audio Function Control Event
134
132
  interface IAudioFunctionControlEvent {
133
+ /**
134
+ * Bắt đầu hoặc tạm dừng phát audio
135
+ */
135
136
  playPause: (event?: Event) => void;
137
+
138
+ /**
139
+ * Bật hoặc tắt âm thanh
140
+ */
136
141
  toggleMute: (event?: Event) => void;
142
+
143
+ /**
144
+ * Điều chỉnh âm lượng (0-100)
145
+ */
137
146
  setVolume: (value: number) => void;
147
+
148
+ /**
149
+ * Di chuyển đến vị trí cụ thể (0-100%)
150
+ */
138
151
  seekTo: (value: number) => void;
152
+
153
+ /**
154
+ * Tải xuống file audio
155
+ */
139
156
  download: (event?: Event) => void;
157
+
158
+ /**
159
+ * Kiểm tra trạng thái đang phát
160
+ */
140
161
  isPlaying: () => boolean;
162
+
163
+ /**
164
+ * Kiểm tra trạng thái tắt tiếng
165
+ */
141
166
  isMuted: () => boolean;
142
167
  }
143
168
  ```
144
169
 
145
- ## Styling Volume Slider
170
+ ## Công nghệ sử dụng
146
171
 
147
- Để tạo hiệu ứng thanh trượt âm lượng có màu nền thay đổi theo giá trị, bạn có thể sử dụng CSS variables:
172
+ - **Angular 18+** - Standalone Components
173
+ - **Angular Signals** - Reactive state management
174
+ - **RxJS** - Event handling với `takeUntilDestroyed`
175
+ - **TailwindCSS** - Styling
148
176
 
149
- ```css
150
- /* Định nghĩa thanh trượt âm lượng */
151
- input[type='range'].volume-slider {
152
- background: linear-gradient(to right, #3b82f6 var(--volume-percent, 50%), #e5e7eb var(--volume-percent, 50%));
153
- }
154
-
155
- /* Track styling cho WebKit browsers */
156
- input[type='range'].volume-slider::-webkit-slider-runnable-track {
157
- background: linear-gradient(to right, #3b82f6 var(--volume-percent, 50%), #e5e7eb var(--volume-percent, 50%));
158
- }
177
+ ## Demo
159
178
 
160
- /* Track styling cho Firefox */
161
- input[type='range'].volume-slider::-moz-range-track {
162
- background: linear-gradient(to right, #3b82f6 var(--volume-percent, 50%), #e5e7eb var(--volume-percent, 50%));
163
- }
164
- ```
179
+ Demo sẵn trong ứng dụng `core-ui`:
165
180
 
166
- Và trong template:
167
-
168
- ```html
169
- <input
170
- type="range"
171
- min="0"
172
- max="100"
173
- [value]="volumePercent()"
174
- (input)="changeVolume($event)"
175
- class="volume-slider"
176
- [style.--volume-percent.%]="volumePercent()" />
181
+ ```bash
182
+ npx nx serve core-ui
177
183
  ```
178
184
 
179
- ## dụ
180
-
181
- ### Sử dụng Function Control
185
+ **File demo:** `apps/core-ui/src/app/components/audio/audio.component.ts`
182
186
 
183
- ```typescript
184
- import { Component, signal, computed } from '@angular/core';
185
- import { IAudioFunctionControlEvent } from '@libs-ui/components-audio';
187
+ Truy cập: `http://localhost:4500/audio`
186
188
 
187
- @Component({
188
- selector: 'app-example',
189
- template: `
190
- <libs_ui-components-audio
191
- [fileAudio]="audioSource()"
192
- [checkPermissionDownloadAudio]="checkDownloadPermission"
193
- (outFunctionsControl)="registerFunctions($event)"></libs_ui-components-audio>
194
-
195
- <div class="audio-controls">
196
- <button (click)="playPauseAudio()">{{ isPlaying() ? 'Tạm dừng' : 'Phát' }}</button>
197
- <button (click)="toggleMuteAudio()">{{ isMuted() ? 'Bật tiếng' : 'Tắt tiếng' }}</button>
198
- <div class="volume-control">
199
- <span>Âm lượng:</span>
200
- <input
201
- type="range"
202
- min="0"
203
- max="100"
204
- [value]="volumePercent()"
205
- (input)="changeVolume($event)"
206
- class="volume-slider"
207
- [style.--volume-percent.%]="volumePercent()" />
208
- <span>{{ volumePercent() }}%</span>
209
- </div>
210
- <div class="progress-control">
211
- <span>Tiến độ:</span>
212
- <input
213
- type="range"
214
- min="0"
215
- max="100"
216
- [value]="progress()"
217
- (input)="changeProgress($event)" />
218
- <span>{{ progress() }}%</span>
219
- </div>
220
- <button (click)="downloadAudio()">Tải xuống</button>
221
- </div>
222
- `,
223
- })
224
- export class ExampleComponent {
225
- audioSource = signal('path/to/audio.mp3');
226
- functionControls: IAudioFunctionControlEvent | null = null;
227
- isPlaying = signal(false);
228
- isMuted = signal(false);
229
- volume = signal(80);
230
- progress = signal(0);
231
-
232
- // Computed properties
233
- volumePercent = computed(() => Math.round(this.volume()));
234
-
235
- checkDownloadPermission = (): Promise<boolean> => {
236
- return Promise.resolve(true);
237
- };
238
-
239
- registerFunctions(event: IAudioFunctionControlEvent) {
240
- this.functionControls = event;
241
-
242
- // Initialize state
243
- if (this.functionControls) {
244
- this.isPlaying.set(this.functionControls.isPlaying());
245
- this.isMuted.set(this.functionControls.isMuted());
246
- }
247
- }
189
+ ### Tính năng Demo
248
190
 
249
- playPauseAudio() {
250
- if (this.functionControls) {
251
- this.functionControls.playPause();
252
- this.isPlaying.set(this.functionControls.isPlaying());
253
- }
254
- }
191
+ - 🎮 **Basic Example**: Cách sử dụng cơ bản
192
+ - 🎛️ **External Controls**: Điều khiển audio từ component cha
193
+ - 📊 **Real-time Status**: Hiển thị trạng thái play, mute, time, volume
255
194
 
256
- toggleMuteAudio() {
257
- if (this.functionControls) {
258
- this.functionControls.toggleMute();
259
- this.isMuted.set(this.functionControls.isMuted());
260
- }
261
- }
195
+ ## Unit Tests
262
196
 
263
- changeVolume(event: Event) {
264
- if (this.functionControls && event.target) {
265
- const value = parseInt((event.target as HTMLInputElement).value);
266
- this.volume.set(value);
267
- this.functionControls.setVolume(value / 100); // Convert to 0-1 range
268
- }
269
- }
197
+ ```bash
198
+ # Chạy tests
199
+ npx nx test components-audio
270
200
 
271
- changeProgress(event: Event) {
272
- if (this.functionControls && event.target) {
273
- const value = parseInt((event.target as HTMLInputElement).value);
274
- this.progress.set(value);
275
- this.functionControls.seekTo(value);
276
- }
277
- }
201
+ # Chạy tests với coverage
202
+ npx nx test components-audio --coverage
278
203
 
279
- downloadAudio() {
280
- if (this.functionControls) {
281
- this.functionControls.download();
282
- }
283
- }
284
- }
204
+ # Watch mode
205
+ npx nx test components-audio --watch
285
206
  ```
286
207
 
287
- ### Sử dụng Events
288
-
289
- Sử dụng các events để phản ứng với thay đổi từ audio player:
290
-
291
- ```typescript
292
- import { Component, signal } from '@angular/core';
293
-
294
- @Component({
295
- selector: 'app-example',
296
- template: `
297
- <libs_ui-components-audio
298
- [fileAudio]="audioSource()"
299
- [checkPermissionDownloadAudio]="checkPermission"
300
- (outTimeUpdate)="handleTimeUpdate($event)"
301
- (outVolumeControl)="handleVolumeChange($event)"
302
- (outPlay)="handlePlayChange($event)"
303
- (outMute)="handleMuteChange($event)"
304
- (outEnded)="handleEnded()"></libs_ui-components-audio>
305
-
306
- <div class="audio-info">
307
- <p>Trạng thái: {{ isPlaying() ? 'Đang phát' : 'Tạm dừng' }}</p>
308
- <p>Thời gian hiện tại: {{ currentTime() }}</p>
309
- <p>Tổng thời gian: {{ duration() }}</p>
310
- <p>Âm lượng: {{ volumeLevel() }}%</p>
311
- </div>
312
- `,
313
- })
314
- export class ExampleComponent {
315
- audioSource = signal('path/to/audio.mp3');
316
- isPlaying = signal(false);
317
- isMuted = signal(false);
318
- currentTime = signal('00:00:00');
319
- duration = signal('00:00:00');
320
- volumeLevel = signal(100);
321
-
322
- checkPermission = (): Promise<boolean> => {
323
- return Promise.resolve(true);
324
- };
325
-
326
- handleTimeUpdate(timeInfo: { currentTime: string; duration: string }) {
327
- this.currentTime.set(timeInfo.currentTime);
328
- this.duration.set(timeInfo.duration);
329
- }
330
-
331
- handleVolumeChange(volume: number) {
332
- this.volumeLevel.set(volume);
333
- }
334
-
335
- handlePlayChange(isPlaying: boolean) {
336
- this.isPlaying.set(isPlaying);
337
- }
208
+ ## License
338
209
 
339
- handleMuteChange(isMuted: boolean) {
340
- this.isMuted.set(isMuted);
341
- }
342
-
343
- handleEnded() {
344
- this.isPlaying.set(false);
345
- }
346
- }
347
- ```
210
+ MIT
@@ -34,10 +34,20 @@ export declare class LibsUiComponentsAudioComponent implements AfterViewInit {
34
34
  protected handlerAudioPausePlay(event?: Event): Promise<void>;
35
35
  protected handlerLoadedData(event: Event): Promise<void>;
36
36
  protected handlerTimeUpdate(event?: Event): Promise<void>;
37
+ /**
38
+ * Format seconds -> HH:MM:SS.
39
+ * Used for both `currentTime` and `duration` outputs to keep output stable.
40
+ */
37
41
  private toHHMMSS;
38
42
  protected handlerChangeAudio(value: number): Promise<void>;
39
43
  protected handlerChangeVolume(value: number): Promise<void>;
40
44
  protected handlerEnded(event: Event): Promise<void>;
45
+ /**
46
+ * Download chỉ được thực hiện nếu callback permission trả về `true`.
47
+ * Lưu ý: `checkPermissionDownloadAudio` là input function (factory), nên cần gọi 2 lần:
48
+ * - lần 1: lấy function
49
+ * - lần 2: execute function để lấy Promise<boolean>
50
+ */
41
51
  protected handlerDownload(e?: Event): Promise<void>;
42
52
  static ɵfac: i0.ɵɵFactoryDeclaration<LibsUiComponentsAudioComponent, never>;
43
53
  static ɵcmp: i0.ɵɵComponentDeclaration<LibsUiComponentsAudioComponent, "libs_ui-components-audio", never, { "fileAudio": { "alias": "fileAudio"; "required": true; "isSignal": true; }; "checkPermissionDownloadAudio": { "alias": "checkPermissionDownloadAudio"; "required": true; "isSignal": true; }; }, { "outFunctionsControl": "outFunctionsControl"; "outVolumeControl": "outVolumeControl"; "outTimeUpdate": "outTimeUpdate"; "outEnded": "outEnded"; "outMute": "outMute"; "outPlay": "outPlay"; }, never, never, true, never>;
@@ -4,7 +4,9 @@ import { LibsUiComponentsInputsRangeSliderComponent } from '@libs-ui/components-
4
4
  import { fromEvent, merge, tap } from 'rxjs';
5
5
  import * as i0 from "@angular/core";
6
6
  export class LibsUiComponentsAudioComponent {
7
- // #region PROPERTY
7
+ // =========================================
8
+ // INTERNAL SIGNALS
9
+ // =========================================
8
10
  audioRatioValue = signal(0);
9
11
  volumeRatioValue = signal(100);
10
12
  isPlay = signal(false);
@@ -14,25 +16,36 @@ export class LibsUiComponentsAudioComponent {
14
16
  audioTimeCurrent = signal('_:_:_');
15
17
  audioTimeDuration = signal('_:_:_');
16
18
  showFullControlVolume = signal(false);
17
- // #region INPUT
19
+ // =========================================
20
+ // INPUTS
21
+ // =========================================
18
22
  fileAudio = input.required();
19
23
  checkPermissionDownloadAudio = input.required();
20
- /* VIEW CHILD */
24
+ // =========================================
25
+ // VIEW CHILDREN
26
+ // =========================================
21
27
  audioRef = viewChild.required('audioRef');
22
28
  volumeControlRef = viewChild.required('volumeControlRef');
23
- /* OUTPUTS */
29
+ // =========================================
30
+ // OUTPUTS
31
+ // =========================================
24
32
  outFunctionsControl = output();
25
33
  outVolumeControl = output();
26
34
  outTimeUpdate = output();
27
35
  outEnded = output();
28
36
  outMute = output();
29
37
  outPlay = output();
38
+ // =========================================
39
+ // PRIVATE PROPERTIES
40
+ // =========================================
30
41
  destroyRef = inject(DestroyRef);
42
+ // =========================================
43
+ // CONSTRUCTOR
44
+ // =========================================
31
45
  constructor() {
32
46
  // Watch for file audio changes
33
47
  effect(() => {
34
48
  if (this.fileAudio() && this.audioRef()) {
35
- // Skip initial setup, only reload on changes
36
49
  setTimeout(() => {
37
50
  this.audioRef().nativeElement.load();
38
51
  }, 0);
@@ -51,6 +64,9 @@ export class LibsUiComponentsAudioComponent {
51
64
  this.outPlay.emit(this.isPlay());
52
65
  });
53
66
  }
67
+ // =========================================
68
+ // LIFECYCLE HOOKS
69
+ // =========================================
54
70
  ngAfterViewInit() {
55
71
  merge(this.initObservable(this.volumeControlRef().nativeElement, 'mouseenter').pipe(tap(() => this.showFullControlVolume.set(true))), this.initObservable(this.volumeControlRef().nativeElement, 'mouseleave').pipe(tap(() => this.showFullControlVolume.set(false))))
56
72
  .pipe(takeUntilDestroyed(this.destroyRef))
@@ -58,6 +74,9 @@ export class LibsUiComponentsAudioComponent {
58
74
  // Emit function control event after view is initialized
59
75
  this.outFunctionsControl.emit(this.FunctionsControl);
60
76
  }
77
+ // =========================================
78
+ // PUBLIC API
79
+ // =========================================
61
80
  get FunctionsControl() {
62
81
  return {
63
82
  playPause: (event) => this.handlerAudioPausePlay(event),
@@ -69,7 +88,9 @@ export class LibsUiComponentsAudioComponent {
69
88
  isMuted: () => this.isMute(),
70
89
  };
71
90
  }
72
- /* FUNCTIONS */
91
+ // =========================================
92
+ // PRIVATE METHODS
93
+ // =========================================
73
94
  initObservable(el, eventName) {
74
95
  return fromEvent(el, eventName).pipe(tap((e) => e.stopPropagation()), takeUntilDestroyed(this.destroyRef));
75
96
  }
@@ -137,6 +158,10 @@ export class LibsUiComponentsAudioComponent {
137
158
  }
138
159
  this.audioRatioValue.set(Math.floor(((this.audioRef().nativeElement.currentTime || 0) / (this.audioRef().nativeElement.duration || 1)) * 100));
139
160
  }
161
+ /**
162
+ * Format seconds -> HH:MM:SS.
163
+ * Used for both `currentTime` and `duration` outputs to keep output stable.
164
+ */
140
165
  async toHHMMSS(time) {
141
166
  const hours = Math.floor(time / 3600);
142
167
  const minutes = Math.floor((time - hours * 3600) / 60);
@@ -173,6 +198,12 @@ export class LibsUiComponentsAudioComponent {
173
198
  this.isPlay.set(false);
174
199
  this.outEnded.emit();
175
200
  }
201
+ /**
202
+ * Download chỉ được thực hiện nếu callback permission trả về `true`.
203
+ * Lưu ý: `checkPermissionDownloadAudio` là input function (factory), nên cần gọi 2 lần:
204
+ * - lần 1: lấy function
205
+ * - lần 2: execute function để lấy Promise<boolean>
206
+ */
176
207
  async handlerDownload(e) {
177
208
  if (!this.checkPermissionDownloadAudio() || !(await this.checkPermissionDownloadAudio()())) {
178
209
  return;
@@ -183,7 +214,7 @@ export class LibsUiComponentsAudioComponent {
183
214
  if (!this.fileAudio()) {
184
215
  return;
185
216
  }
186
- window.open(this.fileAudio(), `_blank`);
217
+ window.open(this.fileAudio(), '_blank');
187
218
  }
188
219
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LibsUiComponentsAudioComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
189
220
  static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.2.0", version: "18.2.14", type: LibsUiComponentsAudioComponent, isStandalone: true, selector: "libs_ui-components-audio", inputs: { fileAudio: { classPropertyName: "fileAudio", publicName: "fileAudio", isSignal: true, isRequired: true, transformFunction: null }, checkPermissionDownloadAudio: { classPropertyName: "checkPermissionDownloadAudio", publicName: "checkPermissionDownloadAudio", isSignal: true, isRequired: true, transformFunction: null } }, outputs: { outFunctionsControl: "outFunctionsControl", outVolumeControl: "outVolumeControl", outTimeUpdate: "outTimeUpdate", outEnded: "outEnded", outMute: "outMute", outPlay: "outPlay" }, viewQueries: [{ propertyName: "audioRef", first: true, predicate: ["audioRef"], descendants: true, isSignal: true }, { propertyName: "volumeControlRef", first: true, predicate: ["volumeControlRef"], descendants: true, isSignal: true }], ngImport: i0, template: "<audio\n controls\n #audioRef\n class=\"hidden\"\n (timeupdate)=\"handlerTimeUpdate($event)\"\n (loadeddata)=\"handlerLoadedData($event)\"\n (ended)=\"handlerEnded($event)\">\n <source\n [src]=\"fileAudio()\"\n type=\"audio/mpeg\" />\n</audio>\n<div\n [class.libs-ui-disable]=\"isDisable()\"\n [class.pointer-events-none]=\"isDisable()\">\n <div class=\"flex justify-between items-center\">\n <div class=\"w-[70%] flex p-0 items-center\">\n <div\n class=\"flex mr-[16px] cursor-pointer\"\n (click)=\"handlerAudioPausePlay($event)\">\n <i\n class=\"text-[16px]\"\n [class.libs-ui-icon-play-solid]=\"!isPlay()\"\n [class.libs-ui-icon-pause-solid]=\"isPlay()\"></i>\n </div>\n <div class=\"libs-ui-font-h5r mr-[16px]\">{{ audioTimeCurrent() }} /{{ audioTimeDuration() }}</div>\n </div>\n <div class=\"w-[30%] flex p-0 items-center justify-end\">\n <div\n #volumeControlRef\n class=\"flex py-[3px] items-center rounded-[12px] h-[28px]\"\n [class.bg-[#e6e7ea]]=\"showFullControlVolume()\"\n [class.px-[12px]]=\"showFullControlVolume()\">\n <i\n class=\"text-[16px] cursor-pointer\"\n [class.libs-ui-icon-speaker-on-solid]=\"!isMute()\"\n [class.libs-ui-icon-speaker-off-solid]=\"isMute()\"\n (click)=\"handlerAudioMuteMuted($event)\"></i>\n <libs_ui-components-inputs-range_slider\n [class.hidden]=\"!showFullControlVolume()\"\n [mode]=\"'audio'\"\n classInclude=\"flex items-center !w-[54px] cursor-pointer ml-[8px]\"\n [value]=\"volumeRatioValue()\"\n (outChange)=\"handlerChangeVolume($event)\" />\n </div>\n\n <i\n class=\"libs-ui-icon-download-solid ml-[16px] cursor-pointer\"\n (click)=\"handlerDownload($event)\"></i>\n </div>\n </div>\n <div class=\"h-[24px]\">\n <libs_ui-components-inputs-range_slider\n [mode]=\"'audio'\"\n [value]=\"audioRatioValue()\"\n [disable]=\"isDisable()\"\n (outChange)=\"handlerChangeAudio($event)\" />\n </div>\n</div>\n", dependencies: [{ kind: "component", type: LibsUiComponentsInputsRangeSliderComponent, selector: "libs_ui-components-inputs-range_slider", inputs: ["mode", "min", "max", "value", "classInclude", "disable", "unit", "step", "hideProgressingValue", "formatNumber"], outputs: ["valueChange", "outChange"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
@@ -192,4 +223,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
192
223
  type: Component,
193
224
  args: [{ selector: 'libs_ui-components-audio', standalone: true, changeDetection: ChangeDetectionStrategy.OnPush, imports: [LibsUiComponentsInputsRangeSliderComponent], template: "<audio\n controls\n #audioRef\n class=\"hidden\"\n (timeupdate)=\"handlerTimeUpdate($event)\"\n (loadeddata)=\"handlerLoadedData($event)\"\n (ended)=\"handlerEnded($event)\">\n <source\n [src]=\"fileAudio()\"\n type=\"audio/mpeg\" />\n</audio>\n<div\n [class.libs-ui-disable]=\"isDisable()\"\n [class.pointer-events-none]=\"isDisable()\">\n <div class=\"flex justify-between items-center\">\n <div class=\"w-[70%] flex p-0 items-center\">\n <div\n class=\"flex mr-[16px] cursor-pointer\"\n (click)=\"handlerAudioPausePlay($event)\">\n <i\n class=\"text-[16px]\"\n [class.libs-ui-icon-play-solid]=\"!isPlay()\"\n [class.libs-ui-icon-pause-solid]=\"isPlay()\"></i>\n </div>\n <div class=\"libs-ui-font-h5r mr-[16px]\">{{ audioTimeCurrent() }} /{{ audioTimeDuration() }}</div>\n </div>\n <div class=\"w-[30%] flex p-0 items-center justify-end\">\n <div\n #volumeControlRef\n class=\"flex py-[3px] items-center rounded-[12px] h-[28px]\"\n [class.bg-[#e6e7ea]]=\"showFullControlVolume()\"\n [class.px-[12px]]=\"showFullControlVolume()\">\n <i\n class=\"text-[16px] cursor-pointer\"\n [class.libs-ui-icon-speaker-on-solid]=\"!isMute()\"\n [class.libs-ui-icon-speaker-off-solid]=\"isMute()\"\n (click)=\"handlerAudioMuteMuted($event)\"></i>\n <libs_ui-components-inputs-range_slider\n [class.hidden]=\"!showFullControlVolume()\"\n [mode]=\"'audio'\"\n classInclude=\"flex items-center !w-[54px] cursor-pointer ml-[8px]\"\n [value]=\"volumeRatioValue()\"\n (outChange)=\"handlerChangeVolume($event)\" />\n </div>\n\n <i\n class=\"libs-ui-icon-download-solid ml-[16px] cursor-pointer\"\n (click)=\"handlerDownload($event)\"></i>\n </div>\n </div>\n <div class=\"h-[24px]\">\n <libs_ui-components-inputs-range_slider\n [mode]=\"'audio'\"\n [value]=\"audioRatioValue()\"\n [disable]=\"isDisable()\"\n (outChange)=\"handlerChangeAudio($event)\" />\n </div>\n</div>\n" }]
194
225
  }], ctorParameters: () => [] });
195
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audio.component.js","sourceRoot":"","sources":["../../../../../libs-ui/components/audio/src/audio.component.ts","../../../../../libs-ui/components/audio/src/audio.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, ChangeDetectionStrategy, Component, DestroyRef, effect, ElementRef, inject, input, output, signal, viewChild } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { LibsUiComponentsInputsRangeSliderComponent } from '@libs-ui/components-inputs-range-slider';\nimport { fromEvent, merge, Observable, tap } from 'rxjs';\nimport { IAudioFunctionControlEvent } from './interfaces/function-control-event.interface';\n\n@Component({\n  // eslint-disable-next-line @angular-eslint/component-selector\n  selector: 'libs_ui-components-audio',\n  templateUrl: './audio.component.html',\n  standalone: true,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [LibsUiComponentsInputsRangeSliderComponent],\n})\nexport class LibsUiComponentsAudioComponent implements AfterViewInit {\n  // #region PROPERTY\n  protected audioRatioValue = signal<number>(0);\n  protected volumeRatioValue = signal<number>(100);\n  protected isPlay = signal<boolean>(false);\n  protected isMute = signal<boolean>(false);\n  protected isSliderAudioPress = signal<boolean>(false);\n  protected isDisable = signal<boolean>(true);\n  protected audioTimeCurrent = signal<string>('_:_:_');\n  protected audioTimeDuration = signal<string>('_:_:_');\n  protected showFullControlVolume = signal<boolean>(false);\n\n  // #region INPUT\n  readonly fileAudio = input.required<string>();\n  readonly checkPermissionDownloadAudio = input.required<() => Promise<boolean>>();\n\n  /* VIEW CHILD */\n  readonly audioRef = viewChild.required<ElementRef>('audioRef');\n  readonly volumeControlRef = viewChild.required<ElementRef>('volumeControlRef');\n\n  /* OUTPUTS */\n  readonly outFunctionsControl = output<IAudioFunctionControlEvent>();\n  readonly outVolumeControl = output<number>();\n  readonly outTimeUpdate = output<{ currentTime: string; duration: string }>();\n  readonly outEnded = output<void>();\n  readonly outMute = output<boolean>();\n  readonly outPlay = output<boolean>();\n\n  private destroyRef = inject(DestroyRef);\n\n  constructor() {\n    // Watch for file audio changes\n    effect(() => {\n      if (this.fileAudio() && this.audioRef()) {\n        // Skip initial setup, only reload on changes\n        setTimeout(() => {\n          this.audioRef().nativeElement.load();\n        }, 0);\n      }\n    });\n    effect(() => {\n      this.outVolumeControl.emit(this.volumeRatioValue());\n    });\n    effect(() => {\n      this.outTimeUpdate.emit({ currentTime: this.audioTimeCurrent(), duration: this.audioTimeDuration() });\n    });\n    effect(() => {\n      this.outMute.emit(this.isMute());\n    });\n    effect(() => {\n      this.outPlay.emit(this.isPlay());\n    });\n  }\n\n  ngAfterViewInit() {\n    merge(\n      this.initObservable(this.volumeControlRef().nativeElement, 'mouseenter').pipe(tap(() => this.showFullControlVolume.set(true))),\n      this.initObservable(this.volumeControlRef().nativeElement, 'mouseleave').pipe(tap(() => this.showFullControlVolume.set(false)))\n    )\n      .pipe(takeUntilDestroyed(this.destroyRef))\n      .subscribe();\n\n    // Emit function control event after view is initialized\n    this.outFunctionsControl.emit(this.FunctionsControl);\n  }\n\n  public get FunctionsControl(): IAudioFunctionControlEvent {\n    return {\n      playPause: (event?: Event) => this.handlerAudioPausePlay(event),\n      toggleMute: (event?: Event) => this.handlerAudioMuteMuted(event),\n      seekTo: this.handlerChangeAudio.bind(this),\n      setVolume: this.handlerChangeVolume.bind(this),\n      download: (event?: Event) => this.handlerDownload(event),\n      isPlaying: () => this.isPlay(),\n      isMuted: () => this.isMute(),\n    };\n  }\n\n  /* FUNCTIONS */\n  private initObservable(el: HTMLElement, eventName: string): Observable<MouseEvent> {\n    return fromEvent<MouseEvent>(el, eventName).pipe(\n      tap((e) => e.stopPropagation()),\n      takeUntilDestroyed(this.destroyRef)\n    );\n  }\n\n  protected async handlerKeyPressAudio() {\n    this.isSliderAudioPress.set(true);\n  }\n\n  protected async handlerAudioMuteMuted(event?: Event) {\n    if (event) {\n      event.stopPropagation();\n    }\n\n    if (this.audioRef().nativeElement.muted === true) {\n      this.audioRef().nativeElement.muted = false;\n      this.isMute.set(false);\n      this.volumeRatioValue.set(50);\n\n      return;\n    }\n    this.volumeRatioValue.set(0);\n    this.isMute.set(true);\n    this.audioRef().nativeElement.muted = true;\n  }\n\n  protected async handlerAudioPausePlay(event?: Event) {\n    if (event) {\n      event.stopPropagation();\n    }\n\n    const audioElement = this.audioRef().nativeElement;\n    if (!audioElement.paused) {\n      audioElement.pause();\n      this.isPlay.set(false);\n      return;\n    }\n\n    try {\n      await audioElement.play();\n      this.isPlay.set(true);\n    } catch (error) {\n      console.error('Error playing audio:', error);\n    }\n  }\n\n  protected async handlerLoadedData(event: Event) {\n    if (event) {\n      event.stopPropagation();\n    }\n\n    if (this.audioRef().nativeElement) {\n      this.audioTimeDuration.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.duration)));\n      this.audioTimeCurrent.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.currentTime || 0)));\n      this.isDisable.set(false);\n      this.isPlay.set(false);\n      this.audioRatioValue.set(0);\n      this.audioRef().nativeElement.pause();\n    }\n  }\n\n  protected async handlerTimeUpdate(event?: Event) {\n    if (event) {\n      event.stopPropagation();\n    }\n    this.isDisable.set(!(this.audioRef().nativeElement.duration || 0));\n    if (!this.audioRef().nativeElement) {\n      return;\n    }\n    this.audioTimeDuration.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.duration)));\n    this.audioTimeCurrent.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.currentTime || 0)));\n    if (this.isSliderAudioPress()) {\n      this.audioRef().nativeElement.currentTime = (this.audioRatioValue() * Math.floor(this.audioRef().nativeElement.duration || 0)) / 100;\n\n      return;\n    }\n    this.audioRatioValue.set(Math.floor(((this.audioRef().nativeElement.currentTime || 0) / (this.audioRef().nativeElement.duration || 1)) * 100));\n  }\n\n  private async toHHMMSS(time: number) {\n    const hours = Math.floor(time / 3600);\n    const minutes = Math.floor((time - hours * 3600) / 60);\n    const seconds = time - hours * 3600 - minutes * 60;\n\n    const getLabel = (val: number) => {\n      val = val || 0;\n      return `${val < 10 ? '0' : ''}${val}`;\n    };\n\n    return `${getLabel(hours)}:${getLabel(minutes)}:${getLabel(seconds)}`;\n  }\n\n  protected async handlerChangeAudio(value: number) {\n    if (value === this.audioRatioValue()) {\n      return;\n    }\n    this.audioRef().nativeElement.currentTime = ((value || 0) * (this.audioRef().nativeElement.duration || 0)) / 100;\n    this.audioRatioValue.set(value);\n    this.isSliderAudioPress.set(false);\n  }\n\n  protected async handlerChangeVolume(value: number) {\n    this.audioRef().nativeElement.volume = value / 100;\n    this.volumeRatioValue.set(value);\n    if (this.audioRef().nativeElement.volume) {\n      this.audioRef().nativeElement.muted = false;\n      this.isMute.set(false);\n\n      return;\n    }\n    this.audioRef().nativeElement.muted = true;\n    this.isMute.set(true);\n  }\n\n  protected async handlerEnded(event: Event) {\n    if (event) {\n      event.stopPropagation();\n    }\n\n    this.isPlay.set(false);\n    this.outEnded.emit();\n  }\n\n  protected async handlerDownload(e?: Event) {\n    if (!this.checkPermissionDownloadAudio() || !(await this.checkPermissionDownloadAudio()())) {\n      return;\n    }\n\n    if (e) {\n      e.stopPropagation();\n    }\n\n    if (!this.fileAudio()) {\n      return;\n    }\n    window.open(this.fileAudio(), `_blank`);\n  }\n}\n","<audio\n  controls\n  #audioRef\n  class=\"hidden\"\n  (timeupdate)=\"handlerTimeUpdate($event)\"\n  (loadeddata)=\"handlerLoadedData($event)\"\n  (ended)=\"handlerEnded($event)\">\n  <source\n    [src]=\"fileAudio()\"\n    type=\"audio/mpeg\" />\n</audio>\n<div\n  [class.libs-ui-disable]=\"isDisable()\"\n  [class.pointer-events-none]=\"isDisable()\">\n  <div class=\"flex justify-between items-center\">\n    <div class=\"w-[70%] flex p-0 items-center\">\n      <div\n        class=\"flex mr-[16px] cursor-pointer\"\n        (click)=\"handlerAudioPausePlay($event)\">\n        <i\n          class=\"text-[16px]\"\n          [class.libs-ui-icon-play-solid]=\"!isPlay()\"\n          [class.libs-ui-icon-pause-solid]=\"isPlay()\"></i>\n      </div>\n      <div class=\"libs-ui-font-h5r mr-[16px]\">{{ audioTimeCurrent() }} /{{ audioTimeDuration() }}</div>\n    </div>\n    <div class=\"w-[30%] flex p-0 items-center justify-end\">\n      <div\n        #volumeControlRef\n        class=\"flex py-[3px] items-center rounded-[12px] h-[28px]\"\n        [class.bg-[#e6e7ea]]=\"showFullControlVolume()\"\n        [class.px-[12px]]=\"showFullControlVolume()\">\n        <i\n          class=\"text-[16px] cursor-pointer\"\n          [class.libs-ui-icon-speaker-on-solid]=\"!isMute()\"\n          [class.libs-ui-icon-speaker-off-solid]=\"isMute()\"\n          (click)=\"handlerAudioMuteMuted($event)\"></i>\n        <libs_ui-components-inputs-range_slider\n          [class.hidden]=\"!showFullControlVolume()\"\n          [mode]=\"'audio'\"\n          classInclude=\"flex items-center !w-[54px] cursor-pointer ml-[8px]\"\n          [value]=\"volumeRatioValue()\"\n          (outChange)=\"handlerChangeVolume($event)\" />\n      </div>\n\n      <i\n        class=\"libs-ui-icon-download-solid ml-[16px] cursor-pointer\"\n        (click)=\"handlerDownload($event)\"></i>\n    </div>\n  </div>\n  <div class=\"h-[24px]\">\n    <libs_ui-components-inputs-range_slider\n      [mode]=\"'audio'\"\n      [value]=\"audioRatioValue()\"\n      [disable]=\"isDisable()\"\n      (outChange)=\"handlerChangeAudio($event)\" />\n  </div>\n</div>\n"]}
226
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audio.component.js","sourceRoot":"","sources":["../../../../../libs-ui/components/audio/src/audio.component.ts","../../../../../libs-ui/components/audio/src/audio.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, ChangeDetectionStrategy, Component, DestroyRef, effect, ElementRef, inject, input, output, signal, viewChild } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { LibsUiComponentsInputsRangeSliderComponent } from '@libs-ui/components-inputs-range-slider';\nimport { fromEvent, merge, Observable, tap } from 'rxjs';\nimport { IAudioFunctionControlEvent } from './interfaces/function-control-event.interface';\n@Component({\n  // eslint-disable-next-line @angular-eslint/component-selector\n  selector: 'libs_ui-components-audio',\n  templateUrl: './audio.component.html',\n  standalone: true,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [LibsUiComponentsInputsRangeSliderComponent],\n})\nexport class LibsUiComponentsAudioComponent implements AfterViewInit {\n  // =========================================\n  // INTERNAL SIGNALS\n  // =========================================\n\n  protected audioRatioValue = signal<number>(0);\n\n  protected volumeRatioValue = signal<number>(100);\n\n  protected isPlay = signal<boolean>(false);\n\n  protected isMute = signal<boolean>(false);\n\n  protected isSliderAudioPress = signal<boolean>(false);\n\n  protected isDisable = signal<boolean>(true);\n\n  protected audioTimeCurrent = signal<string>('_:_:_');\n\n  protected audioTimeDuration = signal<string>('_:_:_');\n\n  protected showFullControlVolume = signal<boolean>(false);\n\n  // =========================================\n  // INPUTS\n  // =========================================\n\n  readonly fileAudio = input.required<string>();\n\n  readonly checkPermissionDownloadAudio = input.required<() => Promise<boolean>>();\n\n  // =========================================\n  // VIEW CHILDREN\n  // =========================================\n\n  readonly audioRef = viewChild.required<ElementRef>('audioRef');\n\n  readonly volumeControlRef = viewChild.required<ElementRef>('volumeControlRef');\n\n  // =========================================\n  // OUTPUTS\n  // =========================================\n\n  readonly outFunctionsControl = output<IAudioFunctionControlEvent>();\n\n  readonly outVolumeControl = output<number>();\n\n  readonly outTimeUpdate = output<{ currentTime: string; duration: string }>();\n\n  readonly outEnded = output<void>();\n\n  readonly outMute = output<boolean>();\n\n  readonly outPlay = output<boolean>();\n\n  // =========================================\n  // PRIVATE PROPERTIES\n  // =========================================\n\n  private destroyRef = inject(DestroyRef);\n\n  // =========================================\n  // CONSTRUCTOR\n  // =========================================\n\n  constructor() {\n    // Watch for file audio changes\n    effect(() => {\n      if (this.fileAudio() && this.audioRef()) {\n        setTimeout(() => {\n          this.audioRef().nativeElement.load();\n        }, 0);\n      }\n    });\n\n    effect(() => {\n      this.outVolumeControl.emit(this.volumeRatioValue());\n    });\n\n    effect(() => {\n      this.outTimeUpdate.emit({ currentTime: this.audioTimeCurrent(), duration: this.audioTimeDuration() });\n    });\n\n    effect(() => {\n      this.outMute.emit(this.isMute());\n    });\n\n    effect(() => {\n      this.outPlay.emit(this.isPlay());\n    });\n  }\n\n  // =========================================\n  // LIFECYCLE HOOKS\n  // =========================================\n\n  ngAfterViewInit(): void {\n    merge(\n      this.initObservable(this.volumeControlRef().nativeElement, 'mouseenter').pipe(tap(() => this.showFullControlVolume.set(true))),\n      this.initObservable(this.volumeControlRef().nativeElement, 'mouseleave').pipe(tap(() => this.showFullControlVolume.set(false)))\n    )\n      .pipe(takeUntilDestroyed(this.destroyRef))\n      .subscribe();\n\n    // Emit function control event after view is initialized\n    this.outFunctionsControl.emit(this.FunctionsControl);\n  }\n\n  // =========================================\n  // PUBLIC API\n  // =========================================\n\n  public get FunctionsControl(): IAudioFunctionControlEvent {\n    return {\n      playPause: (event?: Event) => this.handlerAudioPausePlay(event),\n      toggleMute: (event?: Event) => this.handlerAudioMuteMuted(event),\n      seekTo: this.handlerChangeAudio.bind(this),\n      setVolume: this.handlerChangeVolume.bind(this),\n      download: (event?: Event) => this.handlerDownload(event),\n      isPlaying: () => this.isPlay(),\n      isMuted: () => this.isMute(),\n    };\n  }\n\n  // =========================================\n  // PRIVATE METHODS\n  // =========================================\n\n  private initObservable(el: HTMLElement, eventName: string): Observable<MouseEvent> {\n    return fromEvent<MouseEvent>(el, eventName).pipe(\n      tap((e) => e.stopPropagation()),\n      takeUntilDestroyed(this.destroyRef)\n    );\n  }\n\n  protected async handlerKeyPressAudio(): Promise<void> {\n    this.isSliderAudioPress.set(true);\n  }\n\n  protected async handlerAudioMuteMuted(event?: Event): Promise<void> {\n    if (event) {\n      event.stopPropagation();\n    }\n\n    if (this.audioRef().nativeElement.muted === true) {\n      this.audioRef().nativeElement.muted = false;\n      this.isMute.set(false);\n      this.volumeRatioValue.set(50);\n      return;\n    }\n\n    this.volumeRatioValue.set(0);\n    this.isMute.set(true);\n    this.audioRef().nativeElement.muted = true;\n  }\n\n  protected async handlerAudioPausePlay(event?: Event): Promise<void> {\n    if (event) {\n      event.stopPropagation();\n    }\n\n    const audioElement = this.audioRef().nativeElement;\n    if (!audioElement.paused) {\n      audioElement.pause();\n      this.isPlay.set(false);\n      return;\n    }\n\n    try {\n      await audioElement.play();\n      this.isPlay.set(true);\n    } catch (error) {\n      console.error('Error playing audio:', error);\n    }\n  }\n\n  protected async handlerLoadedData(event: Event): Promise<void> {\n    if (event) {\n      event.stopPropagation();\n    }\n\n    if (this.audioRef().nativeElement) {\n      this.audioTimeDuration.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.duration)));\n      this.audioTimeCurrent.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.currentTime || 0)));\n      this.isDisable.set(false);\n      this.isPlay.set(false);\n      this.audioRatioValue.set(0);\n      this.audioRef().nativeElement.pause();\n    }\n  }\n\n  protected async handlerTimeUpdate(event?: Event): Promise<void> {\n    if (event) {\n      event.stopPropagation();\n    }\n\n    this.isDisable.set(!(this.audioRef().nativeElement.duration || 0));\n\n    if (!this.audioRef().nativeElement) {\n      return;\n    }\n\n    this.audioTimeDuration.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.duration)));\n    this.audioTimeCurrent.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.currentTime || 0)));\n\n    if (this.isSliderAudioPress()) {\n      this.audioRef().nativeElement.currentTime = (this.audioRatioValue() * Math.floor(this.audioRef().nativeElement.duration || 0)) / 100;\n      return;\n    }\n\n    this.audioRatioValue.set(Math.floor(((this.audioRef().nativeElement.currentTime || 0) / (this.audioRef().nativeElement.duration || 1)) * 100));\n  }\n\n  /**\n   * Format seconds -> HH:MM:SS.\n   * Used for both `currentTime` and `duration` outputs to keep output stable.\n   */\n  private async toHHMMSS(time: number): Promise<string> {\n    const hours = Math.floor(time / 3600);\n    const minutes = Math.floor((time - hours * 3600) / 60);\n    const seconds = time - hours * 3600 - minutes * 60;\n\n    const getLabel = (val: number) => {\n      val = val || 0;\n      return `${val < 10 ? '0' : ''}${val}`;\n    };\n\n    return `${getLabel(hours)}:${getLabel(minutes)}:${getLabel(seconds)}`;\n  }\n\n  protected async handlerChangeAudio(value: number): Promise<void> {\n    if (value === this.audioRatioValue()) {\n      return;\n    }\n\n    this.audioRef().nativeElement.currentTime = ((value || 0) * (this.audioRef().nativeElement.duration || 0)) / 100;\n    this.audioRatioValue.set(value);\n    this.isSliderAudioPress.set(false);\n  }\n\n  protected async handlerChangeVolume(value: number): Promise<void> {\n    this.audioRef().nativeElement.volume = value / 100;\n    this.volumeRatioValue.set(value);\n\n    if (this.audioRef().nativeElement.volume) {\n      this.audioRef().nativeElement.muted = false;\n      this.isMute.set(false);\n      return;\n    }\n\n    this.audioRef().nativeElement.muted = true;\n    this.isMute.set(true);\n  }\n\n  protected async handlerEnded(event: Event): Promise<void> {\n    if (event) {\n      event.stopPropagation();\n    }\n\n    this.isPlay.set(false);\n    this.outEnded.emit();\n  }\n\n  /**\n   * Download chỉ được thực hiện nếu callback permission trả về `true`.\n   * Lưu ý: `checkPermissionDownloadAudio` là input function (factory), nên cần gọi 2 lần:\n   * - lần 1: lấy function\n   * - lần 2: execute function để lấy Promise<boolean>\n   */\n  protected async handlerDownload(e?: Event): Promise<void> {\n    if (!this.checkPermissionDownloadAudio() || !(await this.checkPermissionDownloadAudio()())) {\n      return;\n    }\n\n    if (e) {\n      e.stopPropagation();\n    }\n\n    if (!this.fileAudio()) {\n      return;\n    }\n\n    window.open(this.fileAudio(), '_blank');\n  }\n}\n","<audio\n  controls\n  #audioRef\n  class=\"hidden\"\n  (timeupdate)=\"handlerTimeUpdate($event)\"\n  (loadeddata)=\"handlerLoadedData($event)\"\n  (ended)=\"handlerEnded($event)\">\n  <source\n    [src]=\"fileAudio()\"\n    type=\"audio/mpeg\" />\n</audio>\n<div\n  [class.libs-ui-disable]=\"isDisable()\"\n  [class.pointer-events-none]=\"isDisable()\">\n  <div class=\"flex justify-between items-center\">\n    <div class=\"w-[70%] flex p-0 items-center\">\n      <div\n        class=\"flex mr-[16px] cursor-pointer\"\n        (click)=\"handlerAudioPausePlay($event)\">\n        <i\n          class=\"text-[16px]\"\n          [class.libs-ui-icon-play-solid]=\"!isPlay()\"\n          [class.libs-ui-icon-pause-solid]=\"isPlay()\"></i>\n      </div>\n      <div class=\"libs-ui-font-h5r mr-[16px]\">{{ audioTimeCurrent() }} /{{ audioTimeDuration() }}</div>\n    </div>\n    <div class=\"w-[30%] flex p-0 items-center justify-end\">\n      <div\n        #volumeControlRef\n        class=\"flex py-[3px] items-center rounded-[12px] h-[28px]\"\n        [class.bg-[#e6e7ea]]=\"showFullControlVolume()\"\n        [class.px-[12px]]=\"showFullControlVolume()\">\n        <i\n          class=\"text-[16px] cursor-pointer\"\n          [class.libs-ui-icon-speaker-on-solid]=\"!isMute()\"\n          [class.libs-ui-icon-speaker-off-solid]=\"isMute()\"\n          (click)=\"handlerAudioMuteMuted($event)\"></i>\n        <libs_ui-components-inputs-range_slider\n          [class.hidden]=\"!showFullControlVolume()\"\n          [mode]=\"'audio'\"\n          classInclude=\"flex items-center !w-[54px] cursor-pointer ml-[8px]\"\n          [value]=\"volumeRatioValue()\"\n          (outChange)=\"handlerChangeVolume($event)\" />\n      </div>\n\n      <i\n        class=\"libs-ui-icon-download-solid ml-[16px] cursor-pointer\"\n        (click)=\"handlerDownload($event)\"></i>\n    </div>\n  </div>\n  <div class=\"h-[24px]\">\n    <libs_ui-components-inputs-range_slider\n      [mode]=\"'audio'\"\n      [value]=\"audioRatioValue()\"\n      [disable]=\"isDisable()\"\n      (outChange)=\"handlerChangeAudio($event)\" />\n  </div>\n</div>\n"]}
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZnVuY3Rpb24tY29udHJvbC1ldmVudC5pbnRlcmZhY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9saWJzLXVpL2NvbXBvbmVudHMvYXVkaW8vc3JjL2ludGVyZmFjZXMvZnVuY3Rpb24tY29udHJvbC1ldmVudC5pbnRlcmZhY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogSW50ZXJmYWNlIGNobyBjw6FjIGNo4bupYyBuxINuZyDEkWnhu4F1IGtoaeG7g24gYXVkaW8gxJHGsOG7o2MgY3VuZyBj4bqlcCBxdWEgb3V0cHV0IGV2ZW50XG4gKiBAZGVzY3JpcHRpb24gQ3VuZyBj4bqlcCBjw6FjIG1ldGhvZCDEkeG7gyDEkWnhu4F1IGtoaeG7g24gYXVkaW8gcGxheWVyIHThu6sgY29tcG9uZW50IGNoYVxuICovXG5leHBvcnQgaW50ZXJmYWNlIElBdWRpb0Z1bmN0aW9uQ29udHJvbEV2ZW50IHtcbiAgLyoqXG4gICAqIEtleSDEkeG7gyDEkWnhu4F1IGtoaeG7g24gYXVkaW9cbiAgICogQHJldHVybnMgdm9pZFxuICAgKi9cblxuICAvKipcbiAgICogQuG6r3QgxJHhuqd1IGhv4bq3YyB04bqhbSBk4burbmcgcGjDoXQgYXVkaW9cbiAgICogQHBhcmFtIGV2ZW50IE9wdGlvbmFsIGV2ZW50IG9iamVjdFxuICAgKiBAcmV0dXJucyB2b2lkXG4gICAqL1xuICBwbGF5UGF1c2U6IChldmVudD86IEV2ZW50KSA9PiB2b2lkO1xuXG4gIC8qKlxuICAgKiBC4bqtdCBob+G6t2MgdOG6r3Qgw6JtIHRoYW5oXG4gICAqIEBwYXJhbSBldmVudCBPcHRpb25hbCBldmVudCBvYmplY3RcbiAgICogQHJldHVybnMgdm9pZFxuICAgKi9cbiAgdG9nZ2xlTXV0ZTogKGV2ZW50PzogRXZlbnQpID0+IHZvaWQ7XG5cbiAgLyoqXG4gICAqIMSQaeG7gXUgY2jhu4luaCDDom0gbMaw4bujbmdcbiAgICogQHBhcmFtIHZhbHVlIEdpw6EgdHLhu4sgw6JtIGzGsOG7o25nIHThu6sgMCDEkeG6v24gMTAwXG4gICAqIEByZXR1cm5zIHZvaWRcbiAgICovXG4gIHNldFZvbHVtZTogKHZhbHVlOiBudW1iZXIpID0+IHZvaWQ7XG5cbiAgLyoqXG4gICAqIERpIGNodXnhu4NuIMSR4bq/biB24buLIHRyw60gY+G7pSB0aOG7gyB0cm9uZyBhdWRpb1xuICAgKiBAcGFyYW0gdmFsdWUgR2nDoSB0cuG7iyBwaOG6p24gdHLEg20gdOG7qyAwIMSR4bq/biAxMDBcbiAgICogQHJldHVybnMgdm9pZFxuICAgKi9cbiAgc2Vla1RvOiAodmFsdWU6IG51bWJlcikgPT4gdm9pZDtcblxuICAvKipcbiAgICogVOG6o2kgeHXhu5FuZyBmaWxlIGF1ZGlvXG4gICAqIEBwYXJhbSBldmVudCBPcHRpb25hbCBldmVudCBvYmplY3RcbiAgICogQHJldHVybnMgdm9pZFxuICAgKi9cbiAgZG93bmxvYWQ6IChldmVudD86IEV2ZW50KSA9PiB2b2lkO1xuXG4gIC8qKlxuICAgKiBLaeG7g20gdHJhIHRy4bqhbmcgdGjDoWkgxJFhbmcgcGjDoXQgYXVkaW9cbiAgICogQHJldHVybnMgYm9vbGVhbiBUcnVlIG7hur91IMSRYW5nIHBow6F0LCBGYWxzZSBu4bq/dSDEkWFuZyB04bqhbSBk4burbmdcbiAgICovXG4gIGlzUGxheWluZzogKCkgPT4gYm9vbGVhbjtcblxuICAvKipcbiAgICogS2nhu4NtIHRyYSB0cuG6oW5nIHRow6FpIHThuq90IHRp4bq/bmdcbiAgICogQHJldHVybnMgYm9vbGVhbiBUcnVlIG7hur91IMSRYW5nIHThuq90IHRp4bq/bmcsIEZhbHNlIG7hur91IMSRYW5nIGLhuq10IHRp4bq/bmdcbiAgICovXG4gIGlzTXV0ZWQ6ICgpID0+IGJvb2xlYW47XG59XG4iXX0=
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZnVuY3Rpb24tY29udHJvbC1ldmVudC5pbnRlcmZhY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9saWJzLXVpL2NvbXBvbmVudHMvYXVkaW8vc3JjL2ludGVyZmFjZXMvZnVuY3Rpb24tY29udHJvbC1ldmVudC5pbnRlcmZhY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogUHVibGljIGNvbnRyb2wgc3VyZmFjZSDEkeG7gyDEkWnhu4F1IGtoaeG7g24gYXVkaW8gdOG7qyBjb21wb25lbnQgY2hhLlxuICogQ2jhu4kgbcO0IHThuqMgbmjhu69uZyDEkWnhu4NtIGThu4UgZMO5bmcgc2FpIChuaMawIHJhbmdlL3ZhbHVlKS5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBJQXVkaW9GdW5jdGlvbkNvbnRyb2xFdmVudCB7XG4gIC8qKlxuICAgKiBC4bqvdCDEkeG6p3UgaG/hurdjIHThuqFtIGThu6tuZyBwaMOhdCBhdWRpb1xuICAgKiBAcGFyYW0gZXZlbnQgT3B0aW9uYWwgZXZlbnQgb2JqZWN0XG4gICAqL1xuICBwbGF5UGF1c2U6IChldmVudD86IEV2ZW50KSA9PiB2b2lkO1xuXG4gIC8qKlxuICAgKiBC4bqtdCBob+G6t2MgdOG6r3Qgw6JtIHRoYW5oXG4gICAqIEBwYXJhbSBldmVudCBPcHRpb25hbCBldmVudCBvYmplY3RcbiAgICovXG4gIHRvZ2dsZU11dGU6IChldmVudD86IEV2ZW50KSA9PiB2b2lkO1xuXG4gIC8qKlxuICAgKiDEkGnhu4F1IGNo4buJbmggw6JtIGzGsOG7o25nXG4gICAqIEBwYXJhbSB2YWx1ZSBHacOhIHRy4buLIMOibSBsxrDhu6NuZyB04burIDAgxJHhur9uIDEwMFxuICAgKi9cbiAgc2V0Vm9sdW1lOiAodmFsdWU6IG51bWJlcikgPT4gdm9pZDtcblxuICAvKipcbiAgICogRGkgY2h1eeG7g24gxJHhur9uIHbhu4sgdHLDrSBj4bulIHRo4buDIHRyb25nIGF1ZGlvXG4gICAqIEBwYXJhbSB2YWx1ZSBHacOhIHRy4buLIHBo4bqnbiB0csSDbSB04burIDAgxJHhur9uIDEwMFxuICAgKi9cbiAgc2Vla1RvOiAodmFsdWU6IG51bWJlcikgPT4gdm9pZDtcblxuICAvKipcbiAgICogVOG6o2kgeHXhu5FuZyBmaWxlIGF1ZGlvXG4gICAqIEBwYXJhbSBldmVudCBPcHRpb25hbCBldmVudCBvYmplY3RcbiAgICovXG4gIGRvd25sb2FkOiAoZXZlbnQ/OiBFdmVudCkgPT4gdm9pZDtcblxuICAvKipcbiAgICogS2nhu4NtIHRyYSB0cuG6oW5nIHRow6FpIMSRYW5nIHBow6F0IGF1ZGlvXG4gICAqL1xuICBpc1BsYXlpbmc6ICgpID0+IGJvb2xlYW47XG5cbiAgLyoqXG4gICAqIEtp4buDbSB0cmEgdHLhuqFuZyB0aMOhaSB04bqvdCB0aeG6v25nXG4gICAqL1xuICBpc011dGVkOiAoKSA9PiBib29sZWFuO1xufVxuIl19
@@ -5,7 +5,9 @@ import { LibsUiComponentsInputsRangeSliderComponent } from '@libs-ui/components-
5
5
  import { merge, tap, fromEvent } from 'rxjs';
6
6
 
7
7
  class LibsUiComponentsAudioComponent {
8
- // #region PROPERTY
8
+ // =========================================
9
+ // INTERNAL SIGNALS
10
+ // =========================================
9
11
  audioRatioValue = signal(0);
10
12
  volumeRatioValue = signal(100);
11
13
  isPlay = signal(false);
@@ -15,25 +17,36 @@ class LibsUiComponentsAudioComponent {
15
17
  audioTimeCurrent = signal('_:_:_');
16
18
  audioTimeDuration = signal('_:_:_');
17
19
  showFullControlVolume = signal(false);
18
- // #region INPUT
20
+ // =========================================
21
+ // INPUTS
22
+ // =========================================
19
23
  fileAudio = input.required();
20
24
  checkPermissionDownloadAudio = input.required();
21
- /* VIEW CHILD */
25
+ // =========================================
26
+ // VIEW CHILDREN
27
+ // =========================================
22
28
  audioRef = viewChild.required('audioRef');
23
29
  volumeControlRef = viewChild.required('volumeControlRef');
24
- /* OUTPUTS */
30
+ // =========================================
31
+ // OUTPUTS
32
+ // =========================================
25
33
  outFunctionsControl = output();
26
34
  outVolumeControl = output();
27
35
  outTimeUpdate = output();
28
36
  outEnded = output();
29
37
  outMute = output();
30
38
  outPlay = output();
39
+ // =========================================
40
+ // PRIVATE PROPERTIES
41
+ // =========================================
31
42
  destroyRef = inject(DestroyRef);
43
+ // =========================================
44
+ // CONSTRUCTOR
45
+ // =========================================
32
46
  constructor() {
33
47
  // Watch for file audio changes
34
48
  effect(() => {
35
49
  if (this.fileAudio() && this.audioRef()) {
36
- // Skip initial setup, only reload on changes
37
50
  setTimeout(() => {
38
51
  this.audioRef().nativeElement.load();
39
52
  }, 0);
@@ -52,6 +65,9 @@ class LibsUiComponentsAudioComponent {
52
65
  this.outPlay.emit(this.isPlay());
53
66
  });
54
67
  }
68
+ // =========================================
69
+ // LIFECYCLE HOOKS
70
+ // =========================================
55
71
  ngAfterViewInit() {
56
72
  merge(this.initObservable(this.volumeControlRef().nativeElement, 'mouseenter').pipe(tap(() => this.showFullControlVolume.set(true))), this.initObservable(this.volumeControlRef().nativeElement, 'mouseleave').pipe(tap(() => this.showFullControlVolume.set(false))))
57
73
  .pipe(takeUntilDestroyed(this.destroyRef))
@@ -59,6 +75,9 @@ class LibsUiComponentsAudioComponent {
59
75
  // Emit function control event after view is initialized
60
76
  this.outFunctionsControl.emit(this.FunctionsControl);
61
77
  }
78
+ // =========================================
79
+ // PUBLIC API
80
+ // =========================================
62
81
  get FunctionsControl() {
63
82
  return {
64
83
  playPause: (event) => this.handlerAudioPausePlay(event),
@@ -70,7 +89,9 @@ class LibsUiComponentsAudioComponent {
70
89
  isMuted: () => this.isMute(),
71
90
  };
72
91
  }
73
- /* FUNCTIONS */
92
+ // =========================================
93
+ // PRIVATE METHODS
94
+ // =========================================
74
95
  initObservable(el, eventName) {
75
96
  return fromEvent(el, eventName).pipe(tap((e) => e.stopPropagation()), takeUntilDestroyed(this.destroyRef));
76
97
  }
@@ -138,6 +159,10 @@ class LibsUiComponentsAudioComponent {
138
159
  }
139
160
  this.audioRatioValue.set(Math.floor(((this.audioRef().nativeElement.currentTime || 0) / (this.audioRef().nativeElement.duration || 1)) * 100));
140
161
  }
162
+ /**
163
+ * Format seconds -> HH:MM:SS.
164
+ * Used for both `currentTime` and `duration` outputs to keep output stable.
165
+ */
141
166
  async toHHMMSS(time) {
142
167
  const hours = Math.floor(time / 3600);
143
168
  const minutes = Math.floor((time - hours * 3600) / 60);
@@ -174,6 +199,12 @@ class LibsUiComponentsAudioComponent {
174
199
  this.isPlay.set(false);
175
200
  this.outEnded.emit();
176
201
  }
202
+ /**
203
+ * Download chỉ được thực hiện nếu callback permission trả về `true`.
204
+ * Lưu ý: `checkPermissionDownloadAudio` là input function (factory), nên cần gọi 2 lần:
205
+ * - lần 1: lấy function
206
+ * - lần 2: execute function để lấy Promise<boolean>
207
+ */
177
208
  async handlerDownload(e) {
178
209
  if (!this.checkPermissionDownloadAudio() || !(await this.checkPermissionDownloadAudio()())) {
179
210
  return;
@@ -184,7 +215,7 @@ class LibsUiComponentsAudioComponent {
184
215
  if (!this.fileAudio()) {
185
216
  return;
186
217
  }
187
- window.open(this.fileAudio(), `_blank`);
218
+ window.open(this.fileAudio(), '_blank');
188
219
  }
189
220
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LibsUiComponentsAudioComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
190
221
  static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.2.0", version: "18.2.14", type: LibsUiComponentsAudioComponent, isStandalone: true, selector: "libs_ui-components-audio", inputs: { fileAudio: { classPropertyName: "fileAudio", publicName: "fileAudio", isSignal: true, isRequired: true, transformFunction: null }, checkPermissionDownloadAudio: { classPropertyName: "checkPermissionDownloadAudio", publicName: "checkPermissionDownloadAudio", isSignal: true, isRequired: true, transformFunction: null } }, outputs: { outFunctionsControl: "outFunctionsControl", outVolumeControl: "outVolumeControl", outTimeUpdate: "outTimeUpdate", outEnded: "outEnded", outMute: "outMute", outPlay: "outPlay" }, viewQueries: [{ propertyName: "audioRef", first: true, predicate: ["audioRef"], descendants: true, isSignal: true }, { propertyName: "volumeControlRef", first: true, predicate: ["volumeControlRef"], descendants: true, isSignal: true }], ngImport: i0, template: "<audio\n controls\n #audioRef\n class=\"hidden\"\n (timeupdate)=\"handlerTimeUpdate($event)\"\n (loadeddata)=\"handlerLoadedData($event)\"\n (ended)=\"handlerEnded($event)\">\n <source\n [src]=\"fileAudio()\"\n type=\"audio/mpeg\" />\n</audio>\n<div\n [class.libs-ui-disable]=\"isDisable()\"\n [class.pointer-events-none]=\"isDisable()\">\n <div class=\"flex justify-between items-center\">\n <div class=\"w-[70%] flex p-0 items-center\">\n <div\n class=\"flex mr-[16px] cursor-pointer\"\n (click)=\"handlerAudioPausePlay($event)\">\n <i\n class=\"text-[16px]\"\n [class.libs-ui-icon-play-solid]=\"!isPlay()\"\n [class.libs-ui-icon-pause-solid]=\"isPlay()\"></i>\n </div>\n <div class=\"libs-ui-font-h5r mr-[16px]\">{{ audioTimeCurrent() }} /{{ audioTimeDuration() }}</div>\n </div>\n <div class=\"w-[30%] flex p-0 items-center justify-end\">\n <div\n #volumeControlRef\n class=\"flex py-[3px] items-center rounded-[12px] h-[28px]\"\n [class.bg-[#e6e7ea]]=\"showFullControlVolume()\"\n [class.px-[12px]]=\"showFullControlVolume()\">\n <i\n class=\"text-[16px] cursor-pointer\"\n [class.libs-ui-icon-speaker-on-solid]=\"!isMute()\"\n [class.libs-ui-icon-speaker-off-solid]=\"isMute()\"\n (click)=\"handlerAudioMuteMuted($event)\"></i>\n <libs_ui-components-inputs-range_slider\n [class.hidden]=\"!showFullControlVolume()\"\n [mode]=\"'audio'\"\n classInclude=\"flex items-center !w-[54px] cursor-pointer ml-[8px]\"\n [value]=\"volumeRatioValue()\"\n (outChange)=\"handlerChangeVolume($event)\" />\n </div>\n\n <i\n class=\"libs-ui-icon-download-solid ml-[16px] cursor-pointer\"\n (click)=\"handlerDownload($event)\"></i>\n </div>\n </div>\n <div class=\"h-[24px]\">\n <libs_ui-components-inputs-range_slider\n [mode]=\"'audio'\"\n [value]=\"audioRatioValue()\"\n [disable]=\"isDisable()\"\n (outChange)=\"handlerChangeAudio($event)\" />\n </div>\n</div>\n", dependencies: [{ kind: "component", type: LibsUiComponentsInputsRangeSliderComponent, selector: "libs_ui-components-inputs-range_slider", inputs: ["mode", "min", "max", "value", "classInclude", "disable", "unit", "step", "hideProgressingValue", "formatNumber"], outputs: ["valueChange", "outChange"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
@@ -1 +1 @@
1
- {"version":3,"file":"libs-ui-components-audio.mjs","sources":["../../../../../libs-ui/components/audio/src/audio.component.ts","../../../../../libs-ui/components/audio/src/audio.component.html","../../../../../libs-ui/components/audio/src/libs-ui-components-audio.ts"],"sourcesContent":["import { AfterViewInit, ChangeDetectionStrategy, Component, DestroyRef, effect, ElementRef, inject, input, output, signal, viewChild } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { LibsUiComponentsInputsRangeSliderComponent } from '@libs-ui/components-inputs-range-slider';\nimport { fromEvent, merge, Observable, tap } from 'rxjs';\nimport { IAudioFunctionControlEvent } from './interfaces/function-control-event.interface';\n\n@Component({\n // eslint-disable-next-line @angular-eslint/component-selector\n selector: 'libs_ui-components-audio',\n templateUrl: './audio.component.html',\n standalone: true,\n changeDetection: ChangeDetectionStrategy.OnPush,\n imports: [LibsUiComponentsInputsRangeSliderComponent],\n})\nexport class LibsUiComponentsAudioComponent implements AfterViewInit {\n // #region PROPERTY\n protected audioRatioValue = signal<number>(0);\n protected volumeRatioValue = signal<number>(100);\n protected isPlay = signal<boolean>(false);\n protected isMute = signal<boolean>(false);\n protected isSliderAudioPress = signal<boolean>(false);\n protected isDisable = signal<boolean>(true);\n protected audioTimeCurrent = signal<string>('_:_:_');\n protected audioTimeDuration = signal<string>('_:_:_');\n protected showFullControlVolume = signal<boolean>(false);\n\n // #region INPUT\n readonly fileAudio = input.required<string>();\n readonly checkPermissionDownloadAudio = input.required<() => Promise<boolean>>();\n\n /* VIEW CHILD */\n readonly audioRef = viewChild.required<ElementRef>('audioRef');\n readonly volumeControlRef = viewChild.required<ElementRef>('volumeControlRef');\n\n /* OUTPUTS */\n readonly outFunctionsControl = output<IAudioFunctionControlEvent>();\n readonly outVolumeControl = output<number>();\n readonly outTimeUpdate = output<{ currentTime: string; duration: string }>();\n readonly outEnded = output<void>();\n readonly outMute = output<boolean>();\n readonly outPlay = output<boolean>();\n\n private destroyRef = inject(DestroyRef);\n\n constructor() {\n // Watch for file audio changes\n effect(() => {\n if (this.fileAudio() && this.audioRef()) {\n // Skip initial setup, only reload on changes\n setTimeout(() => {\n this.audioRef().nativeElement.load();\n }, 0);\n }\n });\n effect(() => {\n this.outVolumeControl.emit(this.volumeRatioValue());\n });\n effect(() => {\n this.outTimeUpdate.emit({ currentTime: this.audioTimeCurrent(), duration: this.audioTimeDuration() });\n });\n effect(() => {\n this.outMute.emit(this.isMute());\n });\n effect(() => {\n this.outPlay.emit(this.isPlay());\n });\n }\n\n ngAfterViewInit() {\n merge(\n this.initObservable(this.volumeControlRef().nativeElement, 'mouseenter').pipe(tap(() => this.showFullControlVolume.set(true))),\n this.initObservable(this.volumeControlRef().nativeElement, 'mouseleave').pipe(tap(() => this.showFullControlVolume.set(false)))\n )\n .pipe(takeUntilDestroyed(this.destroyRef))\n .subscribe();\n\n // Emit function control event after view is initialized\n this.outFunctionsControl.emit(this.FunctionsControl);\n }\n\n public get FunctionsControl(): IAudioFunctionControlEvent {\n return {\n playPause: (event?: Event) => this.handlerAudioPausePlay(event),\n toggleMute: (event?: Event) => this.handlerAudioMuteMuted(event),\n seekTo: this.handlerChangeAudio.bind(this),\n setVolume: this.handlerChangeVolume.bind(this),\n download: (event?: Event) => this.handlerDownload(event),\n isPlaying: () => this.isPlay(),\n isMuted: () => this.isMute(),\n };\n }\n\n /* FUNCTIONS */\n private initObservable(el: HTMLElement, eventName: string): Observable<MouseEvent> {\n return fromEvent<MouseEvent>(el, eventName).pipe(\n tap((e) => e.stopPropagation()),\n takeUntilDestroyed(this.destroyRef)\n );\n }\n\n protected async handlerKeyPressAudio() {\n this.isSliderAudioPress.set(true);\n }\n\n protected async handlerAudioMuteMuted(event?: Event) {\n if (event) {\n event.stopPropagation();\n }\n\n if (this.audioRef().nativeElement.muted === true) {\n this.audioRef().nativeElement.muted = false;\n this.isMute.set(false);\n this.volumeRatioValue.set(50);\n\n return;\n }\n this.volumeRatioValue.set(0);\n this.isMute.set(true);\n this.audioRef().nativeElement.muted = true;\n }\n\n protected async handlerAudioPausePlay(event?: Event) {\n if (event) {\n event.stopPropagation();\n }\n\n const audioElement = this.audioRef().nativeElement;\n if (!audioElement.paused) {\n audioElement.pause();\n this.isPlay.set(false);\n return;\n }\n\n try {\n await audioElement.play();\n this.isPlay.set(true);\n } catch (error) {\n console.error('Error playing audio:', error);\n }\n }\n\n protected async handlerLoadedData(event: Event) {\n if (event) {\n event.stopPropagation();\n }\n\n if (this.audioRef().nativeElement) {\n this.audioTimeDuration.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.duration)));\n this.audioTimeCurrent.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.currentTime || 0)));\n this.isDisable.set(false);\n this.isPlay.set(false);\n this.audioRatioValue.set(0);\n this.audioRef().nativeElement.pause();\n }\n }\n\n protected async handlerTimeUpdate(event?: Event) {\n if (event) {\n event.stopPropagation();\n }\n this.isDisable.set(!(this.audioRef().nativeElement.duration || 0));\n if (!this.audioRef().nativeElement) {\n return;\n }\n this.audioTimeDuration.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.duration)));\n this.audioTimeCurrent.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.currentTime || 0)));\n if (this.isSliderAudioPress()) {\n this.audioRef().nativeElement.currentTime = (this.audioRatioValue() * Math.floor(this.audioRef().nativeElement.duration || 0)) / 100;\n\n return;\n }\n this.audioRatioValue.set(Math.floor(((this.audioRef().nativeElement.currentTime || 0) / (this.audioRef().nativeElement.duration || 1)) * 100));\n }\n\n private async toHHMMSS(time: number) {\n const hours = Math.floor(time / 3600);\n const minutes = Math.floor((time - hours * 3600) / 60);\n const seconds = time - hours * 3600 - minutes * 60;\n\n const getLabel = (val: number) => {\n val = val || 0;\n return `${val < 10 ? '0' : ''}${val}`;\n };\n\n return `${getLabel(hours)}:${getLabel(minutes)}:${getLabel(seconds)}`;\n }\n\n protected async handlerChangeAudio(value: number) {\n if (value === this.audioRatioValue()) {\n return;\n }\n this.audioRef().nativeElement.currentTime = ((value || 0) * (this.audioRef().nativeElement.duration || 0)) / 100;\n this.audioRatioValue.set(value);\n this.isSliderAudioPress.set(false);\n }\n\n protected async handlerChangeVolume(value: number) {\n this.audioRef().nativeElement.volume = value / 100;\n this.volumeRatioValue.set(value);\n if (this.audioRef().nativeElement.volume) {\n this.audioRef().nativeElement.muted = false;\n this.isMute.set(false);\n\n return;\n }\n this.audioRef().nativeElement.muted = true;\n this.isMute.set(true);\n }\n\n protected async handlerEnded(event: Event) {\n if (event) {\n event.stopPropagation();\n }\n\n this.isPlay.set(false);\n this.outEnded.emit();\n }\n\n protected async handlerDownload(e?: Event) {\n if (!this.checkPermissionDownloadAudio() || !(await this.checkPermissionDownloadAudio()())) {\n return;\n }\n\n if (e) {\n e.stopPropagation();\n }\n\n if (!this.fileAudio()) {\n return;\n }\n window.open(this.fileAudio(), `_blank`);\n }\n}\n","<audio\n controls\n #audioRef\n class=\"hidden\"\n (timeupdate)=\"handlerTimeUpdate($event)\"\n (loadeddata)=\"handlerLoadedData($event)\"\n (ended)=\"handlerEnded($event)\">\n <source\n [src]=\"fileAudio()\"\n type=\"audio/mpeg\" />\n</audio>\n<div\n [class.libs-ui-disable]=\"isDisable()\"\n [class.pointer-events-none]=\"isDisable()\">\n <div class=\"flex justify-between items-center\">\n <div class=\"w-[70%] flex p-0 items-center\">\n <div\n class=\"flex mr-[16px] cursor-pointer\"\n (click)=\"handlerAudioPausePlay($event)\">\n <i\n class=\"text-[16px]\"\n [class.libs-ui-icon-play-solid]=\"!isPlay()\"\n [class.libs-ui-icon-pause-solid]=\"isPlay()\"></i>\n </div>\n <div class=\"libs-ui-font-h5r mr-[16px]\">{{ audioTimeCurrent() }} /{{ audioTimeDuration() }}</div>\n </div>\n <div class=\"w-[30%] flex p-0 items-center justify-end\">\n <div\n #volumeControlRef\n class=\"flex py-[3px] items-center rounded-[12px] h-[28px]\"\n [class.bg-[#e6e7ea]]=\"showFullControlVolume()\"\n [class.px-[12px]]=\"showFullControlVolume()\">\n <i\n class=\"text-[16px] cursor-pointer\"\n [class.libs-ui-icon-speaker-on-solid]=\"!isMute()\"\n [class.libs-ui-icon-speaker-off-solid]=\"isMute()\"\n (click)=\"handlerAudioMuteMuted($event)\"></i>\n <libs_ui-components-inputs-range_slider\n [class.hidden]=\"!showFullControlVolume()\"\n [mode]=\"'audio'\"\n classInclude=\"flex items-center !w-[54px] cursor-pointer ml-[8px]\"\n [value]=\"volumeRatioValue()\"\n (outChange)=\"handlerChangeVolume($event)\" />\n </div>\n\n <i\n class=\"libs-ui-icon-download-solid ml-[16px] cursor-pointer\"\n (click)=\"handlerDownload($event)\"></i>\n </div>\n </div>\n <div class=\"h-[24px]\">\n <libs_ui-components-inputs-range_slider\n [mode]=\"'audio'\"\n [value]=\"audioRatioValue()\"\n [disable]=\"isDisable()\"\n (outChange)=\"handlerChangeAudio($event)\" />\n </div>\n</div>\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './index';\n"],"names":[],"mappings":";;;;;;MAca,8BAA8B,CAAA;;AAE/B,IAAA,eAAe,GAAG,MAAM,CAAS,CAAC,CAAC;AACnC,IAAA,gBAAgB,GAAG,MAAM,CAAS,GAAG,CAAC;AACtC,IAAA,MAAM,GAAG,MAAM,CAAU,KAAK,CAAC;AAC/B,IAAA,MAAM,GAAG,MAAM,CAAU,KAAK,CAAC;AAC/B,IAAA,kBAAkB,GAAG,MAAM,CAAU,KAAK,CAAC;AAC3C,IAAA,SAAS,GAAG,MAAM,CAAU,IAAI,CAAC;AACjC,IAAA,gBAAgB,GAAG,MAAM,CAAS,OAAO,CAAC;AAC1C,IAAA,iBAAiB,GAAG,MAAM,CAAS,OAAO,CAAC;AAC3C,IAAA,qBAAqB,GAAG,MAAM,CAAU,KAAK,CAAC;;AAG/C,IAAA,SAAS,GAAG,KAAK,CAAC,QAAQ,EAAU;AACpC,IAAA,4BAA4B,GAAG,KAAK,CAAC,QAAQ,EAA0B;;AAGvE,IAAA,QAAQ,GAAG,SAAS,CAAC,QAAQ,CAAa,UAAU,CAAC;AACrD,IAAA,gBAAgB,GAAG,SAAS,CAAC,QAAQ,CAAa,kBAAkB,CAAC;;IAGrE,mBAAmB,GAAG,MAAM,EAA8B;IAC1D,gBAAgB,GAAG,MAAM,EAAU;IACnC,aAAa,GAAG,MAAM,EAA6C;IACnE,QAAQ,GAAG,MAAM,EAAQ;IACzB,OAAO,GAAG,MAAM,EAAW;IAC3B,OAAO,GAAG,MAAM,EAAW;AAE5B,IAAA,UAAU,GAAG,MAAM,CAAC,UAAU,CAAC;AAEvC,IAAA,WAAA,GAAA;;QAEE,MAAM,CAAC,MAAK;YACV,IAAI,IAAI,CAAC,SAAS,EAAE,IAAI,IAAI,CAAC,QAAQ,EAAE,EAAE;;gBAEvC,UAAU,CAAC,MAAK;oBACd,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,IAAI,EAAE;gBACtC,CAAC,EAAE,CAAC,CAAC;YACP;AACF,QAAA,CAAC,CAAC;QACF,MAAM,CAAC,MAAK;YACV,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC;AACrD,QAAA,CAAC,CAAC;QACF,MAAM,CAAC,MAAK;YACV,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,EAAE,WAAW,EAAE,IAAI,CAAC,gBAAgB,EAAE,EAAE,QAAQ,EAAE,IAAI,CAAC,iBAAiB,EAAE,EAAE,CAAC;AACvG,QAAA,CAAC,CAAC;QACF,MAAM,CAAC,MAAK;YACV,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;AAClC,QAAA,CAAC,CAAC;QACF,MAAM,CAAC,MAAK;YACV,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;AAClC,QAAA,CAAC,CAAC;IACJ;IAEA,eAAe,GAAA;AACb,QAAA,KAAK,CACH,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC,aAAa,EAAE,YAAY,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,MAAM,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,EAC9H,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC,aAAa,EAAE,YAAY,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,MAAM,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC;AAE9H,aAAA,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC;AACxC,aAAA,SAAS,EAAE;;QAGd,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC;IACtD;AAEA,IAAA,IAAW,gBAAgB,GAAA;QACzB,OAAO;YACL,SAAS,EAAE,CAAC,KAAa,KAAK,IAAI,CAAC,qBAAqB,CAAC,KAAK,CAAC;YAC/D,UAAU,EAAE,CAAC,KAAa,KAAK,IAAI,CAAC,qBAAqB,CAAC,KAAK,CAAC;YAChE,MAAM,EAAE,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC;YAC1C,SAAS,EAAE,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC;YAC9C,QAAQ,EAAE,CAAC,KAAa,KAAK,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC;AACxD,YAAA,SAAS,EAAE,MAAM,IAAI,CAAC,MAAM,EAAE;AAC9B,YAAA,OAAO,EAAE,MAAM,IAAI,CAAC,MAAM,EAAE;SAC7B;IACH;;IAGQ,cAAc,CAAC,EAAe,EAAE,SAAiB,EAAA;AACvD,QAAA,OAAO,SAAS,CAAa,EAAE,EAAE,SAAS,CAAC,CAAC,IAAI,CAC9C,GAAG,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,eAAe,EAAE,CAAC,EAC/B,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC,CACpC;IACH;AAEU,IAAA,MAAM,oBAAoB,GAAA;AAClC,QAAA,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,IAAI,CAAC;IACnC;IAEU,MAAM,qBAAqB,CAAC,KAAa,EAAA;QACjD,IAAI,KAAK,EAAE;YACT,KAAK,CAAC,eAAe,EAAE;QACzB;QAEA,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,KAAK,KAAK,IAAI,EAAE;YAChD,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,KAAK,GAAG,KAAK;AAC3C,YAAA,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC;AACtB,YAAA,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,EAAE,CAAC;YAE7B;QACF;AACA,QAAA,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,CAAC,CAAC;AAC5B,QAAA,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC;QACrB,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,KAAK,GAAG,IAAI;IAC5C;IAEU,MAAM,qBAAqB,CAAC,KAAa,EAAA;QACjD,IAAI,KAAK,EAAE;YACT,KAAK,CAAC,eAAe,EAAE;QACzB;QAEA,MAAM,YAAY,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa;AAClD,QAAA,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE;YACxB,YAAY,CAAC,KAAK,EAAE;AACpB,YAAA,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC;YACtB;QACF;AAEA,QAAA,IAAI;AACF,YAAA,MAAM,YAAY,CAAC,IAAI,EAAE;AACzB,YAAA,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC;QACvB;QAAE,OAAO,KAAK,EAAE;AACd,YAAA,OAAO,CAAC,KAAK,CAAC,sBAAsB,EAAE,KAAK,CAAC;QAC9C;IACF;IAEU,MAAM,iBAAiB,CAAC,KAAY,EAAA;QAC5C,IAAI,KAAK,EAAE;YACT,KAAK,CAAC,eAAe,EAAE;QACzB;AAEA,QAAA,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,EAAE;YACjC,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,MAAM,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC,CAAC;YACnG,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,MAAM,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,WAAW,IAAI,CAAC,CAAC,CAAC,CAAC;AAC1G,YAAA,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC;AACzB,YAAA,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC;AACtB,YAAA,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC,CAAC;YAC3B,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,KAAK,EAAE;QACvC;IACF;IAEU,MAAM,iBAAiB,CAAC,KAAa,EAAA;QAC7C,IAAI,KAAK,EAAE;YACT,KAAK,CAAC,eAAe,EAAE;QACzB;AACA,QAAA,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,QAAQ,IAAI,CAAC,CAAC,CAAC;QAClE,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,EAAE;YAClC;QACF;QACA,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,MAAM,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC,CAAC;QACnG,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,MAAM,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,WAAW,IAAI,CAAC,CAAC,CAAC,CAAC;AAC1G,QAAA,IAAI,IAAI,CAAC,kBAAkB,EAAE,EAAE;AAC7B,YAAA,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,WAAW,GAAG,CAAC,IAAI,CAAC,eAAe,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,QAAQ,IAAI,CAAC,CAAC,IAAI,GAAG;YAEpI;QACF;AACA,QAAA,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,WAAW,IAAI,CAAC,KAAK,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,QAAQ,IAAI,CAAC,CAAC,IAAI,GAAG,CAAC,CAAC;IAChJ;IAEQ,MAAM,QAAQ,CAAC,IAAY,EAAA;QACjC,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,GAAG,IAAI,CAAC;AACrC,QAAA,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,GAAG,KAAK,GAAG,IAAI,IAAI,EAAE,CAAC;QACtD,MAAM,OAAO,GAAG,IAAI,GAAG,KAAK,GAAG,IAAI,GAAG,OAAO,GAAG,EAAE;AAElD,QAAA,MAAM,QAAQ,GAAG,CAAC,GAAW,KAAI;AAC/B,YAAA,GAAG,GAAG,GAAG,IAAI,CAAC;AACd,YAAA,OAAO,CAAA,EAAG,GAAG,GAAG,EAAE,GAAG,GAAG,GAAG,EAAE,CAAA,EAAG,GAAG,EAAE;AACvC,QAAA,CAAC;AAED,QAAA,OAAO,GAAG,QAAQ,CAAC,KAAK,CAAC,IAAI,QAAQ,CAAC,OAAO,CAAC,IAAI,QAAQ,CAAC,OAAO,CAAC,EAAE;IACvE;IAEU,MAAM,kBAAkB,CAAC,KAAa,EAAA;AAC9C,QAAA,IAAI,KAAK,KAAK,IAAI,CAAC,eAAe,EAAE,EAAE;YACpC;QACF;AACA,QAAA,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,WAAW,GAAG,CAAC,CAAC,KAAK,IAAI,CAAC,KAAK,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,QAAQ,IAAI,CAAC,CAAC,IAAI,GAAG;AAChH,QAAA,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,KAAK,CAAC;AAC/B,QAAA,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,KAAK,CAAC;IACpC;IAEU,MAAM,mBAAmB,CAAC,KAAa,EAAA;QAC/C,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,MAAM,GAAG,KAAK,GAAG,GAAG;AAClD,QAAA,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,KAAK,CAAC;QAChC,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,MAAM,EAAE;YACxC,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,KAAK,GAAG,KAAK;AAC3C,YAAA,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC;YAEtB;QACF;QACA,IAAI,CAAC,QAAQ,EAAE,CAAC,aAAa,CAAC,KAAK,GAAG,IAAI;AAC1C,QAAA,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC;IACvB;IAEU,MAAM,YAAY,CAAC,KAAY,EAAA;QACvC,IAAI,KAAK,EAAE;YACT,KAAK,CAAC,eAAe,EAAE;QACzB;AAEA,QAAA,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC;AACtB,QAAA,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE;IACtB;IAEU,MAAM,eAAe,CAAC,CAAS,EAAA;AACvC,QAAA,IAAI,CAAC,IAAI,CAAC,4BAA4B,EAAE,IAAI,EAAE,MAAM,IAAI,CAAC,4BAA4B,EAAE,EAAE,CAAC,EAAE;YAC1F;QACF;QAEA,IAAI,CAAC,EAAE;YACL,CAAC,CAAC,eAAe,EAAE;QACrB;AAEA,QAAA,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,EAAE;YACrB;QACF;QACA,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,EAAE,CAAA,MAAA,CAAQ,CAAC;IACzC;wGAzNW,8BAA8B,EAAA,IAAA,EAAA,EAAA,EAAA,MAAA,EAAA,EAAA,CAAA,eAAA,CAAA,SAAA,EAAA,CAAA;4FAA9B,8BAA8B,EAAA,YAAA,EAAA,IAAA,EAAA,QAAA,EAAA,0BAAA,EAAA,MAAA,EAAA,EAAA,SAAA,EAAA,EAAA,iBAAA,EAAA,WAAA,EAAA,UAAA,EAAA,WAAA,EAAA,QAAA,EAAA,IAAA,EAAA,UAAA,EAAA,IAAA,EAAA,iBAAA,EAAA,IAAA,EAAA,EAAA,4BAAA,EAAA,EAAA,iBAAA,EAAA,8BAAA,EAAA,UAAA,EAAA,8BAAA,EAAA,QAAA,EAAA,IAAA,EAAA,UAAA,EAAA,IAAA,EAAA,iBAAA,EAAA,IAAA,EAAA,EAAA,EAAA,OAAA,EAAA,EAAA,mBAAA,EAAA,qBAAA,EAAA,gBAAA,EAAA,kBAAA,EAAA,aAAA,EAAA,eAAA,EAAA,QAAA,EAAA,UAAA,EAAA,OAAA,EAAA,SAAA,EAAA,OAAA,EAAA,SAAA,EAAA,EAAA,WAAA,EAAA,CAAA,EAAA,YAAA,EAAA,UAAA,EAAA,KAAA,EAAA,IAAA,EAAA,SAAA,EAAA,CAAA,UAAA,CAAA,EAAA,WAAA,EAAA,IAAA,EAAA,QAAA,EAAA,IAAA,EAAA,EAAA,EAAA,YAAA,EAAA,kBAAA,EAAA,KAAA,EAAA,IAAA,EAAA,SAAA,EAAA,CAAA,kBAAA,CAAA,EAAA,WAAA,EAAA,IAAA,EAAA,QAAA,EAAA,IAAA,EAAA,CAAA,EAAA,QAAA,EAAA,EAAA,EAAA,QAAA,ECd3C,6kEA0DA,EAAA,YAAA,EAAA,CAAA,EAAA,IAAA,EAAA,WAAA,EAAA,IAAA,ED9CY,0CAA0C,EAAA,QAAA,EAAA,wCAAA,EAAA,MAAA,EAAA,CAAA,MAAA,EAAA,KAAA,EAAA,KAAA,EAAA,OAAA,EAAA,cAAA,EAAA,SAAA,EAAA,MAAA,EAAA,MAAA,EAAA,sBAAA,EAAA,cAAA,CAAA,EAAA,OAAA,EAAA,CAAA,aAAA,EAAA,WAAA,CAAA,EAAA,CAAA,EAAA,eAAA,EAAA,EAAA,CAAA,uBAAA,CAAA,MAAA,EAAA,CAAA;;4FAEzC,8BAA8B,EAAA,UAAA,EAAA,CAAA;kBAR1C,SAAS;+BAEE,0BAA0B,EAAA,UAAA,EAExB,IAAI,EAAA,eAAA,EACC,uBAAuB,CAAC,MAAM,EAAA,OAAA,EACtC,CAAC,0CAA0C,CAAC,EAAA,QAAA,EAAA,6kEAAA,EAAA;;;AEZvD;;AAEG;;;;"}
1
+ {"version":3,"file":"libs-ui-components-audio.mjs","sources":["../../../../../libs-ui/components/audio/src/audio.component.ts","../../../../../libs-ui/components/audio/src/audio.component.html","../../../../../libs-ui/components/audio/src/libs-ui-components-audio.ts"],"sourcesContent":["import { AfterViewInit, ChangeDetectionStrategy, Component, DestroyRef, effect, ElementRef, inject, input, output, signal, viewChild } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { LibsUiComponentsInputsRangeSliderComponent } from '@libs-ui/components-inputs-range-slider';\nimport { fromEvent, merge, Observable, tap } from 'rxjs';\nimport { IAudioFunctionControlEvent } from './interfaces/function-control-event.interface';\n@Component({\n // eslint-disable-next-line @angular-eslint/component-selector\n selector: 'libs_ui-components-audio',\n templateUrl: './audio.component.html',\n standalone: true,\n changeDetection: ChangeDetectionStrategy.OnPush,\n imports: [LibsUiComponentsInputsRangeSliderComponent],\n})\nexport class LibsUiComponentsAudioComponent implements AfterViewInit {\n // =========================================\n // INTERNAL SIGNALS\n // =========================================\n\n protected audioRatioValue = signal<number>(0);\n\n protected volumeRatioValue = signal<number>(100);\n\n protected isPlay = signal<boolean>(false);\n\n protected isMute = signal<boolean>(false);\n\n protected isSliderAudioPress = signal<boolean>(false);\n\n protected isDisable = signal<boolean>(true);\n\n protected audioTimeCurrent = signal<string>('_:_:_');\n\n protected audioTimeDuration = signal<string>('_:_:_');\n\n protected showFullControlVolume = signal<boolean>(false);\n\n // =========================================\n // INPUTS\n // =========================================\n\n readonly fileAudio = input.required<string>();\n\n readonly checkPermissionDownloadAudio = input.required<() => Promise<boolean>>();\n\n // =========================================\n // VIEW CHILDREN\n // =========================================\n\n readonly audioRef = viewChild.required<ElementRef>('audioRef');\n\n readonly volumeControlRef = viewChild.required<ElementRef>('volumeControlRef');\n\n // =========================================\n // OUTPUTS\n // =========================================\n\n readonly outFunctionsControl = output<IAudioFunctionControlEvent>();\n\n readonly outVolumeControl = output<number>();\n\n readonly outTimeUpdate = output<{ currentTime: string; duration: string }>();\n\n readonly outEnded = output<void>();\n\n readonly outMute = output<boolean>();\n\n readonly outPlay = output<boolean>();\n\n // =========================================\n // PRIVATE PROPERTIES\n // =========================================\n\n private destroyRef = inject(DestroyRef);\n\n // =========================================\n // CONSTRUCTOR\n // =========================================\n\n constructor() {\n // Watch for file audio changes\n effect(() => {\n if (this.fileAudio() && this.audioRef()) {\n setTimeout(() => {\n this.audioRef().nativeElement.load();\n }, 0);\n }\n });\n\n effect(() => {\n this.outVolumeControl.emit(this.volumeRatioValue());\n });\n\n effect(() => {\n this.outTimeUpdate.emit({ currentTime: this.audioTimeCurrent(), duration: this.audioTimeDuration() });\n });\n\n effect(() => {\n this.outMute.emit(this.isMute());\n });\n\n effect(() => {\n this.outPlay.emit(this.isPlay());\n });\n }\n\n // =========================================\n // LIFECYCLE HOOKS\n // =========================================\n\n ngAfterViewInit(): void {\n merge(\n this.initObservable(this.volumeControlRef().nativeElement, 'mouseenter').pipe(tap(() => this.showFullControlVolume.set(true))),\n this.initObservable(this.volumeControlRef().nativeElement, 'mouseleave').pipe(tap(() => this.showFullControlVolume.set(false)))\n )\n .pipe(takeUntilDestroyed(this.destroyRef))\n .subscribe();\n\n // Emit function control event after view is initialized\n this.outFunctionsControl.emit(this.FunctionsControl);\n }\n\n // =========================================\n // PUBLIC API\n // =========================================\n\n public get FunctionsControl(): IAudioFunctionControlEvent {\n return {\n playPause: (event?: Event) => this.handlerAudioPausePlay(event),\n toggleMute: (event?: Event) => this.handlerAudioMuteMuted(event),\n seekTo: this.handlerChangeAudio.bind(this),\n setVolume: this.handlerChangeVolume.bind(this),\n download: (event?: Event) => this.handlerDownload(event),\n isPlaying: () => this.isPlay(),\n isMuted: () => this.isMute(),\n };\n }\n\n // =========================================\n // PRIVATE METHODS\n // =========================================\n\n private initObservable(el: HTMLElement, eventName: string): Observable<MouseEvent> {\n return fromEvent<MouseEvent>(el, eventName).pipe(\n tap((e) => e.stopPropagation()),\n takeUntilDestroyed(this.destroyRef)\n );\n }\n\n protected async handlerKeyPressAudio(): Promise<void> {\n this.isSliderAudioPress.set(true);\n }\n\n protected async handlerAudioMuteMuted(event?: Event): Promise<void> {\n if (event) {\n event.stopPropagation();\n }\n\n if (this.audioRef().nativeElement.muted === true) {\n this.audioRef().nativeElement.muted = false;\n this.isMute.set(false);\n this.volumeRatioValue.set(50);\n return;\n }\n\n this.volumeRatioValue.set(0);\n this.isMute.set(true);\n this.audioRef().nativeElement.muted = true;\n }\n\n protected async handlerAudioPausePlay(event?: Event): Promise<void> {\n if (event) {\n event.stopPropagation();\n }\n\n const audioElement = this.audioRef().nativeElement;\n if (!audioElement.paused) {\n audioElement.pause();\n this.isPlay.set(false);\n return;\n }\n\n try {\n await audioElement.play();\n this.isPlay.set(true);\n } catch (error) {\n console.error('Error playing audio:', error);\n }\n }\n\n protected async handlerLoadedData(event: Event): Promise<void> {\n if (event) {\n event.stopPropagation();\n }\n\n if (this.audioRef().nativeElement) {\n this.audioTimeDuration.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.duration)));\n this.audioTimeCurrent.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.currentTime || 0)));\n this.isDisable.set(false);\n this.isPlay.set(false);\n this.audioRatioValue.set(0);\n this.audioRef().nativeElement.pause();\n }\n }\n\n protected async handlerTimeUpdate(event?: Event): Promise<void> {\n if (event) {\n event.stopPropagation();\n }\n\n this.isDisable.set(!(this.audioRef().nativeElement.duration || 0));\n\n if (!this.audioRef().nativeElement) {\n return;\n }\n\n this.audioTimeDuration.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.duration)));\n this.audioTimeCurrent.set(await this.toHHMMSS(Math.floor(this.audioRef().nativeElement.currentTime || 0)));\n\n if (this.isSliderAudioPress()) {\n this.audioRef().nativeElement.currentTime = (this.audioRatioValue() * Math.floor(this.audioRef().nativeElement.duration || 0)) / 100;\n return;\n }\n\n this.audioRatioValue.set(Math.floor(((this.audioRef().nativeElement.currentTime || 0) / (this.audioRef().nativeElement.duration || 1)) * 100));\n }\n\n /**\n * Format seconds -> HH:MM:SS.\n * Used for both `currentTime` and `duration` outputs to keep output stable.\n */\n private async toHHMMSS(time: number): Promise<string> {\n const hours = Math.floor(time / 3600);\n const minutes = Math.floor((time - hours * 3600) / 60);\n const seconds = time - hours * 3600 - minutes * 60;\n\n const getLabel = (val: number) => {\n val = val || 0;\n return `${val < 10 ? '0' : ''}${val}`;\n };\n\n return `${getLabel(hours)}:${getLabel(minutes)}:${getLabel(seconds)}`;\n }\n\n protected async handlerChangeAudio(value: number): Promise<void> {\n if (value === this.audioRatioValue()) {\n return;\n }\n\n this.audioRef().nativeElement.currentTime = ((value || 0) * (this.audioRef().nativeElement.duration || 0)) / 100;\n this.audioRatioValue.set(value);\n this.isSliderAudioPress.set(false);\n }\n\n protected async handlerChangeVolume(value: number): Promise<void> {\n this.audioRef().nativeElement.volume = value / 100;\n this.volumeRatioValue.set(value);\n\n if (this.audioRef().nativeElement.volume) {\n this.audioRef().nativeElement.muted = false;\n this.isMute.set(false);\n return;\n }\n\n this.audioRef().nativeElement.muted = true;\n this.isMute.set(true);\n }\n\n protected async handlerEnded(event: Event): Promise<void> {\n if (event) {\n event.stopPropagation();\n }\n\n this.isPlay.set(false);\n this.outEnded.emit();\n }\n\n /**\n * Download chỉ được thực hiện nếu callback permission trả về `true`.\n * Lưu ý: `checkPermissionDownloadAudio` là input function (factory), nên cần gọi 2 lần:\n * - lần 1: lấy function\n * - lần 2: execute function để lấy Promise<boolean>\n */\n protected async handlerDownload(e?: Event): Promise<void> {\n if (!this.checkPermissionDownloadAudio() || !(await this.checkPermissionDownloadAudio()())) {\n return;\n }\n\n if (e) {\n e.stopPropagation();\n }\n\n if (!this.fileAudio()) {\n return;\n }\n\n window.open(this.fileAudio(), '_blank');\n }\n}\n","<audio\n controls\n #audioRef\n class=\"hidden\"\n (timeupdate)=\"handlerTimeUpdate($event)\"\n (loadeddata)=\"handlerLoadedData($event)\"\n (ended)=\"handlerEnded($event)\">\n <source\n [src]=\"fileAudio()\"\n type=\"audio/mpeg\" />\n</audio>\n<div\n [class.libs-ui-disable]=\"isDisable()\"\n [class.pointer-events-none]=\"isDisable()\">\n <div class=\"flex justify-between items-center\">\n <div class=\"w-[70%] flex p-0 items-center\">\n <div\n class=\"flex mr-[16px] cursor-pointer\"\n (click)=\"handlerAudioPausePlay($event)\">\n <i\n class=\"text-[16px]\"\n [class.libs-ui-icon-play-solid]=\"!isPlay()\"\n [class.libs-ui-icon-pause-solid]=\"isPlay()\"></i>\n </div>\n <div class=\"libs-ui-font-h5r mr-[16px]\">{{ audioTimeCurrent() }} /{{ audioTimeDuration() }}</div>\n </div>\n <div class=\"w-[30%] flex p-0 items-center justify-end\">\n <div\n #volumeControlRef\n class=\"flex py-[3px] items-center rounded-[12px] h-[28px]\"\n [class.bg-[#e6e7ea]]=\"showFullControlVolume()\"\n [class.px-[12px]]=\"showFullControlVolume()\">\n <i\n class=\"text-[16px] cursor-pointer\"\n [class.libs-ui-icon-speaker-on-solid]=\"!isMute()\"\n [class.libs-ui-icon-speaker-off-solid]=\"isMute()\"\n (click)=\"handlerAudioMuteMuted($event)\"></i>\n <libs_ui-components-inputs-range_slider\n [class.hidden]=\"!showFullControlVolume()\"\n [mode]=\"'audio'\"\n classInclude=\"flex items-center !w-[54px] cursor-pointer ml-[8px]\"\n [value]=\"volumeRatioValue()\"\n (outChange)=\"handlerChangeVolume($event)\" />\n </div>\n\n <i\n class=\"libs-ui-icon-download-solid ml-[16px] cursor-pointer\"\n (click)=\"handlerDownload($event)\"></i>\n </div>\n </div>\n <div class=\"h-[24px]\">\n <libs_ui-components-inputs-range_slider\n [mode]=\"'audio'\"\n [value]=\"audioRatioValue()\"\n [disable]=\"isDisable()\"\n (outChange)=\"handlerChangeAudio($event)\" />\n </div>\n</div>\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './index';\n"],"names":[],"mappings":"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"}
@@ -1,50 +1,39 @@
1
1
  /**
2
- * Interface cho các chức năng điều khiển audio được cung cấp qua output event
3
- * @description Cung cấp các method để điều khiển audio player từ component cha
2
+ * Public control surface để điều khiển audio từ component cha.
3
+ * Chỉ tả những điểm dễ dùng sai (như range/value).
4
4
  */
5
5
  export interface IAudioFunctionControlEvent {
6
- /**
7
- * Key để điều khiển audio
8
- * @returns void
9
- */
10
6
  /**
11
7
  * Bắt đầu hoặc tạm dừng phát audio
12
8
  * @param event Optional event object
13
- * @returns void
14
9
  */
15
10
  playPause: (event?: Event) => void;
16
11
  /**
17
12
  * Bật hoặc tắt âm thanh
18
13
  * @param event Optional event object
19
- * @returns void
20
14
  */
21
15
  toggleMute: (event?: Event) => void;
22
16
  /**
23
17
  * Điều chỉnh âm lượng
24
18
  * @param value Giá trị âm lượng từ 0 đến 100
25
- * @returns void
26
19
  */
27
20
  setVolume: (value: number) => void;
28
21
  /**
29
22
  * Di chuyển đến vị trí cụ thể trong audio
30
23
  * @param value Giá trị phần trăm từ 0 đến 100
31
- * @returns void
32
24
  */
33
25
  seekTo: (value: number) => void;
34
26
  /**
35
27
  * Tải xuống file audio
36
28
  * @param event Optional event object
37
- * @returns void
38
29
  */
39
30
  download: (event?: Event) => void;
40
31
  /**
41
32
  * Kiểm tra trạng thái đang phát audio
42
- * @returns boolean True nếu đang phát, False nếu đang tạm dừng
43
33
  */
44
34
  isPlaying: () => boolean;
45
35
  /**
46
36
  * Kiểm tra trạng thái tắt tiếng
47
- * @returns boolean True nếu đang tắt tiếng, False nếu đang bật tiếng
48
37
  */
49
38
  isMuted: () => boolean;
50
39
  }
package/package.json CHANGED
@@ -1,9 +1,9 @@
1
1
  {
2
2
  "name": "@libs-ui/components-audio",
3
- "version": "0.2.355-9",
3
+ "version": "0.2.356-0",
4
4
  "peerDependencies": {
5
5
  "@angular/core": ">=18.0.0",
6
- "@libs-ui/components-inputs-range-slider": "0.2.355-9",
6
+ "@libs-ui/components-inputs-range-slider": "0.2.356-0",
7
7
  "rxjs": "~7.8.0"
8
8
  },
9
9
  "sideEffects": false,