@ruc-lib/screen-recorder 2.9.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 ADDED
@@ -0,0 +1,134 @@
1
+ # ruclib-screen-recorder
2
+
3
+ This library provides a feature to record and download the screen with and without Audio
4
+
5
+ ## RUC Library Installation Guide
6
+
7
+ Users can easily install the RUC (Ruxptest Library) from npm. Additionally, they can also choose to install individual components based on their requirements.
8
+ Install RUC Library.
9
+
10
+ To install the entire RUC library:
11
+
12
+ `npm install @uxpractice/ruc-lib`
13
+
14
+ ## Install Individual Components
15
+ If users only need the dialog component, they can install it separately:
16
+
17
+ `npm install @ruclib/screen-recorder`
18
+
19
+ ## Usage
20
+
21
+ To use the screen recorder component in your project, follow the integration guidelines provided in the documentation.
22
+
23
+ ## Using Selector
24
+
25
+ 1. Import `RuclibScreenRecorderModule` from `@ruc-lib/screen-recorder` into `app.module.ts` file.
26
+ 2. Use the selector in your HTML file.
27
+
28
+ ```
29
+ <uxp-screen-recorder
30
+ [rucInputData]="recorderOptions"></uxp-screen-recorder>
31
+
32
+ ```
33
+
34
+
35
+ ## Input
36
+
37
+ Inputs:
38
+ • rucInputData: Data to be passed to the dialog.
39
+
40
+
41
+
42
+ ## rucInputData configurations options-
43
+
44
+ ```
45
+ ScreenRecorderOptions {
46
+ position?: ScreenRecorderPosition;
47
+ displayFormat?: DisplayFormat;
48
+ showAudioToggle?: boolean;
49
+ defaultAudioState?: boolean; // true for audio on, false for audio off
50
+ downloadFileName?: string;
51
+ icons?: ScreenRecorderIcons; // For custom icons (e.g., CSS classes for icon fonts)
52
+ showTimer?: boolean;
53
+ showPauseResume?: boolean;
54
+ showPlaybackControls?: boolean; // To show/hide the video player preview
55
+ showDownloadButton?: boolean;
56
+ useUTCForTimer?: boolean;
57
+ /* For specific area: getDisplayMedia allows user to choose window/tab/screen.
58
+ If a more specific crop is needed, it's outside standard Web API scope.
59
+ This option implies we ensure the browser's picker is used.
60
+ */
61
+ allowSpecificAreaSelection?: boolean;
62
+ // Accessibility related texts
63
+ accessibilityLabels?: {
64
+ recordButton?: string;
65
+ stopButton?: string;
66
+ pauseButton?: string;
67
+ resumeButton?: string;
68
+ audioToggleButton?: string;
69
+ downloadButton?: string;
70
+ playbackVideo?: string;
71
+ };
72
+ }
73
+
74
+ ScreenRecorderPosition {
75
+ Top = 'top',
76
+ Bottom = 'bottom',
77
+ Left = 'left',
78
+ Right = 'right',
79
+ }
80
+
81
+ DisplayFormat {
82
+ Icon = 'icon',
83
+ Text = 'text',
84
+ }
85
+ ```
86
+
87
+
88
+ ## Default value be like –
89
+
90
+ ```
91
+ rucInputData: {
92
+ position: 'top',
93
+ displayFormat: 'icon',
94
+ showAudioToggle: true,
95
+ defaultAudioState: true,
96
+ downloadFileName: 'screen-recording.webm',
97
+ showTimer: true,
98
+ showPauseResume: true,
99
+ showPlaybackControls: true,
100
+ showDownloadButton: true,
101
+ useUTCForTimer: false,
102
+ allowSpecificAreaSelection: true,
103
+ icons: {
104
+ record: 'fiber_manual_record',
105
+ stop: 'stop',
106
+ pause: 'pause',
107
+ resume: 'play_arrow',
108
+ play: 'play_circle_filled',
109
+ download: 'download',
110
+ audioOn: 'volume_up',
111
+ audioOff: 'volume_off',
112
+ },
113
+ accessibilityLabels: {
114
+ recordButton: 'Start Recording',
115
+ stopButton: 'Stop Recording',
116
+ pauseButton: 'Pause Recording',
117
+ resumeButton: 'Resume Recording',
118
+ audioToggleButton: 'Toggle Audio',
119
+ downloadButton: 'Download Recording',
120
+ playbackVideo: 'Screen Recording Playback',
121
+ },
122
+ },
123
+ ```
124
+
125
+
126
+
127
+ ## Contribution
128
+
129
+ Contributions are welcome! Feel free to open issues or pull requests for any enhancements or fixes.
130
+
131
+ ## Acknowledgements
132
+
133
+ Thank you for choosing the Screen Recorder Component Library. If you have any feedback or suggestions, please let us know!
134
+
@@ -0,0 +1,5 @@
1
+ export * from './lib/ruclib-screen-recorder.module';
2
+ export * from './lib/screen-recorder/screen-recorder.component';
3
+ export * from './lib/services/screen-recorder.service';
4
+ export * from './lib/models/screen-recorder.models';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9saWJzL3J1Y2xpYi9zY3JlZW4tcmVjb3JkZXIvc3JjL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMscUNBQXFDLENBQUM7QUFDcEQsY0FBYyxpREFBaUQsQ0FBQztBQUNoRSxjQUFjLHdDQUF3QyxDQUFDO0FBQ3ZELGNBQWMscUNBQXFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2xpYi9ydWNsaWItc2NyZWVuLXJlY29yZGVyLm1vZHVsZSc7XHJcbmV4cG9ydCAqIGZyb20gJy4vbGliL3NjcmVlbi1yZWNvcmRlci9zY3JlZW4tcmVjb3JkZXIuY29tcG9uZW50JztcclxuZXhwb3J0ICogZnJvbSAnLi9saWIvc2VydmljZXMvc2NyZWVuLXJlY29yZGVyLnNlcnZpY2UnO1xyXG5leHBvcnQgKiBmcm9tICcuL2xpYi9tb2RlbHMvc2NyZWVuLXJlY29yZGVyLm1vZGVscyc7Il19
@@ -0,0 +1,20 @@
1
+ export var ScreenRecorderPosition;
2
+ (function (ScreenRecorderPosition) {
3
+ ScreenRecorderPosition["Top"] = "top";
4
+ ScreenRecorderPosition["Bottom"] = "bottom";
5
+ ScreenRecorderPosition["Left"] = "left";
6
+ ScreenRecorderPosition["Right"] = "right";
7
+ })(ScreenRecorderPosition || (ScreenRecorderPosition = {}));
8
+ export var DisplayFormat;
9
+ (function (DisplayFormat) {
10
+ DisplayFormat["Icon"] = "icon";
11
+ DisplayFormat["Text"] = "text";
12
+ })(DisplayFormat || (DisplayFormat = {}));
13
+ export var RecordingState;
14
+ (function (RecordingState) {
15
+ RecordingState["Idle"] = "idle";
16
+ RecordingState["Recording"] = "recording";
17
+ RecordingState["Paused"] = "paused";
18
+ RecordingState["Stopped"] = "stopped";
19
+ })(RecordingState || (RecordingState = {}));
20
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2NyZWVuLXJlY29yZGVyLm1vZGVscy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvcnVjbGliL3NjcmVlbi1yZWNvcmRlci9zcmMvbGliL21vZGVscy9zY3JlZW4tcmVjb3JkZXIubW9kZWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sQ0FBTixJQUFZLHNCQUtUO0FBTEgsV0FBWSxzQkFBc0I7SUFDOUIscUNBQVcsQ0FBQTtJQUNYLDJDQUFpQixDQUFBO0lBQ2pCLHVDQUFhLENBQUE7SUFDYix5Q0FBZSxDQUFBO0FBQ2pCLENBQUMsRUFMUyxzQkFBc0IsS0FBdEIsc0JBQXNCLFFBSy9CO0FBRUQsTUFBTSxDQUFOLElBQVksYUFHWDtBQUhELFdBQVksYUFBYTtJQUN2Qiw4QkFBYSxDQUFBO0lBQ2IsOEJBQWEsQ0FBQTtBQUNmLENBQUMsRUFIVyxhQUFhLEtBQWIsYUFBYSxRQUd4QjtBQUVELE1BQU0sQ0FBTixJQUFZLGNBS1g7QUFMRCxXQUFZLGNBQWM7SUFDeEIsK0JBQWEsQ0FBQTtJQUNiLHlDQUF1QixDQUFBO0lBQ3ZCLG1DQUFpQixDQUFBO0lBQ2pCLHFDQUFtQixDQUFBO0FBQ3JCLENBQUMsRUFMVyxjQUFjLEtBQWQsY0FBYyxRQUt6QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBlbnVtIFNjcmVlblJlY29yZGVyUG9zaXRpb24ge1xyXG4gICAgVG9wID0gJ3RvcCcsXHJcbiAgICBCb3R0b20gPSAnYm90dG9tJyxcclxuICAgIExlZnQgPSAnbGVmdCcsXHJcbiAgICBSaWdodCA9ICdyaWdodCcsXHJcbiAgfVxyXG4gIFxyXG4gIGV4cG9ydCBlbnVtIERpc3BsYXlGb3JtYXQge1xyXG4gICAgSWNvbiA9ICdpY29uJyxcclxuICAgIFRleHQgPSAndGV4dCcsXHJcbiAgfVxyXG4gIFxyXG4gIGV4cG9ydCBlbnVtIFJlY29yZGluZ1N0YXRlIHtcclxuICAgIElkbGUgPSAnaWRsZScsXHJcbiAgICBSZWNvcmRpbmcgPSAncmVjb3JkaW5nJyxcclxuICAgIFBhdXNlZCA9ICdwYXVzZWQnLFxyXG4gICAgU3RvcHBlZCA9ICdzdG9wcGVkJyxcclxuICB9XHJcbiAgXHJcbiAgZXhwb3J0IGludGVyZmFjZSBTY3JlZW5SZWNvcmRlckljb25zIHtcclxuICAgIHJlY29yZD86IHN0cmluZzsgLy8gRGVmYXVsdDogJ+KXjycgKGZpbGxlZCBjaXJjbGUpIG9yIGEgbWF0ZXJpYWwgaWNvbiBjbGFzc1xyXG4gICAgcmVjb3JkV2l0aEF1ZGlvPzogc3RyaW5nOyAvLyBBbHRlcm5hdGl2ZSBpZiBzcGVjaWZpYyBpY29uIGZvciBcInJlY29yZCB3aXRoIGF1ZGlvXCIgaXMgbmVlZGVkXHJcbiAgICByZWNvcmRXaXRob3V0QXVkaW8/OiBzdHJpbmc7IC8vIEFsdGVybmF0aXZlIGlmIHNwZWNpZmljIGljb24gZm9yIFwicmVjb3JkIHdpdGhvdXQgYXVkaW9cIiBpcyBuZWVkZWRcclxuICAgIHN0b3A/OiBzdHJpbmc7IC8vIERlZmF1bHQ6ICfilqAnIChmaWxsZWQgc3F1YXJlKSBvciBhIG1hdGVyaWFsIGljb24gY2xhc3NcclxuICAgIHBhdXNlPzogc3RyaW5nOyAvLyBEZWZhdWx0OiAn4p2a4p2aJyAocGF1c2Ugc3ltYm9sKSBvciBhIG1hdGVyaWFsIGljb24gY2xhc3NcclxuICAgIHJlc3VtZT86IHN0cmluZzsgLy8gRGVmYXVsdDogJ+KWuicgKHBsYXkgc3ltYm9sKSBvciBhIG1hdGVyaWFsIGljb24gY2xhc3NcclxuICAgIHBsYXk/OiBzdHJpbmc7IC8vIERlZmF1bHQ6ICfilronIChwbGF5IHN5bWJvbCkgb3IgYSBtYXRlcmlhbCBpY29uIGNsYXNzXHJcbiAgICBkb3dubG9hZD86IHN0cmluZzsgLy8gRGVmYXVsdDogJ/Cfkr4nIChmbG9wcHkgZGlzaykgb3IgYSBtYXRlcmlhbCBpY29uIGNsYXNzXHJcbiAgICBhdWRpb09uPzogc3RyaW5nOyAvLyBEZWZhdWx0OiAn8J+UiicgKHNwZWFrZXIgaGlnaCB2b2x1bWUpIG9yIGEgbWF0ZXJpYWwgaWNvbiBjbGFzc1xyXG4gICAgYXVkaW9PZmY/OiBzdHJpbmc7IC8vIERlZmF1bHQ6ICfwn5SHJyAobXV0ZWQgc3BlYWtlcikgb3IgYSBtYXRlcmlhbCBpY29uIGNsYXNzXHJcbiAgfVxyXG4gIFxyXG4gIGV4cG9ydCBpbnRlcmZhY2UgU2NyZWVuUmVjb3JkZXJPcHRpb25zIHtcclxuICAgIHBvc2l0aW9uPzogU2NyZWVuUmVjb3JkZXJQb3NpdGlvbjtcclxuICAgIGRpc3BsYXlGb3JtYXQ/OiBEaXNwbGF5Rm9ybWF0O1xyXG4gICAgc2hvd0F1ZGlvVG9nZ2xlPzogYm9vbGVhbjtcclxuICAgIGRlZmF1bHRBdWRpb1N0YXRlPzogYm9vbGVhbjsgLy8gdHJ1ZSBmb3IgYXVkaW8gb24sIGZhbHNlIGZvciBhdWRpbyBvZmZcclxuICAgIGRvd25sb2FkRmlsZU5hbWU/OiBzdHJpbmc7XHJcbiAgICBpY29ucz86IFNjcmVlblJlY29yZGVySWNvbnM7IC8vIEZvciBjdXN0b20gaWNvbnMgKGUuZy4sIENTUyBjbGFzc2VzIGZvciBpY29uIGZvbnRzKVxyXG4gICAgc2hvd1RpbWVyPzogYm9vbGVhbjtcclxuICAgIHNob3dQYXVzZVJlc3VtZT86IGJvb2xlYW47XHJcbiAgICBzaG93UGxheWJhY2tDb250cm9scz86IGJvb2xlYW47IC8vIFRvIHNob3cvaGlkZSB0aGUgdmlkZW8gcGxheWVyIHByZXZpZXdcclxuICAgIHNob3dEb3dubG9hZEJ1dHRvbj86IGJvb2xlYW47XHJcbiAgICB1c2VVVENGb3JUaW1lcj86IGJvb2xlYW47XHJcbiAgICAvKiAgRm9yIHNwZWNpZmljIGFyZWE6IGdldERpc3BsYXlNZWRpYSBhbGxvd3MgdXNlciB0byBjaG9vc2Ugd2luZG93L3RhYi9zY3JlZW4uXHJcbiAgICAgIElmIGEgbW9yZSBzcGVjaWZpYyBjcm9wIGlzIG5lZWRlZCwgaXQncyBvdXRzaWRlIHN0YW5kYXJkIFdlYiBBUEkgc2NvcGUuXHJcbiAgICAgIFRoaXMgb3B0aW9uIGltcGxpZXMgd2UgZW5zdXJlIHRoZSBicm93c2VyJ3MgcGlja2VyIGlzIHVzZWQuXHJcbiAgICAqL1xyXG4gICAgYWxsb3dTcGVjaWZpY0FyZWFTZWxlY3Rpb24/OiBib29sZWFuO1xyXG4gICAgLy8gQWNjZXNzaWJpbGl0eSByZWxhdGVkIHRleHRzXHJcbiAgICBhY2Nlc3NpYmlsaXR5TGFiZWxzPzoge1xyXG4gICAgICByZWNvcmRCdXR0b24/OiBzdHJpbmc7XHJcbiAgICAgIHN0b3BCdXR0b24/OiBzdHJpbmc7XHJcbiAgICAgIHBhdXNlQnV0dG9uPzogc3RyaW5nO1xyXG4gICAgICByZXN1bWVCdXR0b24/OiBzdHJpbmc7XHJcbiAgICAgIGF1ZGlvVG9nZ2xlQnV0dG9uPzogc3RyaW5nO1xyXG4gICAgICBkb3dubG9hZEJ1dHRvbj86IHN0cmluZztcclxuICAgICAgcGxheWJhY2tWaWRlbz86IHN0cmluZztcclxuICAgIH07XHJcbiAgfVxyXG4gICJdfQ==
@@ -0,0 +1,20 @@
1
+ import { NgModule } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { ScreenRecorderComponent } from './screen-recorder/screen-recorder.component';
4
+ import { ScreenRecorderService } from './services/screen-recorder.service';
5
+ import * as i0 from "@angular/core";
6
+ export class RuclibScreenRecorderModule {
7
+ }
8
+ RuclibScreenRecorderModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RuclibScreenRecorderModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
9
+ RuclibScreenRecorderModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.2.10", ngImport: i0, type: RuclibScreenRecorderModule, declarations: [ScreenRecorderComponent], imports: [CommonModule], exports: [ScreenRecorderComponent] });
10
+ RuclibScreenRecorderModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RuclibScreenRecorderModule, providers: [ScreenRecorderService], imports: [CommonModule] });
11
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RuclibScreenRecorderModule, decorators: [{
12
+ type: NgModule,
13
+ args: [{
14
+ imports: [CommonModule],
15
+ declarations: [ScreenRecorderComponent],
16
+ exports: [ScreenRecorderComponent],
17
+ providers: [ScreenRecorderService],
18
+ }]
19
+ }] });
20
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicnVjbGliLXNjcmVlbi1yZWNvcmRlci5tb2R1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9saWJzL3J1Y2xpYi9zY3JlZW4tcmVjb3JkZXIvc3JjL2xpYi9ydWNsaWItc2NyZWVuLXJlY29yZGVyLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSw2Q0FBNkMsQ0FBQztBQUN0RixPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSxvQ0FBb0MsQ0FBQzs7QUFRM0UsTUFBTSxPQUFPLDBCQUEwQjs7d0hBQTFCLDBCQUEwQjt5SEFBMUIsMEJBQTBCLGlCQUp0Qix1QkFBdUIsYUFENUIsWUFBWSxhQUVaLHVCQUF1Qjt5SEFHdEIsMEJBQTBCLGFBRjFCLENBQUMscUJBQXFCLENBQUMsWUFIeEIsWUFBWTs0RkFLWCwwQkFBMEI7a0JBTnRDLFFBQVE7bUJBQUM7b0JBQ1IsT0FBTyxFQUFFLENBQUMsWUFBWSxDQUFDO29CQUN2QixZQUFZLEVBQUUsQ0FBQyx1QkFBdUIsQ0FBQztvQkFDdkMsT0FBTyxFQUFFLENBQUMsdUJBQXVCLENBQUM7b0JBQ2xDLFNBQVMsRUFBRSxDQUFDLHFCQUFxQixDQUFDO2lCQUNuQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5nTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XHJcbmltcG9ydCB7IFNjcmVlblJlY29yZGVyQ29tcG9uZW50IH0gZnJvbSAnLi9zY3JlZW4tcmVjb3JkZXIvc2NyZWVuLXJlY29yZGVyLmNvbXBvbmVudCc7XHJcbmltcG9ydCB7IFNjcmVlblJlY29yZGVyU2VydmljZSB9IGZyb20gJy4vc2VydmljZXMvc2NyZWVuLXJlY29yZGVyLnNlcnZpY2UnO1xyXG5cclxuQE5nTW9kdWxlKHtcclxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlXSxcclxuICBkZWNsYXJhdGlvbnM6IFtTY3JlZW5SZWNvcmRlckNvbXBvbmVudF0sXHJcbiAgZXhwb3J0czogW1NjcmVlblJlY29yZGVyQ29tcG9uZW50XSxcclxuICBwcm92aWRlcnM6IFtTY3JlZW5SZWNvcmRlclNlcnZpY2VdLFxyXG59KVxyXG5leHBvcnQgY2xhc3MgUnVjbGliU2NyZWVuUmVjb3JkZXJNb2R1bGUge31cclxuIl19
@@ -0,0 +1,44 @@
1
+ import { Injectable } from "@angular/core";
2
+ import * as i0 from "@angular/core";
3
+ export class ScreenRecorderConstants {
4
+ constructor() {
5
+ this.DEFAULT_ICONS = {
6
+ record: 'fiber_manual_record',
7
+ stop: 'stop',
8
+ pause: 'pause',
9
+ resume: 'play_arrow',
10
+ play: 'play_circle_filled',
11
+ download: 'download',
12
+ audioOn: 'volume_up',
13
+ audioOff: 'volume_off',
14
+ };
15
+ this.DEFAULT_LABELS = {
16
+ recordButton: 'Start Recording',
17
+ stopButton: 'Stop Recording',
18
+ pauseButton: 'Pause Recording',
19
+ resumeButton: 'Resume Recording',
20
+ audioToggleButton: 'Toggle Audio',
21
+ downloadButton: 'Download Recording',
22
+ playbackVideo: 'Screen Recording Playback',
23
+ };
24
+ this.INACTIVE = 'inactive';
25
+ this.AUDIO_ON = 'Audio ON';
26
+ this.AUDIO_OFF = 'Audio OFF';
27
+ this.RECORDING_IN_PROGRESS_ERROR = 'Recording is already in progress.';
28
+ this.FAILED_TO_GET_DISPLAY_MEDIA_STREAM_ERROR = 'Failed to get display media stream';
29
+ this.NO_SUPPORT_MIME = 'No supported MIME type found for MediaRecorder.';
30
+ this.NO_RECORDING_DOWNLOAD = 'No recording available to download.';
31
+ this.MEDIA_ERROR = 'MediaRecorder error';
32
+ this.PERMISSION_DENIED = "Error starting screen recording: NotAllowedError - Permission denied";
33
+ this.NO_SUPPORT = 'Screen recording is not supported in this browser.';
34
+ }
35
+ }
36
+ ScreenRecorderConstants.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: ScreenRecorderConstants, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
37
+ ScreenRecorderConstants.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: ScreenRecorderConstants, providedIn: 'root' });
38
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: ScreenRecorderConstants, decorators: [{
39
+ type: Injectable,
40
+ args: [{
41
+ providedIn: 'root',
42
+ }]
43
+ }], ctorParameters: function () { return []; } });
44
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2NyZWVuLXJlY29yZGVyLWNvbnN0YW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9ydWNsaWIvc2NyZWVuLXJlY29yZGVyL3NyYy9saWIvc2NyZWVuLXJlY29yZGVyL3NjcmVlbi1yZWNvcmRlci1jb25zdGFudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQVkzQyxNQUFNLE9BQU8sdUJBQXVCO0lBQ2hDO1FBR08sa0JBQWEsR0FBd0I7WUFDMUMsTUFBTSxFQUFFLHFCQUFxQjtZQUM3QixJQUFJLEVBQUUsTUFBTTtZQUNaLEtBQUssRUFBRSxPQUFPO1lBQ2QsTUFBTSxFQUFFLFlBQVk7WUFDcEIsSUFBSSxFQUFFLG9CQUFvQjtZQUMxQixRQUFRLEVBQUUsVUFBVTtZQUNwQixPQUFPLEVBQUUsV0FBVztZQUNwQixRQUFRLEVBQUUsWUFBWTtTQUN2QixDQUFDO1FBRUssbUJBQWMsR0FBRztZQUN0QixZQUFZLEVBQUUsaUJBQWlCO1lBQy9CLFVBQVUsRUFBRSxnQkFBZ0I7WUFDNUIsV0FBVyxFQUFFLGlCQUFpQjtZQUM5QixZQUFZLEVBQUUsa0JBQWtCO1lBQ2hDLGlCQUFpQixFQUFFLGNBQWM7WUFDakMsY0FBYyxFQUFFLG9CQUFvQjtZQUNwQyxhQUFhLEVBQUUsMkJBQTJCO1NBQzNDLENBQUM7UUFFSyxhQUFRLEdBQUcsVUFBVSxDQUFDO1FBRXRCLGFBQVEsR0FBRyxVQUFVLENBQUM7UUFDdEIsY0FBUyxHQUFHLFdBQVcsQ0FBQztRQUV4QixnQ0FBMkIsR0FBRyxtQ0FBbUMsQ0FBQztRQUNsRSw2Q0FBd0MsR0FBRyxvQ0FBb0MsQ0FBQztRQUNoRixvQkFBZSxHQUFHLGlEQUFpRCxDQUFDO1FBQ3BFLDBCQUFxQixHQUFHLHFDQUFxQyxDQUFDO1FBQzlELGdCQUFXLEdBQUcscUJBQXFCLENBQUM7UUFDcEMsc0JBQWlCLEdBQUcsc0VBQXNFLENBQUM7UUFDM0YsZUFBVSxHQUFHLG9EQUFvRCxDQUFDO0lBakN6RSxDQUFDOztxSEFIUSx1QkFBdUI7eUhBQXZCLHVCQUF1QixjQUZwQixNQUFNOzRGQUVULHVCQUF1QjtrQkFIbkMsVUFBVTttQkFBQztvQkFDUixVQUFVLEVBQUUsTUFBTTtpQkFDckIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlIH0gZnJvbSBcIkBhbmd1bGFyL2NvcmVcIjtcclxuaW1wb3J0IHtcclxuICBTY3JlZW5SZWNvcmRlck9wdGlvbnMsXHJcbiAgU2NyZWVuUmVjb3JkZXJQb3NpdGlvbixcclxuICBEaXNwbGF5Rm9ybWF0LFxyXG4gIFJlY29yZGluZ1N0YXRlLFxyXG4gIFNjcmVlblJlY29yZGVySWNvbnMsXHJcbn0gZnJvbSAnLi4vbW9kZWxzL3NjcmVlbi1yZWNvcmRlci5tb2RlbHMnO1xyXG5cclxuQEluamVjdGFibGUoe1xyXG4gICAgcHJvdmlkZWRJbjogJ3Jvb3QnLFxyXG59KVxyXG5leHBvcnQgY2xhc3MgU2NyZWVuUmVjb3JkZXJDb25zdGFudHN7XHJcbiAgICBjb25zdHJ1Y3Rvcigpe1xyXG5cclxuICAgIH1cclxuICAgIHB1YmxpYyBERUZBVUxUX0lDT05TOiBTY3JlZW5SZWNvcmRlckljb25zID0ge1xyXG4gICAgICByZWNvcmQ6ICdmaWJlcl9tYW51YWxfcmVjb3JkJywgLy8gTWF0ZXJpYWwgSWNvbiBuYW1lXHJcbiAgICAgIHN0b3A6ICdzdG9wJyxcclxuICAgICAgcGF1c2U6ICdwYXVzZScsXHJcbiAgICAgIHJlc3VtZTogJ3BsYXlfYXJyb3cnLFxyXG4gICAgICBwbGF5OiAncGxheV9jaXJjbGVfZmlsbGVkJyxcclxuICAgICAgZG93bmxvYWQ6ICdkb3dubG9hZCcsXHJcbiAgICAgIGF1ZGlvT246ICd2b2x1bWVfdXAnLFxyXG4gICAgICBhdWRpb09mZjogJ3ZvbHVtZV9vZmYnLFxyXG4gICAgfTtcclxuICAgIFxyXG4gICAgcHVibGljIERFRkFVTFRfTEFCRUxTID0ge1xyXG4gICAgICByZWNvcmRCdXR0b246ICdTdGFydCBSZWNvcmRpbmcnLFxyXG4gICAgICBzdG9wQnV0dG9uOiAnU3RvcCBSZWNvcmRpbmcnLFxyXG4gICAgICBwYXVzZUJ1dHRvbjogJ1BhdXNlIFJlY29yZGluZycsXHJcbiAgICAgIHJlc3VtZUJ1dHRvbjogJ1Jlc3VtZSBSZWNvcmRpbmcnLFxyXG4gICAgICBhdWRpb1RvZ2dsZUJ1dHRvbjogJ1RvZ2dsZSBBdWRpbycsXHJcbiAgICAgIGRvd25sb2FkQnV0dG9uOiAnRG93bmxvYWQgUmVjb3JkaW5nJyxcclxuICAgICAgcGxheWJhY2tWaWRlbzogJ1NjcmVlbiBSZWNvcmRpbmcgUGxheWJhY2snLFxyXG4gICAgfTtcclxuXHJcbiAgICBwdWJsaWMgSU5BQ1RJVkUgPSAnaW5hY3RpdmUnO1xyXG5cclxuICAgIHB1YmxpYyBBVURJT19PTiA9ICdBdWRpbyBPTic7XHJcbiAgICBwdWJsaWMgQVVESU9fT0ZGID0gJ0F1ZGlvIE9GRic7XHJcblxyXG4gICAgcHVibGljIFJFQ09SRElOR19JTl9QUk9HUkVTU19FUlJPUiA9ICdSZWNvcmRpbmcgaXMgYWxyZWFkeSBpbiBwcm9ncmVzcy4nO1xyXG4gICAgcHVibGljIEZBSUxFRF9UT19HRVRfRElTUExBWV9NRURJQV9TVFJFQU1fRVJST1IgPSAnRmFpbGVkIHRvIGdldCBkaXNwbGF5IG1lZGlhIHN0cmVhbSc7XHJcbiAgICBwdWJsaWMgTk9fU1VQUE9SVF9NSU1FID0gJ05vIHN1cHBvcnRlZCBNSU1FIHR5cGUgZm91bmQgZm9yIE1lZGlhUmVjb3JkZXIuJztcclxuICAgIHB1YmxpYyBOT19SRUNPUkRJTkdfRE9XTkxPQUQgPSAnTm8gcmVjb3JkaW5nIGF2YWlsYWJsZSB0byBkb3dubG9hZC4nO1xyXG4gICAgcHVibGljIE1FRElBX0VSUk9SID0gJ01lZGlhUmVjb3JkZXIgZXJyb3InO1xyXG4gICAgcHVibGljIFBFUk1JU1NJT05fREVOSUVEID0gXCJFcnJvciBzdGFydGluZyBzY3JlZW4gcmVjb3JkaW5nOiBOb3RBbGxvd2VkRXJyb3IgLSBQZXJtaXNzaW9uIGRlbmllZFwiO1xyXG4gICAgcHVibGljIE5PX1NVUFBPUlQgPSAnU2NyZWVuIHJlY29yZGluZyBpcyBub3Qgc3VwcG9ydGVkIGluIHRoaXMgYnJvd3Nlci4nO1xyXG5cclxufSJdfQ==
@@ -0,0 +1,242 @@
1
+ import { Component, Input, ChangeDetectionStrategy, ChangeDetectorRef, ViewChild, ElementRef } from '@angular/core';
2
+ import { DomSanitizer } from '@angular/platform-browser';
3
+ import { Subscription } from 'rxjs';
4
+ import { ScreenRecorderPosition, DisplayFormat, RecordingState, } from '../models/screen-recorder.models';
5
+ import { ScreenRecorderService } from '../services/screen-recorder.service';
6
+ import { ScreenRecorderConstants } from './screen-recorder-constant';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "../services/screen-recorder.service";
9
+ import * as i2 from "@angular/platform-browser";
10
+ import * as i3 from "./screen-recorder-constant";
11
+ import * as i4 from "@angular/common";
12
+ export class ScreenRecorderComponent {
13
+ set rucInputData(value) {
14
+ this._options = {
15
+ position: ScreenRecorderPosition.Bottom,
16
+ displayFormat: DisplayFormat.Icon,
17
+ showAudioToggle: true,
18
+ defaultAudioState: true,
19
+ downloadFileName: 'screen-recording.webm',
20
+ icons: { ...this.screenRecordingConstant.DEFAULT_ICONS },
21
+ showTimer: true,
22
+ showPauseResume: true,
23
+ showPlaybackControls: true,
24
+ showDownloadButton: true,
25
+ useUTCForTimer: false,
26
+ allowSpecificAreaSelection: true,
27
+ accessibilityLabels: { ...this.screenRecordingConstant.DEFAULT_LABELS },
28
+ ...value,
29
+ };
30
+ // Deep merge icons and labels
31
+ if (value.icons) {
32
+ this._options.icons = { ...this.screenRecordingConstant.DEFAULT_ICONS, ...value.icons };
33
+ }
34
+ if (value.accessibilityLabels) {
35
+ this._options.accessibilityLabels = { ...this.screenRecordingConstant.DEFAULT_LABELS, ...value.accessibilityLabels };
36
+ }
37
+ this.isAudioEnabled = this._options.defaultAudioState ?? true;
38
+ }
39
+ get rucInputData() {
40
+ return this._options;
41
+ }
42
+ /**
43
+ * Checks if recording is currently in progress
44
+ *
45
+ * @returns boolean - True if recording is in progress, false otherwise
46
+ */
47
+ isRecordingInProgress() {
48
+ return this.recordingState === RecordingState.Recording;
49
+ }
50
+ /**
51
+ * Gets the appropriate icon for recording based on audio state and settings
52
+ * @returns string - The icon name to display
53
+ */
54
+ getRecordingIcon() {
55
+ if (this.isAudioEnabled && this.rucInputData.showAudioToggle) {
56
+ return this.getIcon('recordWithAudio');
57
+ }
58
+ return this.getIcon('record');
59
+ }
60
+ constructor(screenRecorderService, sanitizer, cdr, screenRecordingConstant) {
61
+ this.screenRecorderService = screenRecorderService;
62
+ this.sanitizer = sanitizer;
63
+ this.cdr = cdr;
64
+ this.screenRecordingConstant = screenRecordingConstant;
65
+ this.recordingState = RecordingState.Idle;
66
+ this.recordedTime = 0;
67
+ this.formattedTime = '00:00:00';
68
+ this.safeRecordedUrl = null;
69
+ this.recordingTimestamp = '';
70
+ this.errorMessage = null;
71
+ this.isBrowserSupported = true;
72
+ this.isAudioEnabled = true;
73
+ // Expose enums to template
74
+ this.DisplayFormat = DisplayFormat;
75
+ this.RecordingState = RecordingState;
76
+ this.ScreenRecorderPosition = ScreenRecorderPosition;
77
+ this.subscriptions = new Subscription();
78
+ // Initialize with default options. The setter will override if an input is provided.
79
+ if (!this._options) {
80
+ this.rucInputData = {}; // Trigger setter with empty object to apply defaults
81
+ }
82
+ }
83
+ ngOnInit() {
84
+ this.subscriptions.add(this.screenRecorderService.recordingState$.subscribe((state) => {
85
+ this.recordingState = state;
86
+ if (state === RecordingState.Idle || state === RecordingState.Stopped) {
87
+ // Ensure audio toggle is reset to default if needed, or based on user action
88
+ }
89
+ this.cdr.detectChanges();
90
+ }));
91
+ this.subscriptions.add(this.screenRecorderService.recordingTimestamp$.subscribe((timestamp) => {
92
+ this.recordingTimestamp = timestamp;
93
+ this.cdr.detectChanges();
94
+ }));
95
+ this.subscriptions.add(this.screenRecorderService.recordedTime$.subscribe((time) => {
96
+ this.recordedTime = time;
97
+ this.formattedTime = this.formatTimeDisplay(time);
98
+ this.cdr.detectChanges();
99
+ }));
100
+ this.subscriptions.add(this.screenRecorderService.recordedUrl$.subscribe((url) => {
101
+ this.safeRecordedUrl = url ? this.sanitizer.bypassSecurityTrustUrl(url) : null;
102
+ this.cdr.detectChanges();
103
+ }));
104
+ this.subscriptions.add(this.screenRecorderService.error$.subscribe((error) => {
105
+ if (error != this.screenRecordingConstant.PERMISSION_DENIED) {
106
+ this.errorMessage = error;
107
+ }
108
+ this.cdr.detectChanges();
109
+ }));
110
+ }
111
+ ngOnDestroy() {
112
+ if (this.subscriptions) {
113
+ this.subscriptions.unsubscribe();
114
+ }
115
+ }
116
+ /**
117
+ * Toggles the recording state between start and stop
118
+ *
119
+ * @returns Promise<void>
120
+ */
121
+ async toggleRecord() {
122
+ this.errorMessage = null; // Clear previous errors
123
+ if (this.recordingState === RecordingState.Idle || this.recordingState === RecordingState.Stopped) {
124
+ // Set recording timestamp when starting recording
125
+ const now = new Date();
126
+ this.recordingTimestamp = now.toLocaleString('en-IN', {
127
+ year: 'numeric',
128
+ month: '2-digit',
129
+ day: '2-digit',
130
+ hour: '2-digit',
131
+ minute: '2-digit',
132
+ hour12: true
133
+ });
134
+ await this.screenRecorderService.startRecording({ audio: this.isAudioEnabled });
135
+ }
136
+ else {
137
+ this.screenRecorderService.stopRecording();
138
+ }
139
+ }
140
+ /**
141
+ * Toggles between pause and resume states during recording
142
+ *
143
+ * @returns void
144
+ */
145
+ togglePauseResume() {
146
+ this.errorMessage = null;
147
+ if (this.recordingState === RecordingState.Recording) {
148
+ this.screenRecorderService.pauseRecording();
149
+ }
150
+ else if (this.recordingState === RecordingState.Paused) {
151
+ this.screenRecorderService.resumeRecording();
152
+ }
153
+ }
154
+ /**
155
+ * Checks if recording can be resumed
156
+ *
157
+ * @returns boolean - True if recording can be resumed, false otherwise
158
+ */
159
+ canResume() {
160
+ return this.recordingState === RecordingState.Paused;
161
+ }
162
+ /**
163
+ * Downloads the recorded video file
164
+ *
165
+ * @returns void
166
+ */
167
+ onDownload() {
168
+ this.errorMessage = null;
169
+ this.screenRecorderService.downloadRecording(this.rucInputData.downloadFileName);
170
+ }
171
+ /**
172
+ * Checks if recording can be downloaded
173
+ *
174
+ * @returns boolean - True if recording can be downloaded, false otherwise
175
+ */
176
+ canDownload() {
177
+ return this.recordingState === RecordingState.Stopped;
178
+ }
179
+ /**
180
+ * Toggles audio recording on/off
181
+ *
182
+ * @returns void
183
+ */
184
+ toggleAudio() {
185
+ this.isAudioEnabled = !this.isAudioEnabled;
186
+ this.cdr.detectChanges();
187
+ }
188
+ /**
189
+ *
190
+ * @param iconName
191
+ * @returns
192
+ */
193
+ getIcon(iconName) {
194
+ return this.rucInputData.icons?.[iconName] || this.screenRecordingConstant.DEFAULT_ICONS[iconName];
195
+ }
196
+ /**
197
+ *
198
+ * @param labelName
199
+ * @returns
200
+ */
201
+ getLabel(labelName) {
202
+ return this.rucInputData.accessibilityLabels?.[labelName] || this.screenRecordingConstant.DEFAULT_LABELS[labelName];
203
+ }
204
+ /**
205
+ *
206
+ * @param time - The time in seconds to format
207
+ * @returns string - Formatted time string (HH:MM:SS)
208
+ */
209
+ formatTimeDisplay(time) {
210
+ const hours = Math.floor(time / 3600);
211
+ const minutes = Math.floor((time % 3600) / 60);
212
+ const seconds = time % 60;
213
+ const pad = (num) => num.toString().padStart(2, '0');
214
+ if (this.rucInputData && this.rucInputData.useUTCForTimer) {
215
+ // This is a simple duration format, not a specific UTC time.
216
+ // For actual UTC time of day, you'd use new Date().toUTCString() or similar.
217
+ // Assuming timer should show elapsed time formatted as HH:MM:SS.
218
+ return `${pad(hours)}:${pad(minutes)}:${pad(seconds)}`;
219
+ }
220
+ return `${pad(hours)}:${pad(minutes)}:${pad(seconds)}`;
221
+ }
222
+ // Method to reset the component and service to initial state
223
+ reset() {
224
+ this.screenRecorderService.resetToIdle();
225
+ this.isAudioEnabled = this.rucInputData.defaultAudioState ?? true;
226
+ this.cdr.detectChanges();
227
+ }
228
+ }
229
+ ScreenRecorderComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: ScreenRecorderComponent, deps: [{ token: i1.ScreenRecorderService }, { token: i2.DomSanitizer }, { token: i0.ChangeDetectorRef }, { token: i3.ScreenRecorderConstants }], target: i0.ɵɵFactoryTarget.Component });
230
+ ScreenRecorderComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: ScreenRecorderComponent, selector: "uxp-screen-recorder", inputs: { customTheme: "customTheme", rucInputData: "rucInputData" }, viewQueries: [{ propertyName: "videoPlayer", first: true, predicate: ["videoPlayer"], descendants: true }], ngImport: i0, template: "<div class=\"screen-recorder-controls\" [ngClass]=\"customTheme + ' position-' + (rucInputData.position || ScreenRecorderPosition.Bottom)\"\r\n *ngIf=\"isBrowserSupported\">\r\n\r\n <!-- Error Message -->\r\n <div *ngIf=\"errorMessage\" class=\"recorder-error\">\r\n {{ errorMessage }}\r\n </div>\r\n\r\n <!-- Record/Stop Button -->\r\n <button\r\n *ngIf=\"recordingState === RecordingState.Idle || recordingState === RecordingState.Stopped\" (click)=\"toggleRecord()\"\r\n [attr.aria-label]=\"getLabel('recordButton')\" [attr.aria-pressed]=\"false\" class=\"recorder-button record-button\">\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n <span class=\"material-icons\">{{ getRecordingIcon() || getIcon('record') }}</span>\r\n </ng-container>\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">{{ getLabel('recordButton')\r\n }}</ng-container>\r\n </button>\r\n\r\n <button\r\n *ngIf=\"recordingState === RecordingState.Recording || recordingState === RecordingState.Paused\"\r\n (click)=\"toggleRecord()\" [attr.aria-label]=\"getLabel('stopButton')\" [attr.aria-pressed]=\"true\"\r\n class=\"recorder-button stop-button\">\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n <span class=\"material-icons\">{{ getIcon('stop') }}</span>\r\n </ng-container>\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">{{ getLabel('stopButton') }}</ng-container>\r\n </button>\r\n\r\n <!-- Pause/Resume Button -->\r\n <button *ngIf=\"rucInputData.showPauseResume && recordingState === RecordingState.Recording\"\r\n (click)=\"togglePauseResume()\" [attr.aria-label]=\"getLabel('pauseButton')\" [attr.aria-pressed]=\"false\"\r\n class=\"recorder-button pause-button\">\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n <span class=\"material-icons\">{{ getIcon('pause') }}</span>\r\n </ng-container>\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">{{ getLabel('pauseButton') }}</ng-container>\r\n </button>\r\n <button *ngIf=\"rucInputData.showPauseResume && recordingState === RecordingState.Paused\"\r\n (click)=\"togglePauseResume()\" [attr.aria-label]=\"getLabel('resumeButton')\" [attr.aria-pressed]=\"true\"\r\n class=\"recorder-button resume-button\">\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n <span class=\"material-icons\">{{ getIcon('resume') }}</span>\r\n </ng-container>\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">{{ getLabel('resumeButton')\r\n }}</ng-container>\r\n </button>\r\n\r\n <!-- Audio Toggle Button -->\r\n <button\r\n *ngIf=\"rucInputData.showAudioToggle && (recordingState === RecordingState.Idle || recordingState === RecordingState.Stopped)\"\r\n (click)=\"toggleAudio()\" [attr.aria-label]=\"getLabel('audioToggleButton')\" [attr.aria-pressed]=\"isAudioEnabled\"\r\n class=\"recorder-button audio-toggle-button\">\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n <span class=\"material-icons\">{{ isAudioEnabled ? getIcon('audioOn') : getIcon('audioOff') }}</span>\r\n </ng-container>\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">\r\n {{ isAudioEnabled ? screenRecordingConstant.AUDIO_ON : screenRecordingConstant.AUDIO_OFF }}\r\n </ng-container>\r\n </button>\r\n\r\n <!-- Timer -->\r\n <div\r\n *ngIf=\"rucInputData.showTimer && (recordingState === RecordingState.Recording || recordingState === RecordingState.Paused)\"\r\n class=\"recorder-timer\" aria-live=\"polite\" aria-atomic=\"true\">\r\n {{ formattedTime }}\r\n </div>\r\n\r\n <!-- Download Button -->\r\n <button\r\n *ngIf=\"rucInputData.showDownloadButton && recordingState === RecordingState.Stopped && safeRecordedUrl\"\r\n (click)=\"onDownload()\" [attr.aria-label]=\"getLabel('downloadButton')\" class=\"recorder-button download-button\">\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n <span class=\"material-icons\">{{ getIcon('download') }}</span>\r\n </ng-container>\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">{{ getLabel('downloadButton')\r\n }}</ng-container>\r\n </button>\r\n\r\n <!-- Playback Area -->\r\n <div *ngIf=\"rucInputData.showPlaybackControls && recordingState === RecordingState.Stopped && safeRecordedUrl\"\r\n class=\"playback-area\">\r\n <video #videoPlayer [src]=\"safeRecordedUrl\" controls [attr.aria-label]=\"getLabel('playbackVideo')\"></video>\r\n <div class=\"video-timer-overlay\">\r\n <span>Recorded on: {{ recordingTimestamp }}</span>\r\n </div>\r\n </div>\r\n\r\n <!-- Reset Button (optional, good for testing/clearing state) -->\r\n <button *ngIf=\"recordingState === RecordingState.Stopped && safeRecordedUrl\" (click)=\"reset()\"\r\n aria-label=\"New Recording\" class=\"recorder-button reset-button\">\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n <span class=\"material-icons\">refresh</span>\r\n </ng-container>\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">New</ng-container>\r\n </button>\r\n\r\n</div>\r\n\r\n<div *ngIf=\"!isBrowserSupported\" class=\"recorder-notsupported\" [ngClass]=\"customTheme\">\r\n {{screenRecordingConstant.NO_SUPPORT}}\r\n</div>", styles: [":host{display:block;font-family:sans-serif}.screen-recorder-controls{display:flex;gap:8px;align-items:center;padding:8px;border-radius:4px;position:relative;z-index:1000}.screen-recorder-controls.position-top-left{margin-top:10px;margin-left:10px;flex-direction:row}.screen-recorder-controls.position-top-right{margin-top:10px;margin-right:10px;flex-direction:row}.screen-recorder-controls.position-bottom-left{margin-bottom:10px;margin-left:10px;flex-direction:row}.screen-recorder-controls.position-bottom-right{margin-bottom:10px;margin-right:10px;flex-direction:row}.screen-recorder-controls.position-left{margin-left:10px;flex-direction:column}.screen-recorder-controls.position-right{margin-right:10px;flex-direction:column}.recorder-button{padding:8px 12px;border:1px solid #ccc;border-radius:4px;background-color:#fff;cursor:pointer;display:inline-flex;align-items:center;gap:4px}.recorder-button:hover{background-color:#e9e9e9}.recorder-button:active{background-color:#d0d0d0}.recorder-button.record-button{color:red}.material-icons{font-family:Material Icons;font-weight:400;font-style:normal;font-size:20px;display:inline-block;line-height:1;text-transform:none;letter-spacing:normal;word-wrap:normal;white-space:nowrap;direction:ltr;-webkit-font-smoothing:antialiased;text-rendering:optimizeLegibility;-moz-osx-font-smoothing:grayscale;font-feature-settings:\"liga\"}.recorder-timer{padding:0 8px;font-feature-settings:\"tnum\";font-variant-numeric:tabular-nums}.playback-area{width:100%;max-width:100%;margin-top:1rem;position:relative}.playback-area video{width:100%;height:auto;max-width:100%}.playback-area .video-timer-overlay{position:absolute;top:10px;right:10px;background:rgba(0,0,0,.7);color:#fff;padding:5px 10px;border-radius:4px;font-size:14px;z-index:10}.recorder-error{color:red;padding:8px;border:1px solid red;background-color:#ffe0e0;border-radius:4px;margin-bottom:8px}.recorder-notsupported{padding:10px;background-color:#fff3cd;border:1px solid #ffeeba;color:#856404;border-radius:4px}\n"], dependencies: [{ kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
231
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: ScreenRecorderComponent, decorators: [{
232
+ type: Component,
233
+ args: [{ selector: 'uxp-screen-recorder', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"screen-recorder-controls\" [ngClass]=\"customTheme + ' position-' + (rucInputData.position || ScreenRecorderPosition.Bottom)\"\r\n *ngIf=\"isBrowserSupported\">\r\n\r\n <!-- Error Message -->\r\n <div *ngIf=\"errorMessage\" class=\"recorder-error\">\r\n {{ errorMessage }}\r\n </div>\r\n\r\n <!-- Record/Stop Button -->\r\n <button\r\n *ngIf=\"recordingState === RecordingState.Idle || recordingState === RecordingState.Stopped\" (click)=\"toggleRecord()\"\r\n [attr.aria-label]=\"getLabel('recordButton')\" [attr.aria-pressed]=\"false\" class=\"recorder-button record-button\">\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n <span class=\"material-icons\">{{ getRecordingIcon() || getIcon('record') }}</span>\r\n </ng-container>\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">{{ getLabel('recordButton')\r\n }}</ng-container>\r\n </button>\r\n\r\n <button\r\n *ngIf=\"recordingState === RecordingState.Recording || recordingState === RecordingState.Paused\"\r\n (click)=\"toggleRecord()\" [attr.aria-label]=\"getLabel('stopButton')\" [attr.aria-pressed]=\"true\"\r\n class=\"recorder-button stop-button\">\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n <span class=\"material-icons\">{{ getIcon('stop') }}</span>\r\n </ng-container>\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">{{ getLabel('stopButton') }}</ng-container>\r\n </button>\r\n\r\n <!-- Pause/Resume Button -->\r\n <button *ngIf=\"rucInputData.showPauseResume && recordingState === RecordingState.Recording\"\r\n (click)=\"togglePauseResume()\" [attr.aria-label]=\"getLabel('pauseButton')\" [attr.aria-pressed]=\"false\"\r\n class=\"recorder-button pause-button\">\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n <span class=\"material-icons\">{{ getIcon('pause') }}</span>\r\n </ng-container>\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">{{ getLabel('pauseButton') }}</ng-container>\r\n </button>\r\n <button *ngIf=\"rucInputData.showPauseResume && recordingState === RecordingState.Paused\"\r\n (click)=\"togglePauseResume()\" [attr.aria-label]=\"getLabel('resumeButton')\" [attr.aria-pressed]=\"true\"\r\n class=\"recorder-button resume-button\">\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n <span class=\"material-icons\">{{ getIcon('resume') }}</span>\r\n </ng-container>\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">{{ getLabel('resumeButton')\r\n }}</ng-container>\r\n </button>\r\n\r\n <!-- Audio Toggle Button -->\r\n <button\r\n *ngIf=\"rucInputData.showAudioToggle && (recordingState === RecordingState.Idle || recordingState === RecordingState.Stopped)\"\r\n (click)=\"toggleAudio()\" [attr.aria-label]=\"getLabel('audioToggleButton')\" [attr.aria-pressed]=\"isAudioEnabled\"\r\n class=\"recorder-button audio-toggle-button\">\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n <span class=\"material-icons\">{{ isAudioEnabled ? getIcon('audioOn') : getIcon('audioOff') }}</span>\r\n </ng-container>\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">\r\n {{ isAudioEnabled ? screenRecordingConstant.AUDIO_ON : screenRecordingConstant.AUDIO_OFF }}\r\n </ng-container>\r\n </button>\r\n\r\n <!-- Timer -->\r\n <div\r\n *ngIf=\"rucInputData.showTimer && (recordingState === RecordingState.Recording || recordingState === RecordingState.Paused)\"\r\n class=\"recorder-timer\" aria-live=\"polite\" aria-atomic=\"true\">\r\n {{ formattedTime }}\r\n </div>\r\n\r\n <!-- Download Button -->\r\n <button\r\n *ngIf=\"rucInputData.showDownloadButton && recordingState === RecordingState.Stopped && safeRecordedUrl\"\r\n (click)=\"onDownload()\" [attr.aria-label]=\"getLabel('downloadButton')\" class=\"recorder-button download-button\">\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n <span class=\"material-icons\">{{ getIcon('download') }}</span>\r\n </ng-container>\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">{{ getLabel('downloadButton')\r\n }}</ng-container>\r\n </button>\r\n\r\n <!-- Playback Area -->\r\n <div *ngIf=\"rucInputData.showPlaybackControls && recordingState === RecordingState.Stopped && safeRecordedUrl\"\r\n class=\"playback-area\">\r\n <video #videoPlayer [src]=\"safeRecordedUrl\" controls [attr.aria-label]=\"getLabel('playbackVideo')\"></video>\r\n <div class=\"video-timer-overlay\">\r\n <span>Recorded on: {{ recordingTimestamp }}</span>\r\n </div>\r\n </div>\r\n\r\n <!-- Reset Button (optional, good for testing/clearing state) -->\r\n <button *ngIf=\"recordingState === RecordingState.Stopped && safeRecordedUrl\" (click)=\"reset()\"\r\n aria-label=\"New Recording\" class=\"recorder-button reset-button\">\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n <span class=\"material-icons\">refresh</span>\r\n </ng-container>\r\n <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">New</ng-container>\r\n </button>\r\n\r\n</div>\r\n\r\n<div *ngIf=\"!isBrowserSupported\" class=\"recorder-notsupported\" [ngClass]=\"customTheme\">\r\n {{screenRecordingConstant.NO_SUPPORT}}\r\n</div>", styles: [":host{display:block;font-family:sans-serif}.screen-recorder-controls{display:flex;gap:8px;align-items:center;padding:8px;border-radius:4px;position:relative;z-index:1000}.screen-recorder-controls.position-top-left{margin-top:10px;margin-left:10px;flex-direction:row}.screen-recorder-controls.position-top-right{margin-top:10px;margin-right:10px;flex-direction:row}.screen-recorder-controls.position-bottom-left{margin-bottom:10px;margin-left:10px;flex-direction:row}.screen-recorder-controls.position-bottom-right{margin-bottom:10px;margin-right:10px;flex-direction:row}.screen-recorder-controls.position-left{margin-left:10px;flex-direction:column}.screen-recorder-controls.position-right{margin-right:10px;flex-direction:column}.recorder-button{padding:8px 12px;border:1px solid #ccc;border-radius:4px;background-color:#fff;cursor:pointer;display:inline-flex;align-items:center;gap:4px}.recorder-button:hover{background-color:#e9e9e9}.recorder-button:active{background-color:#d0d0d0}.recorder-button.record-button{color:red}.material-icons{font-family:Material Icons;font-weight:400;font-style:normal;font-size:20px;display:inline-block;line-height:1;text-transform:none;letter-spacing:normal;word-wrap:normal;white-space:nowrap;direction:ltr;-webkit-font-smoothing:antialiased;text-rendering:optimizeLegibility;-moz-osx-font-smoothing:grayscale;font-feature-settings:\"liga\"}.recorder-timer{padding:0 8px;font-feature-settings:\"tnum\";font-variant-numeric:tabular-nums}.playback-area{width:100%;max-width:100%;margin-top:1rem;position:relative}.playback-area video{width:100%;height:auto;max-width:100%}.playback-area .video-timer-overlay{position:absolute;top:10px;right:10px;background:rgba(0,0,0,.7);color:#fff;padding:5px 10px;border-radius:4px;font-size:14px;z-index:10}.recorder-error{color:red;padding:8px;border:1px solid red;background-color:#ffe0e0;border-radius:4px;margin-bottom:8px}.recorder-notsupported{padding:10px;background-color:#fff3cd;border:1px solid #ffeeba;color:#856404;border-radius:4px}\n"] }]
234
+ }], ctorParameters: function () { return [{ type: i1.ScreenRecorderService }, { type: i2.DomSanitizer }, { type: i0.ChangeDetectorRef }, { type: i3.ScreenRecorderConstants }]; }, propDecorators: { customTheme: [{
235
+ type: Input
236
+ }], rucInputData: [{
237
+ type: Input
238
+ }], videoPlayer: [{
239
+ type: ViewChild,
240
+ args: ['videoPlayer']
241
+ }] } });
242
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"screen-recorder.component.js","sourceRoot":"","sources":["../../../../../../../libs/ruclib/screen-recorder/src/lib/screen-recorder/screen-recorder.component.ts","../../../../../../../libs/ruclib/screen-recorder/src/lib/screen-recorder/screen-recorder.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnInit, OnDestroy, ChangeDetectionStrategy, ChangeDetectorRef, ViewChild, ElementRef } from '@angular/core';\r\nimport { DomSanitizer, SafeUrl } from '@angular/platform-browser';\r\nimport { Subscription } from 'rxjs';\r\nimport {\r\n  ScreenRecorderOptions,\r\n  ScreenRecorderPosition,\r\n  DisplayFormat,\r\n  RecordingState,\r\n  ScreenRecorderIcons,\r\n} from '../models/screen-recorder.models';\r\nimport { ScreenRecorderService } from '../services/screen-recorder.service';\r\nimport { ScreenRecorderConstants } from './screen-recorder-constant';\r\n\r\n\r\n\r\n@Component({\r\n  selector: 'uxp-screen-recorder',\r\n  templateUrl: './screen-recorder.component.html',\r\n  styleUrls: ['./screen-recorder.component.scss'],\r\n  changeDetection: ChangeDetectionStrategy.OnPush,\r\n})\r\nexport class ScreenRecorderComponent implements OnInit, OnDestroy {\r\n  @Input()\r\n  customTheme: any;\r\n  \r\n  @Input()\r\n  set rucInputData(value: ScreenRecorderOptions) {\r\n    this._options = {\r\n      position: ScreenRecorderPosition.Bottom,\r\n      displayFormat: DisplayFormat.Icon,\r\n      showAudioToggle: true,\r\n      defaultAudioState: true,\r\n      downloadFileName: 'screen-recording.webm',\r\n      icons: { ...this.screenRecordingConstant.DEFAULT_ICONS },\r\n      showTimer: true,\r\n      showPauseResume: true,\r\n      showPlaybackControls: true,\r\n      showDownloadButton: true,\r\n      useUTCForTimer: false,\r\n      allowSpecificAreaSelection: true, // Default to allowing browser's native picker\r\n      accessibilityLabels: { ...this.screenRecordingConstant.DEFAULT_LABELS },\r\n      ...value,\r\n    };\r\n    // Deep merge icons and labels\r\n    if (value.icons) {\r\n      this._options.icons = { ...this.screenRecordingConstant.DEFAULT_ICONS, ...value.icons };\r\n    }\r\n    if (value.accessibilityLabels) {\r\n      this._options.accessibilityLabels = { ...this.screenRecordingConstant.DEFAULT_LABELS, ...value.accessibilityLabels };\r\n    }\r\n    this.isAudioEnabled = this._options.defaultAudioState ?? true;\r\n  }\r\n  get rucInputData(): ScreenRecorderOptions {\r\n    return this._options;\r\n  }\r\n  private _options!: ScreenRecorderOptions;\r\n\r\n  @ViewChild('videoPlayer') videoPlayer!: ElementRef<HTMLVideoElement>;\r\n\r\n  recordingState: RecordingState = RecordingState.Idle;\r\n  recordedTime: number = 0;\r\n  formattedTime: string = '00:00:00';\r\n  safeRecordedUrl: SafeUrl | null = null;\r\n  recordingTimestamp: string = '';\r\n  /**\r\n   * Checks if recording is currently in progress\r\n   * \r\n   * @returns boolean - True if recording is in progress, false otherwise\r\n   */\r\n  isRecordingInProgress(): boolean {\r\n    return this.recordingState === RecordingState.Recording;\r\n  }\r\n  errorMessage: string | null = null;\r\n  isBrowserSupported: boolean = true;\r\n  isAudioEnabled: boolean = true;\r\n\r\n  /**\r\n   * Gets the appropriate icon for recording based on audio state and settings\r\n   * @returns string - The icon name to display\r\n   */\r\n  getRecordingIcon(): string {\r\n    if (this.isAudioEnabled && this.rucInputData.showAudioToggle) {\r\n      return this.getIcon('recordWithAudio');\r\n    }\r\n    return this.getIcon('record');\r\n  }\r\n\r\n  // Expose enums to template\r\n  DisplayFormat = DisplayFormat;\r\n  RecordingState = RecordingState;\r\n  ScreenRecorderPosition = ScreenRecorderPosition;\r\n\r\n  private subscriptions = new Subscription();\r\n\r\n  constructor(\r\n    public screenRecorderService: ScreenRecorderService,\r\n    private sanitizer: DomSanitizer,\r\n    private cdr: ChangeDetectorRef,\r\n    public screenRecordingConstant: ScreenRecorderConstants\r\n  ) {\r\n\r\n    // Initialize with default options. The setter will override if an input is provided.\r\n    if (!this._options) {\r\n        this.rucInputData = {}; // Trigger setter with empty object to apply defaults\r\n    }\r\n  }   \r\n\r\n  ngOnInit(): void {\r\n    this.subscriptions.add(\r\n      this.screenRecorderService.recordingState$.subscribe((state) => {\r\n        this.recordingState = state;\r\n        if (state === RecordingState.Idle || state === RecordingState.Stopped) {\r\n            // Ensure audio toggle is reset to default if needed, or based on user action\r\n        }\r\n        this.cdr.detectChanges();\r\n      })\r\n    );\r\n\r\n    this.subscriptions.add(\r\n      this.screenRecorderService.recordingTimestamp$.subscribe((timestamp) => {\r\n        this.recordingTimestamp = timestamp;\r\n        this.cdr.detectChanges();\r\n      })\r\n    );\r\n\r\n    this.subscriptions.add(\r\n      this.screenRecorderService.recordedTime$.subscribe((time) => {\r\n        this.recordedTime = time;\r\n        this.formattedTime = this.formatTimeDisplay(time);\r\n        this.cdr.detectChanges();\r\n      })\r\n    );\r\n\r\n    this.subscriptions.add(\r\n      this.screenRecorderService.recordedUrl$.subscribe((url) => {\r\n        this.safeRecordedUrl = url ? this.sanitizer.bypassSecurityTrustUrl(url) : null;\r\n        this.cdr.detectChanges();\r\n      })\r\n    );\r\n\r\n    this.subscriptions.add(\r\n      this.screenRecorderService.error$.subscribe((error) => {\r\n        if(error != this.screenRecordingConstant.PERMISSION_DENIED){\r\n          this.errorMessage = error;\r\n        }\r\n        this.cdr.detectChanges();\r\n      })\r\n    );\r\n  }\r\n\r\n  ngOnDestroy() {\r\n    if (this.subscriptions) {\r\n      this.subscriptions.unsubscribe();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Toggles the recording state between start and stop\r\n   * \r\n   * @returns Promise<void>\r\n   */\r\n  async toggleRecord(): Promise<void> {\r\n    this.errorMessage = null; // Clear previous errors\r\n    if (this.recordingState === RecordingState.Idle || this.recordingState === RecordingState.Stopped) {\r\n      // Set recording timestamp when starting recording\r\n      const now = new Date();\r\n      this.recordingTimestamp = now.toLocaleString('en-IN', {\r\n        year: 'numeric',\r\n        month: '2-digit',\r\n        day: '2-digit',\r\n        hour: '2-digit',\r\n        minute: '2-digit',\r\n        hour12: true\r\n      });\r\n      await this.screenRecorderService.startRecording({ audio: this.isAudioEnabled });\r\n    } else {\r\n      this.screenRecorderService.stopRecording();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Toggles between pause and resume states during recording\r\n   * \r\n   * @returns void\r\n   */\r\n  togglePauseResume(): void {\r\n    this.errorMessage = null;\r\n    if (this.recordingState === RecordingState.Recording) {\r\n      this.screenRecorderService.pauseRecording();\r\n    } else if (this.recordingState === RecordingState.Paused) {\r\n      this.screenRecorderService.resumeRecording();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Checks if recording can be resumed\r\n   * \r\n   * @returns boolean - True if recording can be resumed, false otherwise\r\n   */\r\n  canResume(): boolean {\r\n    return this.recordingState === RecordingState.Paused;\r\n  }\r\n\r\n  /**\r\n   * Downloads the recorded video file\r\n   * \r\n   * @returns void\r\n   */\r\n  onDownload(): void {\r\n    this.errorMessage = null;\r\n    this.screenRecorderService.downloadRecording(this.rucInputData.downloadFileName);\r\n  }\r\n\r\n  /**\r\n   * Checks if recording can be downloaded\r\n   * \r\n   * @returns boolean - True if recording can be downloaded, false otherwise\r\n   */\r\n  canDownload(): boolean {\r\n    return this.recordingState === RecordingState.Stopped;\r\n  }\r\n\r\n  /**\r\n   * Toggles audio recording on/off\r\n   * \r\n   * @returns void\r\n   */\r\n  toggleAudio(): void {\r\n    this.isAudioEnabled = !this.isAudioEnabled;\r\n    this.cdr.detectChanges();\r\n  }\r\n\r\n  /**\r\n   * \r\n   * @param iconName \r\n   * @returns \r\n   */\r\n  getIcon(iconName: keyof ScreenRecorderIcons): string {\r\n    return this.rucInputData.icons?.[iconName] || (this.screenRecordingConstant.DEFAULT_ICONS as any)[iconName];\r\n  }\r\n\r\n  /**\r\n   * \r\n   * @param labelName \r\n   * @returns \r\n   */\r\n  getLabel(labelName: keyof NonNullable<ScreenRecorderOptions['accessibilityLabels']>): string {\r\n    return this.rucInputData.accessibilityLabels?.[labelName] || (this.screenRecordingConstant.DEFAULT_LABELS as any)[labelName];\r\n  }\r\n\r\n  /**\r\n   * \r\n   * @param time - The time in seconds to format\r\n   * @returns string - Formatted time string (HH:MM:SS)\r\n   */\r\n  formatTimeDisplay(time: number): string {\r\n    const hours = Math.floor(time / 3600);\r\n    const minutes = Math.floor((time % 3600) / 60);\r\n    const seconds = time % 60;\r\n\r\n    const pad = (num: number) => num.toString().padStart(2, '0');\r\n\r\n    if (this.rucInputData && this.rucInputData.useUTCForTimer) {\r\n        // This is a simple duration format, not a specific UTC time.\r\n        // For actual UTC time of day, you'd use new Date().toUTCString() or similar.\r\n        // Assuming timer should show elapsed time formatted as HH:MM:SS.\r\n        return `${pad(hours)}:${pad(minutes)}:${pad(seconds)}`;\r\n    }\r\n    return `${pad(hours)}:${pad(minutes)}:${pad(seconds)}`;\r\n  }\r\n\r\n  // Method to reset the component and service to initial state\r\n  reset(): void {\r\n    this.screenRecorderService.resetToIdle();\r\n    this.isAudioEnabled = this.rucInputData.defaultAudioState ?? true;\r\n    this.cdr.detectChanges();\r\n  }\r\n}\r\n\r\n","<div class=\"screen-recorder-controls\" [ngClass]=\"customTheme + ' position-' + (rucInputData.position || ScreenRecorderPosition.Bottom)\"\r\n  *ngIf=\"isBrowserSupported\">\r\n\r\n  <!-- Error Message -->\r\n  <div *ngIf=\"errorMessage\" class=\"recorder-error\">\r\n    {{ errorMessage }}\r\n  </div>\r\n\r\n  <!-- Record/Stop Button -->\r\n  <button\r\n    *ngIf=\"recordingState === RecordingState.Idle || recordingState === RecordingState.Stopped\" (click)=\"toggleRecord()\"\r\n    [attr.aria-label]=\"getLabel('recordButton')\" [attr.aria-pressed]=\"false\" class=\"recorder-button record-button\">\r\n    <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n      <span class=\"material-icons\">{{ getRecordingIcon() || getIcon('record') }}</span>\r\n    </ng-container>\r\n    <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">{{ getLabel('recordButton')\r\n      }}</ng-container>\r\n  </button>\r\n\r\n  <button\r\n    *ngIf=\"recordingState === RecordingState.Recording || recordingState === RecordingState.Paused\"\r\n    (click)=\"toggleRecord()\" [attr.aria-label]=\"getLabel('stopButton')\" [attr.aria-pressed]=\"true\"\r\n    class=\"recorder-button stop-button\">\r\n    <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n      <span class=\"material-icons\">{{ getIcon('stop') }}</span>\r\n    </ng-container>\r\n    <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">{{ getLabel('stopButton') }}</ng-container>\r\n  </button>\r\n\r\n  <!-- Pause/Resume Button -->\r\n  <button *ngIf=\"rucInputData.showPauseResume && recordingState === RecordingState.Recording\"\r\n    (click)=\"togglePauseResume()\" [attr.aria-label]=\"getLabel('pauseButton')\" [attr.aria-pressed]=\"false\"\r\n    class=\"recorder-button pause-button\">\r\n    <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n      <span class=\"material-icons\">{{ getIcon('pause') }}</span>\r\n    </ng-container>\r\n    <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">{{ getLabel('pauseButton') }}</ng-container>\r\n  </button>\r\n  <button *ngIf=\"rucInputData.showPauseResume && recordingState === RecordingState.Paused\"\r\n    (click)=\"togglePauseResume()\" [attr.aria-label]=\"getLabel('resumeButton')\" [attr.aria-pressed]=\"true\"\r\n    class=\"recorder-button resume-button\">\r\n    <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n      <span class=\"material-icons\">{{ getIcon('resume') }}</span>\r\n    </ng-container>\r\n    <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">{{ getLabel('resumeButton')\r\n      }}</ng-container>\r\n  </button>\r\n\r\n  <!-- Audio Toggle Button -->\r\n  <button\r\n    *ngIf=\"rucInputData.showAudioToggle && (recordingState === RecordingState.Idle || recordingState === RecordingState.Stopped)\"\r\n    (click)=\"toggleAudio()\" [attr.aria-label]=\"getLabel('audioToggleButton')\" [attr.aria-pressed]=\"isAudioEnabled\"\r\n    class=\"recorder-button audio-toggle-button\">\r\n    <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n      <span class=\"material-icons\">{{ isAudioEnabled ? getIcon('audioOn') : getIcon('audioOff') }}</span>\r\n    </ng-container>\r\n    <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">\r\n      {{ isAudioEnabled ? screenRecordingConstant.AUDIO_ON : screenRecordingConstant.AUDIO_OFF }}\r\n    </ng-container>\r\n  </button>\r\n\r\n  <!-- Timer -->\r\n  <div\r\n    *ngIf=\"rucInputData.showTimer && (recordingState === RecordingState.Recording || recordingState === RecordingState.Paused)\"\r\n    class=\"recorder-timer\" aria-live=\"polite\" aria-atomic=\"true\">\r\n    {{ formattedTime }}\r\n  </div>\r\n\r\n  <!-- Download Button -->\r\n  <button\r\n    *ngIf=\"rucInputData.showDownloadButton && recordingState === RecordingState.Stopped && safeRecordedUrl\"\r\n    (click)=\"onDownload()\" [attr.aria-label]=\"getLabel('downloadButton')\" class=\"recorder-button download-button\">\r\n    <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n      <span class=\"material-icons\">{{ getIcon('download') }}</span>\r\n    </ng-container>\r\n    <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">{{ getLabel('downloadButton')\r\n      }}</ng-container>\r\n  </button>\r\n\r\n  <!-- Playback Area -->\r\n  <div *ngIf=\"rucInputData.showPlaybackControls && recordingState === RecordingState.Stopped && safeRecordedUrl\"\r\n    class=\"playback-area\">\r\n    <video #videoPlayer [src]=\"safeRecordedUrl\" controls [attr.aria-label]=\"getLabel('playbackVideo')\"></video>\r\n    <div class=\"video-timer-overlay\">\r\n      <span>Recorded on: {{ recordingTimestamp }}</span>\r\n    </div>\r\n  </div>\r\n\r\n  <!-- Reset Button (optional, good for testing/clearing state) -->\r\n  <button *ngIf=\"recordingState === RecordingState.Stopped && safeRecordedUrl\" (click)=\"reset()\"\r\n    aria-label=\"New Recording\" class=\"recorder-button reset-button\">\r\n    <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Icon\">\r\n      <span class=\"material-icons\">refresh</span>\r\n    </ng-container>\r\n    <ng-container *ngIf=\"rucInputData.displayFormat === DisplayFormat.Text\">New</ng-container>\r\n  </button>\r\n\r\n</div>\r\n\r\n<div *ngIf=\"!isBrowserSupported\" class=\"recorder-notsupported\" [ngClass]=\"customTheme\">\r\n  {{screenRecordingConstant.NO_SUPPORT}}\r\n</div>"]}