@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 +136 -273
- package/audio.component.d.ts +10 -0
- package/esm2022/audio.component.mjs +39 -8
- package/esm2022/interfaces/function-control-event.interface.mjs +1 -1
- package/fesm2022/libs-ui-components-audio.mjs +38 -7
- package/fesm2022/libs-ui-components-audio.mjs.map +1 -1
- package/interfaces/function-control-event.interface.d.ts +2 -13
- package/package.json +2 -2
package/README.md
CHANGED
|
@@ -1,347 +1,210 @@
|
|
|
1
|
-
#
|
|
1
|
+
# @libs-ui/components-audio
|
|
2
2
|
|
|
3
|
-
|
|
3
|
+
> Component phát âm thanh với đầy đủ chức năng điều khiển cho Angular.
|
|
4
4
|
|
|
5
|
-
|
|
5
|
+
## Giới thiệu
|
|
6
6
|
|
|
7
|
-
|
|
7
|
+
`LibsUiComponentsAudioComponent` là 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
|
|
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
|
-
-
|
|
16
|
-
-
|
|
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
|
-
|
|
24
|
+
# yarn
|
|
29
25
|
yarn add @libs-ui/components-audio
|
|
30
26
|
```
|
|
31
27
|
|
|
32
28
|
## Sử dụng
|
|
33
29
|
|
|
34
|
-
### Import
|
|
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 {
|
|
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: [
|
|
39
|
+
imports: [LibsUiComponentsAudioComponent],
|
|
59
40
|
template: `
|
|
60
41
|
<libs_ui-components-audio
|
|
61
|
-
[fileAudio]="
|
|
62
|
-
[checkPermissionDownloadAudio]="
|
|
63
|
-
(outFunctionsControl)="
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
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
|
-
|
|
73
|
-
|
|
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
|
-
|
|
76
|
-
|
|
77
|
-
return Promise.resolve(true);
|
|
66
|
+
onVolumeChange(volume: number) {
|
|
67
|
+
console.log('Âm lượng:', volume);
|
|
78
68
|
}
|
|
79
69
|
|
|
80
|
-
|
|
81
|
-
|
|
70
|
+
onTimeUpdate(time: { currentTime: string; duration: string }) {
|
|
71
|
+
console.log('Thời gian:', time.currentTime, '/', time.duration);
|
|
82
72
|
}
|
|
83
73
|
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
this.functionControls.playPause();
|
|
87
|
-
}
|
|
74
|
+
onMuteChange(isMuted: boolean) {
|
|
75
|
+
console.log('Tắt tiếng:', isMuted);
|
|
88
76
|
}
|
|
89
77
|
|
|
90
|
-
|
|
91
|
-
|
|
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
|
|
103
|
-
|
|
|
104
|
-
| fileAudio | `string` |
|
|
105
|
-
| checkPermissionDownloadAudio | `() => Promise<boolean>` |
|
|
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
|
|
110
|
-
|
|
|
111
|
-
| outFunctionsControl | `IAudioFunctionControlEvent` |
|
|
112
|
-
| outVolumeControl | `number` |
|
|
113
|
-
| outTimeUpdate | `{ currentTime: string, duration: string }` |
|
|
114
|
-
| outEnded | `void` |
|
|
115
|
-
| outMute | `boolean` |
|
|
116
|
-
| outPlay | `boolean` |
|
|
117
|
-
|
|
118
|
-
###
|
|
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
|
-
##
|
|
170
|
+
## Công nghệ sử dụng
|
|
146
171
|
|
|
147
|
-
|
|
172
|
+
- **Angular 18+** - Standalone Components
|
|
173
|
+
- **Angular Signals** - Reactive state management
|
|
174
|
+
- **RxJS** - Event handling với `takeUntilDestroyed`
|
|
175
|
+
- **TailwindCSS** - Styling
|
|
148
176
|
|
|
149
|
-
|
|
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
|
-
|
|
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 có sẵn trong ứng dụng `core-ui`:
|
|
165
180
|
|
|
166
|
-
|
|
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
|
-
|
|
180
|
-
|
|
181
|
-
### Sử dụng Function Control
|
|
185
|
+
**File demo:** `apps/core-ui/src/app/components/audio/audio.component.ts`
|
|
182
186
|
|
|
183
|
-
|
|
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
|
-
|
|
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
|
-
|
|
250
|
-
|
|
251
|
-
|
|
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
|
-
|
|
257
|
-
if (this.functionControls) {
|
|
258
|
-
this.functionControls.toggleMute();
|
|
259
|
-
this.isMuted.set(this.functionControls.isMuted());
|
|
260
|
-
}
|
|
261
|
-
}
|
|
195
|
+
## Unit Tests
|
|
262
196
|
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
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
|
-
|
|
272
|
-
|
|
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
|
-
|
|
280
|
-
|
|
281
|
-
this.functionControls.download();
|
|
282
|
-
}
|
|
283
|
-
}
|
|
284
|
-
}
|
|
204
|
+
# Watch mode
|
|
205
|
+
npx nx test components-audio --watch
|
|
285
206
|
```
|
|
286
207
|
|
|
287
|
-
|
|
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
|
-
|
|
340
|
-
this.isMuted.set(isMuted);
|
|
341
|
-
}
|
|
342
|
-
|
|
343
|
-
handleEnded() {
|
|
344
|
-
this.isPlaying.set(false);
|
|
345
|
-
}
|
|
346
|
-
}
|
|
347
|
-
```
|
|
210
|
+
MIT
|
package/audio.component.d.ts
CHANGED
|
@@ -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
|
-
//
|
|
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
|
-
//
|
|
19
|
+
// =========================================
|
|
20
|
+
// INPUTS
|
|
21
|
+
// =========================================
|
|
18
22
|
fileAudio = input.required();
|
|
19
23
|
checkPermissionDownloadAudio = input.required();
|
|
20
|
-
|
|
24
|
+
// =========================================
|
|
25
|
+
// VIEW CHILDREN
|
|
26
|
+
// =========================================
|
|
21
27
|
audioRef = viewChild.required('audioRef');
|
|
22
28
|
volumeControlRef = viewChild.required('volumeControlRef');
|
|
23
|
-
|
|
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
|
-
|
|
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(),
|
|
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,
|
|
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
|
-
//
|
|
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
|
-
//
|
|
20
|
+
// =========================================
|
|
21
|
+
// INPUTS
|
|
22
|
+
// =========================================
|
|
19
23
|
fileAudio = input.required();
|
|
20
24
|
checkPermissionDownloadAudio = input.required();
|
|
21
|
-
|
|
25
|
+
// =========================================
|
|
26
|
+
// VIEW CHILDREN
|
|
27
|
+
// =========================================
|
|
22
28
|
audioRef = viewChild.required('audioRef');
|
|
23
29
|
volumeControlRef = viewChild.required('volumeControlRef');
|
|
24
|
-
|
|
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
|
-
|
|
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(),
|
|
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":"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"}
|
|
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
|
-
*
|
|
3
|
-
*
|
|
2
|
+
* Public control surface để điều khiển audio từ component cha.
|
|
3
|
+
* Chỉ mô 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.
|
|
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.
|
|
6
|
+
"@libs-ui/components-inputs-range-slider": "0.2.356-0",
|
|
7
7
|
"rxjs": "~7.8.0"
|
|
8
8
|
},
|
|
9
9
|
"sideEffects": false,
|