@libs-ui/components-audio 0.2.190 → 0.2.192
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 +300 -57
- package/audio.component.d.ts +16 -5
- package/demo/audio-demo.component.d.ts +112 -0
- package/esm2022/audio.component.mjs +74 -15
- package/esm2022/demo/audio-demo.component.mjs +473 -0
- package/esm2022/index.mjs +3 -2
- package/esm2022/interfaces/function-control-event.interface.mjs +2 -0
- package/fesm2022/libs-ui-components-audio.mjs +528 -124
- package/fesm2022/libs-ui-components-audio.mjs.map +1 -1
- package/index.d.ts +2 -1
- package/interfaces/function-control-event.interface.d.ts +46 -0
- package/package.json +2 -2
- package/demo.component.d.ts +0 -20
- package/esm2022/demo.component.mjs +0 -128
|
@@ -0,0 +1,473 @@
|
|
|
1
|
+
import { Component, ViewChild, computed, signal } from '@angular/core';
|
|
2
|
+
import { LibsUiComponentsAudioComponent } from '../audio.component';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export class LibsUiComponentsAudioDemoComponent {
|
|
5
|
+
audioPlayer;
|
|
6
|
+
// State using signals
|
|
7
|
+
isPlaying = signal(false);
|
|
8
|
+
isMuted = signal(false);
|
|
9
|
+
volume = signal(100);
|
|
10
|
+
currentTime = signal('00:00:00');
|
|
11
|
+
duration = signal('00:00:00');
|
|
12
|
+
progress = signal(0);
|
|
13
|
+
// Computed properties for template display
|
|
14
|
+
volumePercent = computed(() => Math.round(this.volume()));
|
|
15
|
+
// Selected audio sample
|
|
16
|
+
selectedAudio = signal(1);
|
|
17
|
+
currentAudioSource = signal('https://nhacchuong123.com/nhac-chuong/abcdefgh/nhac-chuong-nguoi-ay-dau-co-dang-akira-phan-nguyen-van-chung.mp3');
|
|
18
|
+
// Function Control variables
|
|
19
|
+
functionControls = null;
|
|
20
|
+
// Audio samples for demo
|
|
21
|
+
audioSamples = signal([
|
|
22
|
+
{
|
|
23
|
+
id: 1,
|
|
24
|
+
name: 'Bản nhạc mẫu 1',
|
|
25
|
+
src: 'https://nhacchuong123.com/nhac-chuong/abcdefgh/nhac-chuong-nguoi-ay-dau-co-dang-akira-phan-nguyen-van-chung.mp3',
|
|
26
|
+
duration: '01:30'
|
|
27
|
+
},
|
|
28
|
+
{
|
|
29
|
+
id: 2,
|
|
30
|
+
name: 'Bản nhạc mẫu 2',
|
|
31
|
+
src: 'https://www.soundhelix.com/examples/mp3/SoundHelix-Song-1.mp3',
|
|
32
|
+
duration: '02:15'
|
|
33
|
+
},
|
|
34
|
+
{
|
|
35
|
+
id: 3,
|
|
36
|
+
name: 'Bản nhạc mẫu 3',
|
|
37
|
+
src: 'https://dl.dropboxusercontent.com/s/75jpngrgnavyu1f/The-Noisy-Freaks.mp3',
|
|
38
|
+
duration: '03:45'
|
|
39
|
+
}
|
|
40
|
+
]);
|
|
41
|
+
// API Documentation data
|
|
42
|
+
inputsDoc = signal([
|
|
43
|
+
{
|
|
44
|
+
name: 'fileAudio',
|
|
45
|
+
type: 'string',
|
|
46
|
+
default: 'Bắt buộc',
|
|
47
|
+
description: 'URL của file audio cần phát'
|
|
48
|
+
},
|
|
49
|
+
{
|
|
50
|
+
name: 'checkPermissionDownloadAudio',
|
|
51
|
+
type: '() => Promise<boolean>',
|
|
52
|
+
default: 'Bắt buộc',
|
|
53
|
+
description: 'Function trả về promise với kết quả boolean cho biết nếu được phép download'
|
|
54
|
+
}
|
|
55
|
+
]);
|
|
56
|
+
// Output documentation
|
|
57
|
+
outputsDoc = signal([
|
|
58
|
+
{
|
|
59
|
+
name: 'outFunctionsControl',
|
|
60
|
+
type: 'IAudioFunctionControlEvent',
|
|
61
|
+
description: 'Emits các hàm điều khiển audio'
|
|
62
|
+
},
|
|
63
|
+
{
|
|
64
|
+
name: 'outVolumeControl',
|
|
65
|
+
type: 'number',
|
|
66
|
+
description: 'Emits giá trị âm lượng hiện tại (0-100)'
|
|
67
|
+
},
|
|
68
|
+
{
|
|
69
|
+
name: 'outTimeUpdate',
|
|
70
|
+
type: '{ currentTime: string, duration: string }',
|
|
71
|
+
description: 'Emits thông tin thời gian hiện tại và tổng thời gian'
|
|
72
|
+
},
|
|
73
|
+
{
|
|
74
|
+
name: 'outEnded',
|
|
75
|
+
type: 'void',
|
|
76
|
+
description: 'Emits khi audio kết thúc phát'
|
|
77
|
+
},
|
|
78
|
+
{
|
|
79
|
+
name: 'outMute',
|
|
80
|
+
type: 'boolean',
|
|
81
|
+
description: 'Emits trạng thái tắt/bật tiếng'
|
|
82
|
+
},
|
|
83
|
+
{
|
|
84
|
+
name: 'outPlay',
|
|
85
|
+
type: 'boolean',
|
|
86
|
+
description: 'Emits trạng thái phát/tạm dừng'
|
|
87
|
+
}
|
|
88
|
+
]);
|
|
89
|
+
// Interface documentation
|
|
90
|
+
interfacesDoc = signal([
|
|
91
|
+
{
|
|
92
|
+
name: 'IAudioFunctionControlEvent',
|
|
93
|
+
description: 'Interface cho các chức năng điều khiển audio được cung cấp qua output event',
|
|
94
|
+
properties: [
|
|
95
|
+
{
|
|
96
|
+
name: 'playPause',
|
|
97
|
+
type: '(event?: Event) => void',
|
|
98
|
+
description: 'Bắt đầu hoặc tạm dừng phát audio'
|
|
99
|
+
},
|
|
100
|
+
{
|
|
101
|
+
name: 'toggleMute',
|
|
102
|
+
type: '(event?: Event) => void',
|
|
103
|
+
description: 'Bật hoặc tắt âm thanh'
|
|
104
|
+
},
|
|
105
|
+
{
|
|
106
|
+
name: 'setVolume',
|
|
107
|
+
type: '(value: number) => void',
|
|
108
|
+
description: 'Điều chỉnh âm lượng (giá trị từ 0 đến 100)'
|
|
109
|
+
},
|
|
110
|
+
{
|
|
111
|
+
name: 'seekTo',
|
|
112
|
+
type: '(value: number) => void',
|
|
113
|
+
description: 'Di chuyển đến vị trí cụ thể trong audio (giá trị từ 0 đến 100)'
|
|
114
|
+
},
|
|
115
|
+
{
|
|
116
|
+
name: 'download',
|
|
117
|
+
type: '(event?: Event) => void',
|
|
118
|
+
description: 'Tải xuống file audio'
|
|
119
|
+
},
|
|
120
|
+
{
|
|
121
|
+
name: 'isPlaying',
|
|
122
|
+
type: '() => boolean',
|
|
123
|
+
description: 'Kiểm tra trạng thái đang phát audio'
|
|
124
|
+
},
|
|
125
|
+
{
|
|
126
|
+
name: 'isMuted',
|
|
127
|
+
type: '() => boolean',
|
|
128
|
+
description: 'Kiểm tra trạng thái tắt tiếng'
|
|
129
|
+
}
|
|
130
|
+
]
|
|
131
|
+
}
|
|
132
|
+
]);
|
|
133
|
+
// Method documentation
|
|
134
|
+
methodsDoc = signal([
|
|
135
|
+
{
|
|
136
|
+
name: 'playPause',
|
|
137
|
+
params: 'event?: Event',
|
|
138
|
+
returnType: 'void',
|
|
139
|
+
description: 'Phát hoặc tạm dừng audio'
|
|
140
|
+
},
|
|
141
|
+
{
|
|
142
|
+
name: 'toggleMute',
|
|
143
|
+
params: 'event?: Event',
|
|
144
|
+
returnType: 'void',
|
|
145
|
+
description: 'Bật/tắt âm thanh'
|
|
146
|
+
},
|
|
147
|
+
{
|
|
148
|
+
name: 'seekTo',
|
|
149
|
+
params: 'value: number',
|
|
150
|
+
returnType: 'void',
|
|
151
|
+
description: 'Di chuyển đến vị trí cụ thể trong audio (giá trị từ 0-100)'
|
|
152
|
+
},
|
|
153
|
+
{
|
|
154
|
+
name: 'setVolume',
|
|
155
|
+
params: 'value: number',
|
|
156
|
+
returnType: 'void',
|
|
157
|
+
description: 'Điều chỉnh âm lượng (giá trị từ 0-100)'
|
|
158
|
+
},
|
|
159
|
+
{
|
|
160
|
+
name: 'download',
|
|
161
|
+
params: 'event?: Event',
|
|
162
|
+
returnType: 'void',
|
|
163
|
+
description: 'Tải xuống file audio'
|
|
164
|
+
},
|
|
165
|
+
{
|
|
166
|
+
name: 'isPlaying',
|
|
167
|
+
params: '',
|
|
168
|
+
returnType: 'boolean',
|
|
169
|
+
description: 'Kiểm tra nếu audio đang phát'
|
|
170
|
+
},
|
|
171
|
+
{
|
|
172
|
+
name: 'isMuted',
|
|
173
|
+
params: '',
|
|
174
|
+
returnType: 'boolean',
|
|
175
|
+
description: 'Kiểm tra nếu audio đang tắt tiếng'
|
|
176
|
+
}
|
|
177
|
+
]);
|
|
178
|
+
// Features data
|
|
179
|
+
features = signal([
|
|
180
|
+
{
|
|
181
|
+
id: 1,
|
|
182
|
+
icon: '▶️',
|
|
183
|
+
title: 'Điều khiển audio',
|
|
184
|
+
description: 'Điều khiển audio component qua các API'
|
|
185
|
+
},
|
|
186
|
+
{
|
|
187
|
+
id: 2,
|
|
188
|
+
icon: '🔊',
|
|
189
|
+
title: 'Quản lý âm lượng',
|
|
190
|
+
description: 'Điều chỉnh âm lượng và tắt tiếng với thanh trượt trực quan'
|
|
191
|
+
},
|
|
192
|
+
{
|
|
193
|
+
id: 3,
|
|
194
|
+
icon: '⏱️',
|
|
195
|
+
title: 'Hiển thị thời gian',
|
|
196
|
+
description: 'Hiển thị thời gian hiện tại và tổng thời gian theo định dạng HH:MM:SS'
|
|
197
|
+
},
|
|
198
|
+
{
|
|
199
|
+
id: 4,
|
|
200
|
+
icon: '📊',
|
|
201
|
+
title: 'Thanh tiến độ',
|
|
202
|
+
description: 'Thanh tiến độ có thể tương tác để tua nhanh hoặc tua lại'
|
|
203
|
+
},
|
|
204
|
+
{
|
|
205
|
+
id: 5,
|
|
206
|
+
icon: '📥',
|
|
207
|
+
title: 'Tải xuống',
|
|
208
|
+
description: 'Hỗ trợ tải xuống file âm thanh với kiểm soát quyền'
|
|
209
|
+
}
|
|
210
|
+
]);
|
|
211
|
+
// Code examples data
|
|
212
|
+
codeExamples = signal([
|
|
213
|
+
{
|
|
214
|
+
id: 1,
|
|
215
|
+
title: 'Cài đặt cơ bản',
|
|
216
|
+
code: `<libs_ui-components-audio
|
|
217
|
+
[fileAudio]="'path/to/audio.mp3'"
|
|
218
|
+
[checkPermissionDownloadAudio]="checkPermission">
|
|
219
|
+
</libs_ui-components-audio>`
|
|
220
|
+
},
|
|
221
|
+
{
|
|
222
|
+
id: 2,
|
|
223
|
+
title: 'Sử dụng Function Control',
|
|
224
|
+
code: `import { Component, ViewChild, signal } from '@angular/core';
|
|
225
|
+
import { LibsUiComponentsAudioComponent } from '@libs-ui/components-audio';
|
|
226
|
+
import { IAudioFunctionControlEvent } from '@libs-ui/components-audio';
|
|
227
|
+
|
|
228
|
+
@Component({
|
|
229
|
+
selector: 'app-my-component',
|
|
230
|
+
template: \`
|
|
231
|
+
<libs_ui-components-audio
|
|
232
|
+
#audioPlayer
|
|
233
|
+
[fileAudio]="audioSource()"
|
|
234
|
+
[checkPermissionDownloadAudio]="checkPermission"
|
|
235
|
+
(outFunctionsControl)="registerFunctions($event)">
|
|
236
|
+
</libs_ui-components-audio>
|
|
237
|
+
|
|
238
|
+
<button (click)="playAudio()">Phát/Tạm dừng</button>
|
|
239
|
+
\`
|
|
240
|
+
})
|
|
241
|
+
export class MyComponent {
|
|
242
|
+
@ViewChild('audioPlayer') audioPlayer!: LibsUiComponentsAudioComponent;
|
|
243
|
+
audioSource = signal<string>('path/to/audio.mp3');
|
|
244
|
+
functionControls: IAudioFunctionControlEvent | null = null;
|
|
245
|
+
|
|
246
|
+
registerFunctions(event: IAudioFunctionControlEvent) {
|
|
247
|
+
this.functionControls = event;
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
playAudio() {
|
|
251
|
+
if (this.functionControls) {
|
|
252
|
+
this.functionControls.playPause();
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
}`
|
|
256
|
+
},
|
|
257
|
+
{
|
|
258
|
+
id: 3,
|
|
259
|
+
title: 'Sử dụng Events để cập nhật UI',
|
|
260
|
+
code: `import { Component, signal } from '@angular/core';
|
|
261
|
+
import { LibsUiComponentsAudioComponent } from '@libs-ui/components-audio';
|
|
262
|
+
|
|
263
|
+
@Component({
|
|
264
|
+
selector: 'app-my-component',
|
|
265
|
+
template: \`
|
|
266
|
+
<libs_ui-components-audio
|
|
267
|
+
[fileAudio]="audioSource()"
|
|
268
|
+
[checkPermissionDownloadAudio]="checkPermission"
|
|
269
|
+
(outTimeUpdate)="handleTimeUpdate($event)"
|
|
270
|
+
(outVolumeControl)="handleVolumeChange($event)"
|
|
271
|
+
(outPlay)="handlePlayChange($event)"
|
|
272
|
+
(outMute)="handleMuteChange($event)"
|
|
273
|
+
(outEnded)="handleEnded()">
|
|
274
|
+
</libs_ui-components-audio>
|
|
275
|
+
|
|
276
|
+
<div class="audio-info">
|
|
277
|
+
<p>Trạng thái: {{ isPlaying() ? 'Đang phát' : 'Tạm dừng' }}</p>
|
|
278
|
+
<p>Thời gian hiện tại: {{ currentTime() }}</p>
|
|
279
|
+
<p>Tổng thời gian: {{ duration() }}</p>
|
|
280
|
+
<p>Âm lượng: {{ volumeLevel() }}%</p>
|
|
281
|
+
</div>
|
|
282
|
+
\`
|
|
283
|
+
})
|
|
284
|
+
export class MyComponent {
|
|
285
|
+
audioSource = signal<string>('path/to/audio.mp3');
|
|
286
|
+
isPlaying = signal<boolean>(false);
|
|
287
|
+
isMuted = signal<boolean>(false);
|
|
288
|
+
currentTime = signal<string>('00:00:00');
|
|
289
|
+
duration = signal<string>('00:00:00');
|
|
290
|
+
volumeLevel = signal<number>(100);
|
|
291
|
+
|
|
292
|
+
checkPermission = (): Promise<boolean> => {
|
|
293
|
+
return Promise.resolve(true);
|
|
294
|
+
}
|
|
295
|
+
|
|
296
|
+
handleTimeUpdate(timeInfo: { currentTime: string, duration: string }) {
|
|
297
|
+
this.currentTime.set(timeInfo.currentTime);
|
|
298
|
+
this.duration.set(timeInfo.duration);
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
handleVolumeChange(volume: number) {
|
|
302
|
+
this.volumeLevel.set(volume);
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
handlePlayChange(isPlaying: boolean) {
|
|
306
|
+
this.isPlaying.set(isPlaying);
|
|
307
|
+
}
|
|
308
|
+
|
|
309
|
+
handleMuteChange(isMuted: boolean) {
|
|
310
|
+
this.isMuted.set(isMuted);
|
|
311
|
+
}
|
|
312
|
+
|
|
313
|
+
handleEnded() {
|
|
314
|
+
this.isPlaying.set(false);
|
|
315
|
+
}
|
|
316
|
+
}`
|
|
317
|
+
}
|
|
318
|
+
]);
|
|
319
|
+
// Control methods
|
|
320
|
+
playAudio() {
|
|
321
|
+
if (this.functionControls) {
|
|
322
|
+
this.functionControls.playPause();
|
|
323
|
+
this.isPlaying.set(this.functionControls.isPlaying());
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
toggleMute() {
|
|
327
|
+
if (this.functionControls) {
|
|
328
|
+
this.functionControls.toggleMute();
|
|
329
|
+
this.isMuted.set(this.functionControls.isMuted());
|
|
330
|
+
}
|
|
331
|
+
}
|
|
332
|
+
changeVolume(event) {
|
|
333
|
+
if (this.functionControls && event.target) {
|
|
334
|
+
const value = parseInt(event.target.value);
|
|
335
|
+
this.volume.set(value);
|
|
336
|
+
this.functionControls.setVolume(value);
|
|
337
|
+
this.isMuted.set(this.functionControls.isMuted());
|
|
338
|
+
}
|
|
339
|
+
}
|
|
340
|
+
downloadAudio() {
|
|
341
|
+
if (this.functionControls) {
|
|
342
|
+
this.functionControls.download();
|
|
343
|
+
}
|
|
344
|
+
}
|
|
345
|
+
/**
|
|
346
|
+
* Format thời gian từ giây sang chuỗi HH:MM:SS
|
|
347
|
+
*/
|
|
348
|
+
formatTime(timeInSeconds) {
|
|
349
|
+
if (isNaN(timeInSeconds))
|
|
350
|
+
return '00:00:00';
|
|
351
|
+
const hours = Math.floor(timeInSeconds / 3600);
|
|
352
|
+
const minutes = Math.floor((timeInSeconds - (hours * 3600)) / 60);
|
|
353
|
+
const seconds = Math.floor(timeInSeconds - (hours * 3600) - (minutes * 60));
|
|
354
|
+
return `${hours.toString().padStart(2, '0')}:${minutes.toString().padStart(2, '0')}:${seconds.toString().padStart(2, '0')}`;
|
|
355
|
+
}
|
|
356
|
+
/**
|
|
357
|
+
* Sao chép text vào clipboard
|
|
358
|
+
*/
|
|
359
|
+
copyToClipboard(text) {
|
|
360
|
+
navigator.clipboard.writeText(text)
|
|
361
|
+
.then(() => {
|
|
362
|
+
alert('Đã sao chép vào clipboard!');
|
|
363
|
+
})
|
|
364
|
+
.catch(err => {
|
|
365
|
+
console.error('Failed to copy: ', err);
|
|
366
|
+
});
|
|
367
|
+
}
|
|
368
|
+
/**
|
|
369
|
+
* Đăng ký các function control từ component
|
|
370
|
+
*/
|
|
371
|
+
registerFunctions(event) {
|
|
372
|
+
this.functionControls = event;
|
|
373
|
+
// Initialize state
|
|
374
|
+
if (this.functionControls) {
|
|
375
|
+
this.isPlaying.set(this.functionControls.isPlaying());
|
|
376
|
+
this.isMuted.set(this.functionControls.isMuted());
|
|
377
|
+
}
|
|
378
|
+
}
|
|
379
|
+
/**
|
|
380
|
+
* Xử lý sự kiện thay đổi thời gian
|
|
381
|
+
*/
|
|
382
|
+
handleTimeUpdate(timeInfo) {
|
|
383
|
+
this.currentTime.set(timeInfo.currentTime);
|
|
384
|
+
this.duration.set(timeInfo.duration);
|
|
385
|
+
// Calculate progress based on currentTime and duration
|
|
386
|
+
const regex = /(\d+):(\d+):(\d+)/;
|
|
387
|
+
const currentMatches = timeInfo.currentTime.match(regex);
|
|
388
|
+
const durationMatches = timeInfo.duration.match(regex);
|
|
389
|
+
if (currentMatches && durationMatches) {
|
|
390
|
+
const currentSeconds = parseInt(currentMatches[1]) * 3600 +
|
|
391
|
+
parseInt(currentMatches[2]) * 60 +
|
|
392
|
+
parseInt(currentMatches[3]);
|
|
393
|
+
const totalSeconds = parseInt(durationMatches[1]) * 3600 +
|
|
394
|
+
parseInt(durationMatches[2]) * 60 +
|
|
395
|
+
parseInt(durationMatches[3]);
|
|
396
|
+
if (totalSeconds > 0) {
|
|
397
|
+
this.progress.set(Math.floor((currentSeconds / totalSeconds) * 100));
|
|
398
|
+
}
|
|
399
|
+
}
|
|
400
|
+
}
|
|
401
|
+
/**
|
|
402
|
+
* Xử lý sự kiện thay đổi âm lượng
|
|
403
|
+
*/
|
|
404
|
+
handleVolumeChange(volume) {
|
|
405
|
+
this.volume.set(volume);
|
|
406
|
+
}
|
|
407
|
+
/**
|
|
408
|
+
* Xử lý sự kiện thay đổi trạng thái phát
|
|
409
|
+
*/
|
|
410
|
+
handlePlayChange(isPlaying) {
|
|
411
|
+
this.isPlaying.set(isPlaying);
|
|
412
|
+
}
|
|
413
|
+
/**
|
|
414
|
+
* Xử lý sự kiện thay đổi trạng thái tắt tiếng
|
|
415
|
+
*/
|
|
416
|
+
handleMuteChange(isMuted) {
|
|
417
|
+
this.isMuted.set(isMuted);
|
|
418
|
+
}
|
|
419
|
+
/**
|
|
420
|
+
* Xử lý sự kiện kết thúc phát
|
|
421
|
+
*/
|
|
422
|
+
handleEnded() {
|
|
423
|
+
this.isPlaying.set(false);
|
|
424
|
+
}
|
|
425
|
+
/**
|
|
426
|
+
* Chọn mẫu âm thanh
|
|
427
|
+
*/
|
|
428
|
+
selectAudio(id) {
|
|
429
|
+
this.selectedAudio.set(id);
|
|
430
|
+
const sample = this.audioSamples().find(audio => audio.id === id);
|
|
431
|
+
if (sample) {
|
|
432
|
+
// Reset audio state
|
|
433
|
+
this.isPlaying.set(false);
|
|
434
|
+
this.progress.set(0);
|
|
435
|
+
this.currentTime.set('00:00:00');
|
|
436
|
+
this.duration.set('00:00:00');
|
|
437
|
+
// Simply update the source - the component will handle the reload
|
|
438
|
+
this.currentAudioSource.set(sample.src);
|
|
439
|
+
}
|
|
440
|
+
}
|
|
441
|
+
/**
|
|
442
|
+
* Kiểm tra quyền tải xuống
|
|
443
|
+
*/
|
|
444
|
+
checkDownloadPermission = () => {
|
|
445
|
+
return Promise.resolve(true);
|
|
446
|
+
};
|
|
447
|
+
/**
|
|
448
|
+
* Thay đổi vị trí phát bằng cách click trực tiếp vào thanh tiến độ
|
|
449
|
+
*/
|
|
450
|
+
seekAudioByClick(event) {
|
|
451
|
+
if (this.functionControls) {
|
|
452
|
+
const progressBar = event.currentTarget;
|
|
453
|
+
const rect = progressBar.getBoundingClientRect();
|
|
454
|
+
const offsetX = event.clientX - rect.left;
|
|
455
|
+
const percentX = (offsetX / rect.width) * 100;
|
|
456
|
+
// Đảm bảo giá trị nằm trong khoảng 0-100
|
|
457
|
+
const clampedPercent = Math.max(0, Math.min(100, percentX));
|
|
458
|
+
this.progress.set(Math.round(clampedPercent));
|
|
459
|
+
// Gọi method seekTo để cập nhật vị trí phát
|
|
460
|
+
this.functionControls.seekTo(this.progress());
|
|
461
|
+
}
|
|
462
|
+
}
|
|
463
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: LibsUiComponentsAudioDemoComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
464
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: LibsUiComponentsAudioDemoComponent, isStandalone: true, selector: "lib-audio-demo", viewQueries: [{ propertyName: "audioPlayer", first: true, predicate: ["audioPlayer"], descendants: true }], ngImport: i0, template: "<div class=\"max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 font-sans text-gray-800\">\n <header class=\"text-center py-10 bg-white rounded-lg shadow mb-8\">\n <h1 class=\"text-4xl font-bold text-gray-800 mb-2\">Demo Tr\u00ECnh Ph\u00E1t \u00C2m Thanh</h1>\n <p class=\"text-xl text-gray-600\">Th\u01B0 vi\u1EC7n component cho Angular \u0111\u1EC3 ph\u00E1t v\u00E0 \u0111i\u1EC1u khi\u1EC3n \u00E2m thanh</p>\n </header>\n\n <main>\n <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">Gi\u1EDBi thi\u1EC7u</h2>\n <p class=\"text-lg\">\n <code class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">@libs-ui/components-audio</code> l\u00E0 m\u1ED9t\n component Angular m\u1EA1nh m\u1EBD cho ph\u00E9p ng\u01B0\u1EDDi d\u00F9ng ph\u00E1t, t\u1EA1m d\u1EEBng v\u00E0 \u0111i\u1EC1u khi\u1EC3n c\u00E1c t\u1EC7p \u00E2m thanh v\u1EDBi nhi\u1EC1u t\u00EDnh n\u0103ng.\n </p>\n </section>\n\n <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">C\u00E0i \u0111\u1EB7t</h2>\n <p>\u0110\u1EC3 c\u00E0i \u0111\u1EB7t th\u01B0 vi\u1EC7n, s\u1EED d\u1EE5ng npm ho\u1EB7c yarn:</p>\n\n <div class=\"relative my-4\">\n <pre\n class=\"bg-gray-100 p-4 rounded overflow-x-auto mb-4\"><code class=\"font-mono\">npm install @libs-ui/components-audio</code></pre>\n <button\n class=\"absolute top-2 right-2 bg-blue-500 text-white px-3 py-1 rounded text-sm hover:bg-blue-600 transition\"\n (click)=\"copyToClipboard('npm install @libs-ui/components-audio')\">\n Sao ch\u00E9p\n </button>\n </div>\n\n <p>Ho\u1EB7c v\u1EDBi yarn:</p>\n\n <div class=\"relative my-4\">\n <pre\n class=\"bg-gray-100 p-4 rounded overflow-x-auto mb-4\"><code class=\"font-mono\">yarn add @libs-ui/components-audio</code></pre>\n <button\n class=\"absolute top-2 right-2 bg-blue-500 text-white px-3 py-1 rounded text-sm hover:bg-blue-600 transition\"\n (click)=\"copyToClipboard('yarn add @libs-ui/components-audio')\">\n Sao ch\u00E9p\n </button>\n </div>\n </section>\n\n <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">Demo tr\u1EF1c ti\u1EBFp</h2>\n <div class=\"bg-gray-50 p-6 rounded-lg\">\n <p class=\"mb-4\">Th\u1EED nghi\u1EC7m tr\u00ECnh ph\u00E1t \u00E2m thanh v\u1EDBi c\u00E1c file m\u1EABu:</p>\n\n <div class=\"mb-6\">\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Ch\u1ECDn file \u00E2m thanh:</h3>\n <div class=\"flex flex-col gap-2\">\n @for (audio of audioSamples(); track audio.id) {\n <div\n class=\"flex justify-between p-3 bg-white rounded border cursor-pointer transition hover:bg-blue-50 hover:border-blue-500\"\n [class.bg-blue-50]=\"selectedAudio() === audio.id\"\n [class.border-blue-500]=\"selectedAudio() === audio.id\"\n [class.font-medium]=\"selectedAudio() === audio.id\"\n (click)=\"selectAudio(audio.id)\">\n <span>{{ audio.name }}</span>\n <span>{{ audio.duration }}</span>\n </div>\n }\n </div>\n </div>\n\n <div class=\"mb-6\">\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Tr\u00ECnh ph\u00E1t:</h3>\n <libs_ui-components-audio #audioPlayer\n [fileAudio]=\"currentAudioSource()\"\n [checkPermissionDownloadAudio]=\"checkDownloadPermission\"\n (outFunctionsControl)=\"registerFunctions($event)\"\n (outTimeUpdate)=\"handleTimeUpdate($event)\"\n (outVolumeControl)=\"handleVolumeChange($event)\"\n (outPlay)=\"handlePlayChange($event)\"\n (outMute)=\"handleMuteChange($event)\"\n (outEnded)=\"handleEnded()\">\n </libs_ui-components-audio>\n </div>\n\n <div class=\"mb-6\">\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">\u0110i\u1EC1u khi\u1EC3n qua Function Control:</h3>\n <div class=\"flex flex-wrap gap-3 items-center bg-white p-4 rounded border\">\n <button class=\"px-4 py-2 bg-blue-500 text-white font-medium rounded hover:bg-blue-600 transition\"\n (click)=\"playAudio()\">\n Ph\u00E1t/T\u1EA1m d\u1EEBng\n </button>\n <button class=\"px-4 py-2 bg-blue-500 text-white font-medium rounded hover:bg-blue-600 transition\"\n (click)=\"toggleMute()\">\n {{ isMuted() ? 'B\u1EADt ti\u1EBFng' : 'T\u1EAFt ti\u1EBFng' }}\n </button>\n\n <div class=\"flex items-center gap-2 ml-2\">\n <span>\u00C2m l\u01B0\u1EE3ng:</span>\n <input type=\"range\"\n min=\"0\"\n max=\"100\"\n [value]=\"volumePercent()\"\n (input)=\"changeVolume($event)\"\n class=\"w-24 volume-slider\"\n [style.--volume-percent.%]=\"volumePercent()\">\n <span>{{ volumePercent() }}%</span>\n </div>\n\n <div class=\"w-full mt-2\">\n <div class=\"flex items-center gap-2\">\n <span>Ti\u1EBFn \u0111\u1ED9:</span>\n <div class=\"relative h-2 bg-gray-200 rounded-full flex-1 cursor-pointer\"\n (click)=\"seekAudioByClick($event)\">\n <div class=\"absolute top-0 left-0 h-full bg-blue-500 rounded-full\"\n [style.width.%]=\"progress()\"></div>\n </div>\n <span>{{ progress() }}%</span>\n </div>\n </div>\n\n <button class=\"px-4 py-2 bg-green-500 text-white font-medium rounded hover:bg-green-600 transition mt-4\"\n (click)=\"downloadAudio()\">\n T\u1EA3i xu\u1ED1ng\n </button>\n </div>\n\n <div class=\"mt-6\">\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Tr\u1EA1ng th\u00E1i:</h3>\n <div class=\"bg-white p-4 rounded border\">\n <div class=\"mb-2\">\n <strong>Tr\u1EA1ng th\u00E1i:</strong>\n <span [class.text-green-600]=\"isPlaying()\"\n [class.text-red-600]=\"!isPlaying()\">\n {{ isPlaying() ? '\u0110ang ph\u00E1t' : 'T\u1EA1m d\u1EEBng' }}\n </span>\n </div>\n <div class=\"mb-2\">\n <strong>Th\u1EDDi gian hi\u1EC7n t\u1EA1i:</strong>\n <span>{{ currentTime() }}</span>\n </div>\n <div class=\"mb-2\">\n <strong>T\u1ED5ng th\u1EDDi gian:</strong>\n <span>{{ duration() }}</span>\n </div>\n <div class=\"mb-2\">\n <strong>Ti\u1EBFn \u0111\u1ED9:</strong>\n <span>{{ progress() }}%</span>\n </div>\n </div>\n </div>\n </div>\n </div>\n </section>\n\n <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">T\u00E0i li\u1EC7u API</h2>\n\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Inputs</h3>\n <div class=\"overflow-x-auto mb-8\">\n <table class=\"min-w-full bg-white\">\n <thead class=\"bg-gray-100\">\n <tr>\n <th class=\"py-3 px-4 text-left\">T\u00EAn</th>\n <th class=\"py-3 px-4 text-left\">Ki\u1EC3u d\u1EEF li\u1EC7u</th>\n <th class=\"py-3 px-4 text-left\">M\u1EB7c \u0111\u1ECBnh</th>\n <th class=\"py-3 px-4 text-left\">M\u00F4 t\u1EA3</th>\n </tr>\n </thead>\n <tbody>\n @for (input of inputsDoc(); track input.name) {\n <tr class=\"border-b hover:bg-gray-50\">\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ input.name }}</code></td>\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ input.type }}</code></td>\n <td class=\"py-3 px-4\">{{ input.default }}</td>\n <td class=\"py-3 px-4\">{{ input.description }}</td>\n </tr>\n }\n </tbody>\n </table>\n </div>\n\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Outputs</h3>\n <div class=\"overflow-x-auto mb-8\">\n <table class=\"min-w-full bg-white\">\n <thead class=\"bg-gray-100\">\n <tr>\n <th class=\"py-3 px-4 text-left\">T\u00EAn</th>\n <th class=\"py-3 px-4 text-left\">Ki\u1EC3u d\u1EEF li\u1EC7u</th>\n <th class=\"py-3 px-4 text-left\">M\u00F4 t\u1EA3</th>\n </tr>\n </thead>\n <tbody>\n @for (output of outputsDoc(); track output.name) {\n <tr class=\"border-b hover:bg-gray-50\">\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ output.name }}</code></td>\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ output.type }}</code></td>\n <td class=\"py-3 px-4\">{{ output.description }}</td>\n </tr>\n }\n </tbody>\n </table>\n </div>\n\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Function Control Methods</h3>\n <div class=\"overflow-x-auto mb-8\">\n <table class=\"min-w-full bg-white\">\n <thead class=\"bg-gray-100\">\n <tr>\n <th class=\"py-3 px-4 text-left\">T\u00EAn</th>\n <th class=\"py-3 px-4 text-left\">Tham s\u1ED1</th>\n <th class=\"py-3 px-4 text-left\">Ki\u1EC3u tr\u1EA3 v\u1EC1</th>\n <th class=\"py-3 px-4 text-left\">M\u00F4 t\u1EA3</th>\n </tr>\n </thead>\n <tbody>\n @for (method of methodsDoc(); track method.name) {\n <tr class=\"border-b hover:bg-gray-50\">\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ method.name }}</code></td>\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ method.params }}</code></td>\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ method.returnType }}</code></td>\n <td class=\"py-3 px-4\">{{ method.description }}</td>\n </tr>\n }\n </tbody>\n </table>\n </div>\n\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Interfaces</h3>\n <div class=\"overflow-x-auto mb-8\">\n @for (interfaceDoc of interfacesDoc(); track interfaceDoc.name) {\n <div class=\"mb-6 bg-white p-4 rounded border\">\n <h4 class=\"text-lg font-semibold text-gray-700 mb-2\">{{ interfaceDoc.name }}</h4>\n <p class=\"mb-4\">{{ interfaceDoc.description }}</p>\n\n <table class=\"min-w-full bg-white\">\n <thead class=\"bg-gray-100\">\n <tr>\n <th class=\"py-3 px-4 text-left\">T\u00EAn</th>\n <th class=\"py-3 px-4 text-left\">Ki\u1EC3u d\u1EEF li\u1EC7u</th>\n <th class=\"py-3 px-4 text-left\">M\u00F4 t\u1EA3</th>\n </tr>\n </thead>\n <tbody>\n @for (prop of interfaceDoc.properties; track prop.name) {\n <tr class=\"border-b hover:bg-gray-50\">\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ prop.name }}</code></td>\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ prop.type }}</code></td>\n <td class=\"py-3 px-4\">{{ prop.description }}</td>\n </tr>\n }\n </tbody>\n </table>\n </div>\n }\n </div>\n </section>\n\n <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">T\u00EDnh n\u0103ng</h2>\n <ul class=\"space-y-6\">\n @for (feature of features(); track feature.id) {\n <li class=\"flex items-start\">\n <span class=\"text-3xl mr-4 min-w-10 text-center\">{{ feature.icon }}</span>\n <div>\n <h3 class=\"text-xl font-semibold text-gray-700\">{{ feature.title }}</h3>\n <p>{{ feature.description }}</p>\n </div>\n </li>\n }\n </ul>\n </section>\n\n <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">C\u00E1ch s\u1EED d\u1EE5ng</h2>\n <div class=\"flex flex-col gap-4\">\n @for (example of codeExamples(); track example.id) {\n <div class=\"border border-gray-200 rounded-lg p-4\">\n <h3 class=\"text-xl font-semibold text-gray-700 mb-2\">{{ example.title }}</h3>\n <pre\n class=\"bg-gray-100 p-4 rounded overflow-x-auto\"><code [innerHTML]=\"example.code\" class=\"font-mono\"></code></pre>\n </div>\n }\n </div>\n </section>\n </main>\n</div>\n", styles: ["input[type=range]{cursor:pointer;-webkit-appearance:none;appearance:none;height:6px;border-radius:999px;background-color:#e5e7eb;outline:none;width:100%}input[type=range].volume-slider{background:linear-gradient(to right,#3b82f6 var(--volume-percent, 50%),#e5e7eb var(--volume-percent, 50%))}input[type=range]::-webkit-slider-runnable-track{width:100%;height:6px;background-color:#e5e7eb;border-radius:999px}input[type=range]::-moz-range-track{width:100%;height:6px;background-color:#e5e7eb;border-radius:999px}input[type=range]::-webkit-slider-thumb{-webkit-appearance:none;appearance:none;width:16px;height:16px;border-radius:50%;background-color:#3b82f6;cursor:pointer;transition:background-color .2s ease;margin-top:-5px}input[type=range]::-moz-range-thumb{width:16px;height:16px;border-radius:50%;background-color:#3b82f6;cursor:pointer;transition:background-color .2s ease;border:none}input[type=range]::-webkit-slider-thumb:hover{background-color:#2563eb}input[type=range]::-moz-range-thumb:hover{background-color:#2563eb}input[type=range].volume-slider::-webkit-slider-runnable-track{background:linear-gradient(to right,#3b82f6 var(--volume-percent, 50%),#e5e7eb var(--volume-percent, 50%))}input[type=range].volume-slider::-moz-range-track{background:linear-gradient(to right,#3b82f6 var(--volume-percent, 50%),#e5e7eb var(--volume-percent, 50%))}.control-button:active{transform:scale(.95)}libs_ui-components-audio{display:block;width:100%}.transition{transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.3s}pre::-webkit-scrollbar{height:8px}pre::-webkit-scrollbar-track{background:#f1f1f1;border-radius:4px}pre::-webkit-scrollbar-thumb{background:#c1c1c1;border-radius:4px}pre::-webkit-scrollbar-thumb:hover{background:#a1a1a1}\n"], dependencies: [{ kind: "component", type: LibsUiComponentsAudioComponent, selector: "libs_ui-components-audio", inputs: ["fileAudio", "checkPermissionDownloadAudio"], outputs: ["outFunctionsControl", "outVolumeControl", "outTimeUpdate", "outEnded", "outMute", "outPlay"] }] });
|
|
465
|
+
}
|
|
466
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: LibsUiComponentsAudioDemoComponent, decorators: [{
|
|
467
|
+
type: Component,
|
|
468
|
+
args: [{ selector: 'lib-audio-demo', standalone: true, imports: [LibsUiComponentsAudioComponent], template: "<div class=\"max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 font-sans text-gray-800\">\n <header class=\"text-center py-10 bg-white rounded-lg shadow mb-8\">\n <h1 class=\"text-4xl font-bold text-gray-800 mb-2\">Demo Tr\u00ECnh Ph\u00E1t \u00C2m Thanh</h1>\n <p class=\"text-xl text-gray-600\">Th\u01B0 vi\u1EC7n component cho Angular \u0111\u1EC3 ph\u00E1t v\u00E0 \u0111i\u1EC1u khi\u1EC3n \u00E2m thanh</p>\n </header>\n\n <main>\n <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">Gi\u1EDBi thi\u1EC7u</h2>\n <p class=\"text-lg\">\n <code class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">@libs-ui/components-audio</code> l\u00E0 m\u1ED9t\n component Angular m\u1EA1nh m\u1EBD cho ph\u00E9p ng\u01B0\u1EDDi d\u00F9ng ph\u00E1t, t\u1EA1m d\u1EEBng v\u00E0 \u0111i\u1EC1u khi\u1EC3n c\u00E1c t\u1EC7p \u00E2m thanh v\u1EDBi nhi\u1EC1u t\u00EDnh n\u0103ng.\n </p>\n </section>\n\n <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">C\u00E0i \u0111\u1EB7t</h2>\n <p>\u0110\u1EC3 c\u00E0i \u0111\u1EB7t th\u01B0 vi\u1EC7n, s\u1EED d\u1EE5ng npm ho\u1EB7c yarn:</p>\n\n <div class=\"relative my-4\">\n <pre\n class=\"bg-gray-100 p-4 rounded overflow-x-auto mb-4\"><code class=\"font-mono\">npm install @libs-ui/components-audio</code></pre>\n <button\n class=\"absolute top-2 right-2 bg-blue-500 text-white px-3 py-1 rounded text-sm hover:bg-blue-600 transition\"\n (click)=\"copyToClipboard('npm install @libs-ui/components-audio')\">\n Sao ch\u00E9p\n </button>\n </div>\n\n <p>Ho\u1EB7c v\u1EDBi yarn:</p>\n\n <div class=\"relative my-4\">\n <pre\n class=\"bg-gray-100 p-4 rounded overflow-x-auto mb-4\"><code class=\"font-mono\">yarn add @libs-ui/components-audio</code></pre>\n <button\n class=\"absolute top-2 right-2 bg-blue-500 text-white px-3 py-1 rounded text-sm hover:bg-blue-600 transition\"\n (click)=\"copyToClipboard('yarn add @libs-ui/components-audio')\">\n Sao ch\u00E9p\n </button>\n </div>\n </section>\n\n <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">Demo tr\u1EF1c ti\u1EBFp</h2>\n <div class=\"bg-gray-50 p-6 rounded-lg\">\n <p class=\"mb-4\">Th\u1EED nghi\u1EC7m tr\u00ECnh ph\u00E1t \u00E2m thanh v\u1EDBi c\u00E1c file m\u1EABu:</p>\n\n <div class=\"mb-6\">\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Ch\u1ECDn file \u00E2m thanh:</h3>\n <div class=\"flex flex-col gap-2\">\n @for (audio of audioSamples(); track audio.id) {\n <div\n class=\"flex justify-between p-3 bg-white rounded border cursor-pointer transition hover:bg-blue-50 hover:border-blue-500\"\n [class.bg-blue-50]=\"selectedAudio() === audio.id\"\n [class.border-blue-500]=\"selectedAudio() === audio.id\"\n [class.font-medium]=\"selectedAudio() === audio.id\"\n (click)=\"selectAudio(audio.id)\">\n <span>{{ audio.name }}</span>\n <span>{{ audio.duration }}</span>\n </div>\n }\n </div>\n </div>\n\n <div class=\"mb-6\">\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Tr\u00ECnh ph\u00E1t:</h3>\n <libs_ui-components-audio #audioPlayer\n [fileAudio]=\"currentAudioSource()\"\n [checkPermissionDownloadAudio]=\"checkDownloadPermission\"\n (outFunctionsControl)=\"registerFunctions($event)\"\n (outTimeUpdate)=\"handleTimeUpdate($event)\"\n (outVolumeControl)=\"handleVolumeChange($event)\"\n (outPlay)=\"handlePlayChange($event)\"\n (outMute)=\"handleMuteChange($event)\"\n (outEnded)=\"handleEnded()\">\n </libs_ui-components-audio>\n </div>\n\n <div class=\"mb-6\">\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">\u0110i\u1EC1u khi\u1EC3n qua Function Control:</h3>\n <div class=\"flex flex-wrap gap-3 items-center bg-white p-4 rounded border\">\n <button class=\"px-4 py-2 bg-blue-500 text-white font-medium rounded hover:bg-blue-600 transition\"\n (click)=\"playAudio()\">\n Ph\u00E1t/T\u1EA1m d\u1EEBng\n </button>\n <button class=\"px-4 py-2 bg-blue-500 text-white font-medium rounded hover:bg-blue-600 transition\"\n (click)=\"toggleMute()\">\n {{ isMuted() ? 'B\u1EADt ti\u1EBFng' : 'T\u1EAFt ti\u1EBFng' }}\n </button>\n\n <div class=\"flex items-center gap-2 ml-2\">\n <span>\u00C2m l\u01B0\u1EE3ng:</span>\n <input type=\"range\"\n min=\"0\"\n max=\"100\"\n [value]=\"volumePercent()\"\n (input)=\"changeVolume($event)\"\n class=\"w-24 volume-slider\"\n [style.--volume-percent.%]=\"volumePercent()\">\n <span>{{ volumePercent() }}%</span>\n </div>\n\n <div class=\"w-full mt-2\">\n <div class=\"flex items-center gap-2\">\n <span>Ti\u1EBFn \u0111\u1ED9:</span>\n <div class=\"relative h-2 bg-gray-200 rounded-full flex-1 cursor-pointer\"\n (click)=\"seekAudioByClick($event)\">\n <div class=\"absolute top-0 left-0 h-full bg-blue-500 rounded-full\"\n [style.width.%]=\"progress()\"></div>\n </div>\n <span>{{ progress() }}%</span>\n </div>\n </div>\n\n <button class=\"px-4 py-2 bg-green-500 text-white font-medium rounded hover:bg-green-600 transition mt-4\"\n (click)=\"downloadAudio()\">\n T\u1EA3i xu\u1ED1ng\n </button>\n </div>\n\n <div class=\"mt-6\">\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Tr\u1EA1ng th\u00E1i:</h3>\n <div class=\"bg-white p-4 rounded border\">\n <div class=\"mb-2\">\n <strong>Tr\u1EA1ng th\u00E1i:</strong>\n <span [class.text-green-600]=\"isPlaying()\"\n [class.text-red-600]=\"!isPlaying()\">\n {{ isPlaying() ? '\u0110ang ph\u00E1t' : 'T\u1EA1m d\u1EEBng' }}\n </span>\n </div>\n <div class=\"mb-2\">\n <strong>Th\u1EDDi gian hi\u1EC7n t\u1EA1i:</strong>\n <span>{{ currentTime() }}</span>\n </div>\n <div class=\"mb-2\">\n <strong>T\u1ED5ng th\u1EDDi gian:</strong>\n <span>{{ duration() }}</span>\n </div>\n <div class=\"mb-2\">\n <strong>Ti\u1EBFn \u0111\u1ED9:</strong>\n <span>{{ progress() }}%</span>\n </div>\n </div>\n </div>\n </div>\n </div>\n </section>\n\n <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">T\u00E0i li\u1EC7u API</h2>\n\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Inputs</h3>\n <div class=\"overflow-x-auto mb-8\">\n <table class=\"min-w-full bg-white\">\n <thead class=\"bg-gray-100\">\n <tr>\n <th class=\"py-3 px-4 text-left\">T\u00EAn</th>\n <th class=\"py-3 px-4 text-left\">Ki\u1EC3u d\u1EEF li\u1EC7u</th>\n <th class=\"py-3 px-4 text-left\">M\u1EB7c \u0111\u1ECBnh</th>\n <th class=\"py-3 px-4 text-left\">M\u00F4 t\u1EA3</th>\n </tr>\n </thead>\n <tbody>\n @for (input of inputsDoc(); track input.name) {\n <tr class=\"border-b hover:bg-gray-50\">\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ input.name }}</code></td>\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ input.type }}</code></td>\n <td class=\"py-3 px-4\">{{ input.default }}</td>\n <td class=\"py-3 px-4\">{{ input.description }}</td>\n </tr>\n }\n </tbody>\n </table>\n </div>\n\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Outputs</h3>\n <div class=\"overflow-x-auto mb-8\">\n <table class=\"min-w-full bg-white\">\n <thead class=\"bg-gray-100\">\n <tr>\n <th class=\"py-3 px-4 text-left\">T\u00EAn</th>\n <th class=\"py-3 px-4 text-left\">Ki\u1EC3u d\u1EEF li\u1EC7u</th>\n <th class=\"py-3 px-4 text-left\">M\u00F4 t\u1EA3</th>\n </tr>\n </thead>\n <tbody>\n @for (output of outputsDoc(); track output.name) {\n <tr class=\"border-b hover:bg-gray-50\">\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ output.name }}</code></td>\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ output.type }}</code></td>\n <td class=\"py-3 px-4\">{{ output.description }}</td>\n </tr>\n }\n </tbody>\n </table>\n </div>\n\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Function Control Methods</h3>\n <div class=\"overflow-x-auto mb-8\">\n <table class=\"min-w-full bg-white\">\n <thead class=\"bg-gray-100\">\n <tr>\n <th class=\"py-3 px-4 text-left\">T\u00EAn</th>\n <th class=\"py-3 px-4 text-left\">Tham s\u1ED1</th>\n <th class=\"py-3 px-4 text-left\">Ki\u1EC3u tr\u1EA3 v\u1EC1</th>\n <th class=\"py-3 px-4 text-left\">M\u00F4 t\u1EA3</th>\n </tr>\n </thead>\n <tbody>\n @for (method of methodsDoc(); track method.name) {\n <tr class=\"border-b hover:bg-gray-50\">\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ method.name }}</code></td>\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ method.params }}</code></td>\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ method.returnType }}</code></td>\n <td class=\"py-3 px-4\">{{ method.description }}</td>\n </tr>\n }\n </tbody>\n </table>\n </div>\n\n <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Interfaces</h3>\n <div class=\"overflow-x-auto mb-8\">\n @for (interfaceDoc of interfacesDoc(); track interfaceDoc.name) {\n <div class=\"mb-6 bg-white p-4 rounded border\">\n <h4 class=\"text-lg font-semibold text-gray-700 mb-2\">{{ interfaceDoc.name }}</h4>\n <p class=\"mb-4\">{{ interfaceDoc.description }}</p>\n\n <table class=\"min-w-full bg-white\">\n <thead class=\"bg-gray-100\">\n <tr>\n <th class=\"py-3 px-4 text-left\">T\u00EAn</th>\n <th class=\"py-3 px-4 text-left\">Ki\u1EC3u d\u1EEF li\u1EC7u</th>\n <th class=\"py-3 px-4 text-left\">M\u00F4 t\u1EA3</th>\n </tr>\n </thead>\n <tbody>\n @for (prop of interfaceDoc.properties; track prop.name) {\n <tr class=\"border-b hover:bg-gray-50\">\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ prop.name }}</code></td>\n <td class=\"py-3 px-4\"><code\n class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ prop.type }}</code></td>\n <td class=\"py-3 px-4\">{{ prop.description }}</td>\n </tr>\n }\n </tbody>\n </table>\n </div>\n }\n </div>\n </section>\n\n <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">T\u00EDnh n\u0103ng</h2>\n <ul class=\"space-y-6\">\n @for (feature of features(); track feature.id) {\n <li class=\"flex items-start\">\n <span class=\"text-3xl mr-4 min-w-10 text-center\">{{ feature.icon }}</span>\n <div>\n <h3 class=\"text-xl font-semibold text-gray-700\">{{ feature.title }}</h3>\n <p>{{ feature.description }}</p>\n </div>\n </li>\n }\n </ul>\n </section>\n\n <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">C\u00E1ch s\u1EED d\u1EE5ng</h2>\n <div class=\"flex flex-col gap-4\">\n @for (example of codeExamples(); track example.id) {\n <div class=\"border border-gray-200 rounded-lg p-4\">\n <h3 class=\"text-xl font-semibold text-gray-700 mb-2\">{{ example.title }}</h3>\n <pre\n class=\"bg-gray-100 p-4 rounded overflow-x-auto\"><code [innerHTML]=\"example.code\" class=\"font-mono\"></code></pre>\n </div>\n }\n </div>\n </section>\n </main>\n</div>\n", styles: ["input[type=range]{cursor:pointer;-webkit-appearance:none;appearance:none;height:6px;border-radius:999px;background-color:#e5e7eb;outline:none;width:100%}input[type=range].volume-slider{background:linear-gradient(to right,#3b82f6 var(--volume-percent, 50%),#e5e7eb var(--volume-percent, 50%))}input[type=range]::-webkit-slider-runnable-track{width:100%;height:6px;background-color:#e5e7eb;border-radius:999px}input[type=range]::-moz-range-track{width:100%;height:6px;background-color:#e5e7eb;border-radius:999px}input[type=range]::-webkit-slider-thumb{-webkit-appearance:none;appearance:none;width:16px;height:16px;border-radius:50%;background-color:#3b82f6;cursor:pointer;transition:background-color .2s ease;margin-top:-5px}input[type=range]::-moz-range-thumb{width:16px;height:16px;border-radius:50%;background-color:#3b82f6;cursor:pointer;transition:background-color .2s ease;border:none}input[type=range]::-webkit-slider-thumb:hover{background-color:#2563eb}input[type=range]::-moz-range-thumb:hover{background-color:#2563eb}input[type=range].volume-slider::-webkit-slider-runnable-track{background:linear-gradient(to right,#3b82f6 var(--volume-percent, 50%),#e5e7eb var(--volume-percent, 50%))}input[type=range].volume-slider::-moz-range-track{background:linear-gradient(to right,#3b82f6 var(--volume-percent, 50%),#e5e7eb var(--volume-percent, 50%))}.control-button:active{transform:scale(.95)}libs_ui-components-audio{display:block;width:100%}.transition{transition-property:all;transition-timing-function:cubic-bezier(.4,0,.2,1);transition-duration:.3s}pre::-webkit-scrollbar{height:8px}pre::-webkit-scrollbar-track{background:#f1f1f1;border-radius:4px}pre::-webkit-scrollbar-thumb{background:#c1c1c1;border-radius:4px}pre::-webkit-scrollbar-thumb:hover{background:#a1a1a1}\n"] }]
|
|
469
|
+
}], propDecorators: { audioPlayer: [{
|
|
470
|
+
type: ViewChild,
|
|
471
|
+
args: ['audioPlayer']
|
|
472
|
+
}] } });
|
|
473
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audio-demo.component.js","sourceRoot":"","sources":["../../../../../../libs-ui/components/audio/src/demo/audio-demo.component.ts","../../../../../../libs-ui/components/audio/src/demo/audio-demo.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ViewChild, computed, signal } from '@angular/core';\nimport { LibsUiComponentsAudioComponent } from '../audio.component';\nimport { IAudioFunctionControlEvent } from '../interfaces/function-control-event.interface';\n\n@Component({\n  selector: 'lib-audio-demo',\n  standalone: true,\n  imports: [LibsUiComponentsAudioComponent],\n  templateUrl: './audio-demo.component.html',\n  styleUrls: ['./audio-demo.component.css']\n})\nexport class LibsUiComponentsAudioDemoComponent {\n  @ViewChild('audioPlayer') audioPlayer!: LibsUiComponentsAudioComponent;\n\n  // State using signals\n  public isPlaying = signal<boolean>(false);\n  public isMuted = signal<boolean>(false);\n  public volume = signal<number>(100);\n  public currentTime = signal<string>('00:00:00');\n  public duration = signal<string>('00:00:00');\n  public progress = signal<number>(0);\n\n  // Computed properties for template display\n  public volumePercent = computed(() => Math.round(this.volume()));\n\n  // Selected audio sample\n  public selectedAudio = signal<number>(1);\n  public currentAudioSource = signal<string>('https://nhacchuong123.com/nhac-chuong/abcdefgh/nhac-chuong-nguoi-ay-dau-co-dang-akira-phan-nguyen-van-chung.mp3');\n\n  // Function Control variables\n  public functionControls: IAudioFunctionControlEvent | null = null;\n\n  // Audio samples for demo\n  public audioSamples = signal<Array<{ id: number, name: string, src: string, duration: string }>>([\n    {\n      id: 1,\n      name: 'Bản nhạc mẫu 1',\n      src: 'https://nhacchuong123.com/nhac-chuong/abcdefgh/nhac-chuong-nguoi-ay-dau-co-dang-akira-phan-nguyen-van-chung.mp3',\n      duration: '01:30'\n    },\n    {\n      id: 2,\n      name: 'Bản nhạc mẫu 2',\n      src: 'https://www.soundhelix.com/examples/mp3/SoundHelix-Song-1.mp3',\n      duration: '02:15'\n    },\n    {\n      id: 3,\n      name: 'Bản nhạc mẫu 3',\n      src: 'https://dl.dropboxusercontent.com/s/75jpngrgnavyu1f/The-Noisy-Freaks.mp3',\n      duration: '03:45'\n    }\n  ]);\n\n  // API Documentation data\n  public inputsDoc = signal<Array<{ name: string, type: string, default: string, description: string }>>([\n    {\n      name: 'fileAudio',\n      type: 'string',\n      default: 'Bắt buộc',\n      description: 'URL của file audio cần phát'\n    },\n    {\n      name: 'checkPermissionDownloadAudio',\n      type: '() => Promise<boolean>',\n      default: 'Bắt buộc',\n      description: 'Function trả về promise với kết quả boolean cho biết nếu được phép download'\n    }\n  ]);\n\n  // Output documentation\n  public outputsDoc = signal<Array<{ name: string, type: string, description: string }>>([\n    {\n      name: 'outFunctionsControl',\n      type: 'IAudioFunctionControlEvent',\n      description: 'Emits các hàm điều khiển audio'\n    },\n    {\n      name: 'outVolumeControl',\n      type: 'number',\n      description: 'Emits giá trị âm lượng hiện tại (0-100)'\n    },\n    {\n      name: 'outTimeUpdate',\n      type: '{ currentTime: string, duration: string }',\n      description: 'Emits thông tin thời gian hiện tại và tổng thời gian'\n    },\n    {\n      name: 'outEnded',\n      type: 'void',\n      description: 'Emits khi audio kết thúc phát'\n    },\n    {\n      name: 'outMute',\n      type: 'boolean',\n      description: 'Emits trạng thái tắt/bật tiếng'\n    },\n    {\n      name: 'outPlay',\n      type: 'boolean',\n      description: 'Emits trạng thái phát/tạm dừng'\n    }\n  ]);\n\n  // Interface documentation\n  public interfacesDoc = signal<Array<{ name: string, description: string, properties: Array<{ name: string, type: string, description: string }> }>>([\n    {\n      name: 'IAudioFunctionControlEvent',\n      description: 'Interface cho các chức năng điều khiển audio được cung cấp qua output event',\n      properties: [\n        {\n          name: 'playPause',\n          type: '(event?: Event) => void',\n          description: 'Bắt đầu hoặc tạm dừng phát audio'\n        },\n        {\n          name: 'toggleMute',\n          type: '(event?: Event) => void',\n          description: 'Bật hoặc tắt âm thanh'\n        },\n        {\n          name: 'setVolume',\n          type: '(value: number) => void',\n          description: 'Điều chỉnh âm lượng (giá trị từ 0 đến 100)'\n        },\n        {\n          name: 'seekTo',\n          type: '(value: number) => void',\n          description: 'Di chuyển đến vị trí cụ thể trong audio (giá trị từ 0 đến 100)'\n        },\n        {\n          name: 'download',\n          type: '(event?: Event) => void',\n          description: 'Tải xuống file audio'\n        },\n        {\n          name: 'isPlaying',\n          type: '() => boolean',\n          description: 'Kiểm tra trạng thái đang phát audio'\n        },\n        {\n          name: 'isMuted',\n          type: '() => boolean',\n          description: 'Kiểm tra trạng thái tắt tiếng'\n        }\n      ]\n    }\n  ]);\n\n  // Method documentation\n  public methodsDoc = signal<Array<{ name: string, params: string, returnType: string, description: string }>>([\n    {\n      name: 'playPause',\n      params: 'event?: Event',\n      returnType: 'void',\n      description: 'Phát hoặc tạm dừng audio'\n    },\n    {\n      name: 'toggleMute',\n      params: 'event?: Event',\n      returnType: 'void',\n      description: 'Bật/tắt âm thanh'\n    },\n    {\n      name: 'seekTo',\n      params: 'value: number',\n      returnType: 'void',\n      description: 'Di chuyển đến vị trí cụ thể trong audio (giá trị từ 0-100)'\n    },\n    {\n      name: 'setVolume',\n      params: 'value: number',\n      returnType: 'void',\n      description: 'Điều chỉnh âm lượng (giá trị từ 0-100)'\n    },\n    {\n      name: 'download',\n      params: 'event?: Event',\n      returnType: 'void',\n      description: 'Tải xuống file audio'\n    },\n    {\n      name: 'isPlaying',\n      params: '',\n      returnType: 'boolean',\n      description: 'Kiểm tra nếu audio đang phát'\n    },\n    {\n      name: 'isMuted',\n      params: '',\n      returnType: 'boolean',\n      description: 'Kiểm tra nếu audio đang tắt tiếng'\n    }\n  ]);\n\n  // Features data\n  public features = signal<Array<{ id: number, icon: string, title: string, description: string }>>([\n    {\n      id: 1,\n      icon: '▶️',\n      title: 'Điều khiển audio',\n      description: 'Điều khiển audio component qua các API'\n    },\n    {\n      id: 2,\n      icon: '🔊',\n      title: 'Quản lý âm lượng',\n      description: 'Điều chỉnh âm lượng và tắt tiếng với thanh trượt trực quan'\n    },\n    {\n      id: 3,\n      icon: '⏱️',\n      title: 'Hiển thị thời gian',\n      description: 'Hiển thị thời gian hiện tại và tổng thời gian theo định dạng HH:MM:SS'\n    },\n    {\n      id: 4,\n      icon: '📊',\n      title: 'Thanh tiến độ',\n      description: 'Thanh tiến độ có thể tương tác để tua nhanh hoặc tua lại'\n    },\n    {\n      id: 5,\n      icon: '📥',\n      title: 'Tải xuống',\n      description: 'Hỗ trợ tải xuống file âm thanh với kiểm soát quyền'\n    }\n  ]);\n\n  // Code examples data\n  public codeExamples = signal<Array<{ id: number, title: string, code: string }>>([\n    {\n      id: 1,\n      title: 'Cài đặt cơ bản',\n      code: `&lt;libs_ui-components-audio\n  [fileAudio]=\"'path/to/audio.mp3'\"\n  [checkPermissionDownloadAudio]=\"checkPermission\"&gt;\n&lt;/libs_ui-components-audio&gt;`\n    },\n    {\n      id: 2,\n      title: 'Sử dụng Function Control',\n      code: `import { Component, ViewChild, signal } from '@angular/core';\nimport { LibsUiComponentsAudioComponent } from '@libs-ui/components-audio';\nimport { IAudioFunctionControlEvent } from '@libs-ui/components-audio';\n\n@Component({\n  selector: 'app-my-component',\n  template: \\`\n    &lt;libs_ui-components-audio\n      #audioPlayer\n      [fileAudio]=\"audioSource()\"\n      [checkPermissionDownloadAudio]=\"checkPermission\"\n      (outFunctionsControl)=\"registerFunctions($event)\"&gt;\n    &lt;/libs_ui-components-audio&gt;\n    \n    &lt;button (click)=\"playAudio()\"&gt;Phát/Tạm dừng&lt;/button&gt;\n  \\`\n})\nexport class MyComponent {\n  @ViewChild('audioPlayer') audioPlayer!: LibsUiComponentsAudioComponent;\n  audioSource = signal<string>('path/to/audio.mp3');\n  functionControls: IAudioFunctionControlEvent | null = null;\n  \n  registerFunctions(event: IAudioFunctionControlEvent) {\n    this.functionControls = event;\n  }\n  \n  playAudio() {\n    if (this.functionControls) {\n      this.functionControls.playPause();\n    }\n  }\n}`\n    },\n    {\n      id: 3,\n      title: 'Sử dụng Events để cập nhật UI',\n      code: `import { Component, signal } from '@angular/core';\nimport { LibsUiComponentsAudioComponent } from '@libs-ui/components-audio';\n\n@Component({\n  selector: 'app-my-component',\n  template: \\`\n    &lt;libs_ui-components-audio\n      [fileAudio]=\"audioSource()\"\n      [checkPermissionDownloadAudio]=\"checkPermission\"\n      (outTimeUpdate)=\"handleTimeUpdate($event)\"\n      (outVolumeControl)=\"handleVolumeChange($event)\"\n      (outPlay)=\"handlePlayChange($event)\"\n      (outMute)=\"handleMuteChange($event)\"\n      (outEnded)=\"handleEnded()\"&gt;\n    &lt;/libs_ui-components-audio&gt;\n    \n    &lt;div class=\"audio-info\"&gt;\n      &lt;p&gt;Trạng thái: {{ isPlaying() ? 'Đang phát' : 'Tạm dừng' }}&lt;/p&gt;\n      &lt;p&gt;Thời gian hiện tại: {{ currentTime() }}&lt;/p&gt;\n      &lt;p&gt;Tổng thời gian: {{ duration() }}&lt;/p&gt;\n      &lt;p&gt;Âm lượng: {{ volumeLevel() }}%&lt;/p&gt;\n    &lt;/div&gt;\n  \\`\n})\nexport class MyComponent {\n  audioSource = signal<string>('path/to/audio.mp3');\n  isPlaying = signal<boolean>(false);\n  isMuted = signal<boolean>(false);\n  currentTime = signal<string>('00:00:00');\n  duration = signal<string>('00:00:00');\n  volumeLevel = signal<number>(100);\n  \n  checkPermission = (): Promise<boolean> => {\n    return Promise.resolve(true);\n  }\n  \n  handleTimeUpdate(timeInfo: { currentTime: string, duration: string }) {\n    this.currentTime.set(timeInfo.currentTime);\n    this.duration.set(timeInfo.duration);\n  }\n  \n  handleVolumeChange(volume: number) {\n    this.volumeLevel.set(volume);\n  }\n  \n  handlePlayChange(isPlaying: boolean) {\n    this.isPlaying.set(isPlaying);\n  }\n  \n  handleMuteChange(isMuted: boolean) {\n    this.isMuted.set(isMuted);\n  }\n  \n  handleEnded() {\n    this.isPlaying.set(false);\n  }\n}`\n    }\n  ]);\n\n  // Control methods\n  public playAudio(): void {\n    if (this.functionControls) {\n      this.functionControls.playPause();\n      this.isPlaying.set(this.functionControls.isPlaying());\n    }\n  }\n\n  public toggleMute(): void {\n    if (this.functionControls) {\n      this.functionControls.toggleMute();\n      this.isMuted.set(this.functionControls.isMuted());\n    }\n  }\n\n  public changeVolume(event: Event): void {\n    if (this.functionControls && event.target) {\n      const value = parseInt((event.target as HTMLInputElement).value);\n      this.volume.set(value);\n      this.functionControls.setVolume(value);\n      this.isMuted.set(this.functionControls.isMuted());\n    }\n  }\n\n  public downloadAudio(): void {\n    if (this.functionControls) {\n      this.functionControls.download();\n    }\n  }\n\n  /**\n   * Format thời gian từ giây sang chuỗi HH:MM:SS\n   */\n  public formatTime(timeInSeconds: number): string {\n    if (isNaN(timeInSeconds)) return '00:00:00';\n\n    const hours = Math.floor(timeInSeconds / 3600);\n    const minutes = Math.floor((timeInSeconds - (hours * 3600)) / 60);\n    const seconds = Math.floor(timeInSeconds - (hours * 3600) - (minutes * 60));\n\n    return `${hours.toString().padStart(2, '0')}:${minutes.toString().padStart(2, '0')}:${seconds.toString().padStart(2, '0')}`;\n  }\n\n  /**\n   * Sao chép text vào clipboard\n   */\n  public copyToClipboard(text: string): void {\n    navigator.clipboard.writeText(text)\n      .then(() => {\n        alert('Đã sao chép vào clipboard!');\n      })\n      .catch(err => {\n        console.error('Failed to copy: ', err);\n      });\n  }\n\n  /**\n   * Đăng ký các function control từ component\n   */\n  public registerFunctions(event: IAudioFunctionControlEvent): void {\n    this.functionControls = event;\n\n    // Initialize state\n    if (this.functionControls) {\n      this.isPlaying.set(this.functionControls.isPlaying());\n      this.isMuted.set(this.functionControls.isMuted());\n    }\n  }\n\n  /**\n   * Xử lý sự kiện thay đổi thời gian\n   */\n  public handleTimeUpdate(timeInfo: { currentTime: string, duration: string }): void {\n    this.currentTime.set(timeInfo.currentTime);\n    this.duration.set(timeInfo.duration);\n\n    // Calculate progress based on currentTime and duration\n    const regex = /(\\d+):(\\d+):(\\d+)/;\n    const currentMatches = timeInfo.currentTime.match(regex);\n    const durationMatches = timeInfo.duration.match(regex);\n\n    if (currentMatches && durationMatches) {\n      const currentSeconds =\n        parseInt(currentMatches[1]) * 3600 +\n        parseInt(currentMatches[2]) * 60 +\n        parseInt(currentMatches[3]);\n\n      const totalSeconds =\n        parseInt(durationMatches[1]) * 3600 +\n        parseInt(durationMatches[2]) * 60 +\n        parseInt(durationMatches[3]);\n\n      if (totalSeconds > 0) {\n        this.progress.set(Math.floor((currentSeconds / totalSeconds) * 100));\n      }\n    }\n  }\n\n  /**\n   * Xử lý sự kiện thay đổi âm lượng\n   */\n  public handleVolumeChange(volume: number): void {\n    this.volume.set(volume);\n  }\n\n  /**\n   * Xử lý sự kiện thay đổi trạng thái phát\n   */\n  public handlePlayChange(isPlaying: boolean): void {\n    this.isPlaying.set(isPlaying);\n  }\n\n  /**\n   * Xử lý sự kiện thay đổi trạng thái tắt tiếng\n   */\n  public handleMuteChange(isMuted: boolean): void {\n    this.isMuted.set(isMuted);\n  }\n\n  /**\n   * Xử lý sự kiện kết thúc phát\n   */\n  public handleEnded(): void {\n    this.isPlaying.set(false);\n  }\n\n  /**\n   * Chọn mẫu âm thanh\n   */\n  public selectAudio(id: number): void {\n    this.selectedAudio.set(id);\n    const sample = this.audioSamples().find(audio => audio.id === id);\n    if (sample) {\n      // Reset audio state\n      this.isPlaying.set(false);\n      this.progress.set(0);\n      this.currentTime.set('00:00:00');\n      this.duration.set('00:00:00');\n\n      // Simply update the source - the component will handle the reload\n      this.currentAudioSource.set(sample.src);\n    }\n  }\n\n  /**\n   * Kiểm tra quyền tải xuống\n   */\n  public checkDownloadPermission = (): Promise<boolean> => {\n    return Promise.resolve(true);\n  }\n\n  /**\n   * Thay đổi vị trí phát bằng cách click trực tiếp vào thanh tiến độ\n   */\n  public seekAudioByClick(event: MouseEvent): void {\n    if (this.functionControls) {\n      const progressBar = event.currentTarget as HTMLElement;\n      const rect = progressBar.getBoundingClientRect();\n      const offsetX = event.clientX - rect.left;\n      const percentX = (offsetX / rect.width) * 100;\n\n      // Đảm bảo giá trị nằm trong khoảng 0-100\n      const clampedPercent = Math.max(0, Math.min(100, percentX));\n      this.progress.set(Math.round(clampedPercent));\n\n      // Gọi method seekTo để cập nhật vị trí phát\n      this.functionControls.seekTo(this.progress());\n    }\n  }\n} \n","<div class=\"max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 font-sans text-gray-800\">\n  <header class=\"text-center py-10 bg-white rounded-lg shadow mb-8\">\n    <h1 class=\"text-4xl font-bold text-gray-800 mb-2\">Demo Trình Phát Âm Thanh</h1>\n    <p class=\"text-xl text-gray-600\">Thư viện component cho Angular để phát và điều khiển âm thanh</p>\n  </header>\n\n  <main>\n    <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n      <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">Giới thiệu</h2>\n      <p class=\"text-lg\">\n        <code class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">&#64;libs-ui/components-audio</code> là một\n        component Angular mạnh mẽ cho phép người dùng 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.\n      </p>\n    </section>\n\n    <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n      <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">Cài đặt</h2>\n      <p>Để cài đặt thư viện, sử dụng npm hoặc yarn:</p>\n\n      <div class=\"relative my-4\">\n        <pre\n          class=\"bg-gray-100 p-4 rounded overflow-x-auto mb-4\"><code class=\"font-mono\">npm install &#64;libs-ui/components-audio</code></pre>\n        <button\n          class=\"absolute top-2 right-2 bg-blue-500 text-white px-3 py-1 rounded text-sm hover:bg-blue-600 transition\"\n          (click)=\"copyToClipboard('npm install @libs-ui/components-audio')\">\n          Sao chép\n        </button>\n      </div>\n\n      <p>Hoặc với yarn:</p>\n\n      <div class=\"relative my-4\">\n        <pre\n          class=\"bg-gray-100 p-4 rounded overflow-x-auto mb-4\"><code class=\"font-mono\">yarn add &#64;libs-ui/components-audio</code></pre>\n        <button\n          class=\"absolute top-2 right-2 bg-blue-500 text-white px-3 py-1 rounded text-sm hover:bg-blue-600 transition\"\n          (click)=\"copyToClipboard('yarn add @libs-ui/components-audio')\">\n          Sao chép\n        </button>\n      </div>\n    </section>\n\n    <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n      <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">Demo trực tiếp</h2>\n      <div class=\"bg-gray-50 p-6 rounded-lg\">\n        <p class=\"mb-4\">Thử nghiệm trình phát âm thanh với các file mẫu:</p>\n\n        <div class=\"mb-6\">\n          <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Chọn file âm thanh:</h3>\n          <div class=\"flex flex-col gap-2\">\n            @for (audio of audioSamples(); track audio.id) {\n              <div\n                class=\"flex justify-between p-3 bg-white rounded border cursor-pointer transition hover:bg-blue-50 hover:border-blue-500\"\n                [class.bg-blue-50]=\"selectedAudio() === audio.id\"\n                [class.border-blue-500]=\"selectedAudio() === audio.id\"\n                [class.font-medium]=\"selectedAudio() === audio.id\"\n                (click)=\"selectAudio(audio.id)\">\n                <span>{{ audio.name }}</span>\n                <span>{{ audio.duration }}</span>\n              </div>\n            }\n          </div>\n        </div>\n\n        <div class=\"mb-6\">\n          <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Trình phát:</h3>\n          <libs_ui-components-audio #audioPlayer\n            [fileAudio]=\"currentAudioSource()\"\n            [checkPermissionDownloadAudio]=\"checkDownloadPermission\"\n            (outFunctionsControl)=\"registerFunctions($event)\"\n            (outTimeUpdate)=\"handleTimeUpdate($event)\"\n            (outVolumeControl)=\"handleVolumeChange($event)\"\n            (outPlay)=\"handlePlayChange($event)\"\n            (outMute)=\"handleMuteChange($event)\"\n            (outEnded)=\"handleEnded()\">\n          </libs_ui-components-audio>\n        </div>\n\n        <div class=\"mb-6\">\n          <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Điều khiển qua Function Control:</h3>\n          <div class=\"flex flex-wrap gap-3 items-center bg-white p-4 rounded border\">\n            <button class=\"px-4 py-2 bg-blue-500 text-white font-medium rounded hover:bg-blue-600 transition\"\n              (click)=\"playAudio()\">\n              Phát/Tạm dừng\n            </button>\n            <button class=\"px-4 py-2 bg-blue-500 text-white font-medium rounded hover:bg-blue-600 transition\"\n              (click)=\"toggleMute()\">\n              {{ isMuted() ? 'Bật tiếng' : 'Tắt tiếng' }}\n            </button>\n\n            <div class=\"flex items-center gap-2 ml-2\">\n              <span>Âm lượng:</span>\n              <input type=\"range\"\n                min=\"0\"\n                max=\"100\"\n                [value]=\"volumePercent()\"\n                (input)=\"changeVolume($event)\"\n                class=\"w-24 volume-slider\"\n                [style.--volume-percent.%]=\"volumePercent()\">\n              <span>{{ volumePercent() }}%</span>\n            </div>\n\n            <div class=\"w-full mt-2\">\n              <div class=\"flex items-center gap-2\">\n                <span>Tiến độ:</span>\n                <div class=\"relative h-2 bg-gray-200 rounded-full flex-1 cursor-pointer\"\n                  (click)=\"seekAudioByClick($event)\">\n                  <div class=\"absolute top-0 left-0 h-full bg-blue-500 rounded-full\"\n                    [style.width.%]=\"progress()\"></div>\n                </div>\n                <span>{{ progress() }}%</span>\n              </div>\n            </div>\n\n            <button class=\"px-4 py-2 bg-green-500 text-white font-medium rounded hover:bg-green-600 transition mt-4\"\n              (click)=\"downloadAudio()\">\n              Tải xuống\n            </button>\n          </div>\n\n          <div class=\"mt-6\">\n            <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Trạng thái:</h3>\n            <div class=\"bg-white p-4 rounded border\">\n              <div class=\"mb-2\">\n                <strong>Trạng thái:</strong>\n                <span [class.text-green-600]=\"isPlaying()\"\n                  [class.text-red-600]=\"!isPlaying()\">\n                  {{ isPlaying() ? 'Đang phát' : 'Tạm dừng' }}\n                </span>\n              </div>\n              <div class=\"mb-2\">\n                <strong>Thời gian hiện tại:</strong>\n                <span>{{ currentTime() }}</span>\n              </div>\n              <div class=\"mb-2\">\n                <strong>Tổng thời gian:</strong>\n                <span>{{ duration() }}</span>\n              </div>\n              <div class=\"mb-2\">\n                <strong>Tiến độ:</strong>\n                <span>{{ progress() }}%</span>\n              </div>\n            </div>\n          </div>\n        </div>\n      </div>\n    </section>\n\n    <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n      <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">Tài liệu API</h2>\n\n      <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Inputs</h3>\n      <div class=\"overflow-x-auto mb-8\">\n        <table class=\"min-w-full bg-white\">\n          <thead class=\"bg-gray-100\">\n            <tr>\n              <th class=\"py-3 px-4 text-left\">Tên</th>\n              <th class=\"py-3 px-4 text-left\">Kiểu dữ liệu</th>\n              <th class=\"py-3 px-4 text-left\">Mặc định</th>\n              <th class=\"py-3 px-4 text-left\">Mô tả</th>\n            </tr>\n          </thead>\n          <tbody>\n            @for (input of inputsDoc(); track input.name) {\n              <tr class=\"border-b hover:bg-gray-50\">\n                <td class=\"py-3 px-4\"><code\n                    class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ input.name }}</code></td>\n                <td class=\"py-3 px-4\"><code\n                    class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ input.type }}</code></td>\n                <td class=\"py-3 px-4\">{{ input.default }}</td>\n                <td class=\"py-3 px-4\">{{ input.description }}</td>\n              </tr>\n            }\n          </tbody>\n        </table>\n      </div>\n\n      <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Outputs</h3>\n      <div class=\"overflow-x-auto mb-8\">\n        <table class=\"min-w-full bg-white\">\n          <thead class=\"bg-gray-100\">\n            <tr>\n              <th class=\"py-3 px-4 text-left\">Tên</th>\n              <th class=\"py-3 px-4 text-left\">Kiểu dữ liệu</th>\n              <th class=\"py-3 px-4 text-left\">Mô tả</th>\n            </tr>\n          </thead>\n          <tbody>\n            @for (output of outputsDoc(); track output.name) {\n              <tr class=\"border-b hover:bg-gray-50\">\n                <td class=\"py-3 px-4\"><code\n                    class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ output.name }}</code></td>\n                <td class=\"py-3 px-4\"><code\n                    class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ output.type }}</code></td>\n                <td class=\"py-3 px-4\">{{ output.description }}</td>\n              </tr>\n            }\n          </tbody>\n        </table>\n      </div>\n\n      <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Function Control Methods</h3>\n      <div class=\"overflow-x-auto mb-8\">\n        <table class=\"min-w-full bg-white\">\n          <thead class=\"bg-gray-100\">\n            <tr>\n              <th class=\"py-3 px-4 text-left\">Tên</th>\n              <th class=\"py-3 px-4 text-left\">Tham số</th>\n              <th class=\"py-3 px-4 text-left\">Kiểu trả về</th>\n              <th class=\"py-3 px-4 text-left\">Mô tả</th>\n            </tr>\n          </thead>\n          <tbody>\n            @for (method of methodsDoc(); track method.name) {\n              <tr class=\"border-b hover:bg-gray-50\">\n                <td class=\"py-3 px-4\"><code\n                    class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ method.name }}</code></td>\n                <td class=\"py-3 px-4\"><code\n                    class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ method.params }}</code></td>\n                <td class=\"py-3 px-4\"><code\n                    class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ method.returnType }}</code></td>\n                <td class=\"py-3 px-4\">{{ method.description }}</td>\n              </tr>\n            }\n          </tbody>\n        </table>\n      </div>\n\n      <h3 class=\"text-xl font-semibold text-gray-700 mb-3\">Interfaces</h3>\n      <div class=\"overflow-x-auto mb-8\">\n        @for (interfaceDoc of interfacesDoc(); track interfaceDoc.name) {\n          <div class=\"mb-6 bg-white p-4 rounded border\">\n            <h4 class=\"text-lg font-semibold text-gray-700 mb-2\">{{ interfaceDoc.name }}</h4>\n            <p class=\"mb-4\">{{ interfaceDoc.description }}</p>\n\n            <table class=\"min-w-full bg-white\">\n              <thead class=\"bg-gray-100\">\n                <tr>\n                  <th class=\"py-3 px-4 text-left\">Tên</th>\n                  <th class=\"py-3 px-4 text-left\">Kiểu dữ liệu</th>\n                  <th class=\"py-3 px-4 text-left\">Mô tả</th>\n                </tr>\n              </thead>\n              <tbody>\n                @for (prop of interfaceDoc.properties; track prop.name) {\n                  <tr class=\"border-b hover:bg-gray-50\">\n                    <td class=\"py-3 px-4\"><code\n                        class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ prop.name }}</code></td>\n                    <td class=\"py-3 px-4\"><code\n                        class=\"bg-gray-100 px-1 py-0.5 rounded text-sm font-mono\">{{ prop.type }}</code></td>\n                    <td class=\"py-3 px-4\">{{ prop.description }}</td>\n                  </tr>\n                }\n              </tbody>\n            </table>\n          </div>\n        }\n      </div>\n    </section>\n\n    <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n      <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">Tính năng</h2>\n      <ul class=\"space-y-6\">\n        @for (feature of features(); track feature.id) {\n          <li class=\"flex items-start\">\n            <span class=\"text-3xl mr-4 min-w-10 text-center\">{{ feature.icon }}</span>\n            <div>\n              <h3 class=\"text-xl font-semibold text-gray-700\">{{ feature.title }}</h3>\n              <p>{{ feature.description }}</p>\n            </div>\n          </li>\n        }\n      </ul>\n    </section>\n\n    <section class=\"bg-white rounded-lg shadow p-8 mb-8\">\n      <h2 class=\"text-2xl font-bold text-gray-800 mb-4 pb-2 border-b border-gray-200\">Cách sử dụng</h2>\n      <div class=\"flex flex-col gap-4\">\n        @for (example of codeExamples(); track example.id) {\n          <div class=\"border border-gray-200 rounded-lg p-4\">\n            <h3 class=\"text-xl font-semibold text-gray-700 mb-2\">{{ example.title }}</h3>\n            <pre\n              class=\"bg-gray-100 p-4 rounded overflow-x-auto\"><code [innerHTML]=\"example.code\" class=\"font-mono\"></code></pre>\n          </div>\n        }\n      </div>\n    </section>\n  </main>\n</div>\n"]}
|