@aakash58/chatbot 1.1.15 → 1.1.17
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/esm2022/aakash58-chatbot.mjs +5 -0
- package/esm2022/lib/app/chat/chat-ui-state.service.mjs +170 -0
- package/esm2022/lib/app/chat/chat.service.mjs +445 -0
- package/esm2022/lib/app/chat/components/chat-button/chat-button.component.mjs +50 -0
- package/esm2022/lib/app/chat/components/chat-footer/chat-footer.component.mjs +12 -0
- package/esm2022/lib/app/chat/components/chat-header/chat-header.component.mjs +66 -0
- package/esm2022/lib/app/chat/components/chat-history-sidebar/chat-history-sidebar.component.mjs +186 -0
- package/esm2022/lib/app/chat/components/chat-window/chat-window.component.mjs +312 -0
- package/esm2022/lib/app/chat/components/message-input/message-input.component.mjs +36 -0
- package/esm2022/lib/app/chat/components/message-list/message-list.component.mjs +115 -0
- package/esm2022/lib/app/chat/model/chat-history.model.mjs +2 -0
- package/esm2022/lib/app/chat/model/chat-request.model.mjs +2 -0
- package/esm2022/lib/app/chat/model/chat-response.model.mjs +2 -0
- package/esm2022/lib/app/chat/model/chat-session.model.mjs +2 -0
- package/esm2022/lib/app/chat/model/chat-stream-message.model.mjs +2 -0
- package/esm2022/lib/app/chat/model/chat-stream-response.model.mjs +2 -0
- package/esm2022/lib/app/chat/services/chat-api.service.mjs +61 -0
- package/esm2022/lib/app/chat/services/chat-audio.service.mjs +50 -0
- package/esm2022/lib/app/chat/services/chat-history.service.mjs +252 -0
- package/esm2022/lib/app/login/login-form.component.mjs +46 -0
- package/esm2022/lib/app/personalization/personalization-dialog.component.mjs +194 -0
- package/esm2022/lib/app/personalization/personalization.service.mjs +149 -0
- package/esm2022/lib/app/personalization/sections/account/account-section.component.mjs +122 -0
- package/esm2022/lib/app/personalization/sections/preferences/color-picker-dialog.component.mjs +86 -0
- package/esm2022/lib/app/personalization/sections/preferences/preferences-section.component.mjs +115 -0
- package/esm2022/lib/app/personalization/sections/profile/profile-section.component.mjs +29 -0
- package/esm2022/lib/app/personalization/sections/settings/setting-section.component.mjs +30 -0
- package/esm2022/lib/app/personalization/sections/terms/terms-section.component.mjs +12 -0
- package/esm2022/lib/constant/doohbot-constant.mjs +28 -0
- package/esm2022/lib/constant/html-entities.mjs +9 -0
- package/esm2022/lib/constant/utf8.mjs +10 -0
- package/esm2022/lib/core/app-const.mjs +61 -0
- package/esm2022/lib/core/auth/account-api.service.mjs +40 -0
- package/esm2022/lib/core/auth/auth.service.mjs +391 -0
- package/esm2022/lib/core/auth/models/auth-result.model.mjs +3 -0
- package/esm2022/lib/core/auth/models/federated-login-request.model.mjs +6 -0
- package/esm2022/lib/core/auth/models/login-request.model.mjs +6 -0
- package/esm2022/lib/core/auth/storage.service.mjs +110 -0
- package/esm2022/lib/core/directives/draggable/draggable-dialog.directive.mjs +112 -0
- package/esm2022/lib/core/directives/fullscreen/fullscreen.directive.mjs +55 -0
- package/esm2022/lib/core/directives/resizable/resizable-dialog.directive.mjs +179 -0
- package/esm2022/lib/core/environments/environment.mjs +15 -0
- package/esm2022/lib/core/environments/environment.prod.mjs +15 -0
- package/esm2022/lib/core/helpers/crypto-helper.service.mjs +52 -0
- package/esm2022/lib/core/http/http-stream.service.mjs +97 -0
- package/esm2022/lib/core/http/http.service.mjs +103 -0
- package/esm2022/lib/core/interceptors/auth.interceptor.mjs +96 -0
- package/esm2022/lib/core/interceptors/license.interceptor.mjs +44 -0
- package/esm2022/lib/core/models/api-config.model.mjs +18 -0
- package/esm2022/lib/core/models/api-request.model.mjs +2 -0
- package/esm2022/lib/core/models/api-response.model.mjs +8 -0
- package/esm2022/lib/core/models/doohbot-config.model.mjs +18 -0
- package/esm2022/lib/core/models/license.model.mjs +2 -0
- package/esm2022/lib/core/models/message.mjs +2 -0
- package/esm2022/lib/core/models/theme-config.model.mjs +2 -0
- package/esm2022/lib/core/services/core-config.service.mjs +52 -0
- package/esm2022/lib/core/services/license.service.mjs +145 -0
- package/esm2022/lib/core/services/markdown.service.mjs +64 -0
- package/esm2022/lib/core/services/theme.service.mjs +248 -0
- package/esm2022/lib/core/types/auth-mode.type.mjs +2 -0
- package/esm2022/lib/core/types/auth-status.type.mjs +5 -0
- package/esm2022/lib/core/types/chat-stream.type.mjs +2 -0
- package/esm2022/lib/core/types/message-role.type.mjs +2 -0
- package/esm2022/lib/core/types/prompt-mode.type.mjs +5 -0
- package/esm2022/lib/core/types/snackbar-error.type.mjs +5 -0
- package/esm2022/lib/core/types/tenant-resolution-strategy.type.mjs +2 -0
- package/esm2022/lib/core/utils/logger.service.mjs +42 -0
- package/esm2022/lib/doohbot-input.mjs +20 -0
- package/esm2022/lib/doohbot.component.mjs +444 -0
- package/esm2022/lib/predefined_messages.mjs +15 -0
- package/esm2022/lib/shared/chips/chips.component.mjs +28 -0
- package/esm2022/lib/shared/dialog/dialog.component.mjs +36 -0
- package/esm2022/lib/shared/dialog/dialog.service.mjs +64 -0
- package/esm2022/lib/shared/dialog/dialog.utils.mjs +85 -0
- package/esm2022/lib/shared/dropdown-menu/dropdown-menu.component.mjs +29 -0
- package/esm2022/lib/shared/input-dialog/input-dialog.component.mjs +38 -0
- package/esm2022/lib/shared/menu-item/menu-item.component.mjs +24 -0
- package/esm2022/lib/shared/pipes/simple-markdown.pipe.mjs +27 -0
- package/esm2022/lib/shared/snackbar/snackbar.component.mjs +43 -0
- package/esm2022/lib/shared/snackbar/snackbar.service.mjs +46 -0
- package/esm2022/lib/shared/snackbar/snackbar.utils.mjs +43 -0
- package/esm2022/public-api.mjs +37 -0
- package/fesm2022/aakash58-chatbot.mjs +1114 -867
- package/fesm2022/aakash58-chatbot.mjs.map +1 -1
- package/index.d.ts +3 -377
- package/lib/app/chat/chat-ui-state.service.d.ts +96 -0
- package/lib/app/chat/chat.service.d.ts +88 -0
- package/lib/app/chat/components/chat-button/chat-button.component.d.ts +16 -0
- package/lib/app/chat/components/chat-footer/chat-footer.component.d.ts +5 -0
- package/lib/app/chat/components/chat-header/chat-header.component.d.ts +24 -0
- package/lib/app/chat/components/chat-history-sidebar/chat-history-sidebar.component.d.ts +49 -0
- package/lib/app/chat/components/chat-window/chat-window.component.d.ts +107 -0
- package/lib/app/chat/components/message-input/message-input.component.d.ts +12 -0
- package/lib/app/chat/components/message-list/message-list.component.d.ts +40 -0
- package/lib/app/chat/model/chat-history.model.d.ts +51 -0
- package/lib/app/chat/model/chat-request.model.d.ts +10 -0
- package/lib/app/chat/model/chat-response.model.d.ts +9 -0
- package/lib/app/chat/model/chat-session.model.d.ts +12 -0
- package/lib/app/chat/model/chat-stream-message.model.d.ts +5 -0
- package/lib/app/chat/model/chat-stream-response.model.d.ts +10 -0
- package/lib/app/chat/services/chat-api.service.d.ts +30 -0
- package/lib/app/chat/services/chat-audio.service.d.ts +19 -0
- package/lib/app/chat/services/chat-history.service.d.ts +53 -0
- package/lib/app/login/login-form.component.d.ts +20 -0
- package/lib/app/personalization/personalization-dialog.component.d.ts +53 -0
- package/lib/app/personalization/personalization.service.d.ts +66 -0
- package/lib/app/personalization/sections/account/account-section.component.d.ts +30 -0
- package/lib/app/personalization/sections/preferences/color-picker-dialog.component.d.ts +17 -0
- package/lib/app/personalization/sections/preferences/preferences-section.component.d.ts +27 -0
- package/lib/app/personalization/sections/profile/profile-section.component.d.ts +17 -0
- package/lib/app/personalization/sections/settings/setting-section.component.d.ts +10 -0
- package/lib/app/personalization/sections/terms/terms-section.component.d.ts +5 -0
- package/lib/constant/doohbot-constant.d.ts +12 -0
- package/lib/constant/html-entities.d.ts +8 -0
- package/lib/constant/utf8.d.ts +9 -0
- package/lib/core/app-const.d.ts +11 -0
- package/lib/core/auth/account-api.service.d.ts +20 -0
- package/lib/core/auth/auth.service.d.ts +90 -0
- package/lib/core/auth/models/auth-result.model.d.ts +4 -0
- package/lib/core/auth/models/federated-login-request.model.d.ts +5 -0
- package/lib/core/auth/models/login-request.model.d.ts +6 -0
- package/lib/core/auth/storage.service.d.ts +21 -0
- package/lib/core/directives/draggable/draggable-dialog.directive.d.ts +23 -0
- package/lib/core/directives/fullscreen/fullscreen.directive.d.ts +14 -0
- package/lib/core/directives/resizable/resizable-dialog.directive.d.ts +30 -0
- package/lib/core/environments/environment.d.ts +7 -0
- package/lib/core/environments/environment.prod.d.ts +7 -0
- package/lib/core/helpers/crypto-helper.service.d.ts +12 -0
- package/lib/core/http/http-stream.service.d.ts +18 -0
- package/lib/core/http/http.service.d.ts +20 -0
- package/lib/core/interceptors/auth.interceptor.d.ts +18 -0
- package/lib/core/interceptors/license.interceptor.d.ts +11 -0
- package/lib/core/models/api-config.model.d.ts +58 -0
- package/lib/core/models/api-request.model.d.ts +77 -0
- package/lib/core/models/api-response.model.d.ts +6 -0
- package/lib/core/models/doohbot-config.model.d.ts +81 -0
- package/lib/core/models/license.model.d.ts +23 -0
- package/lib/core/models/message.d.ts +16 -0
- package/lib/core/models/theme-config.model.d.ts +28 -0
- package/lib/core/services/core-config.service.d.ts +23 -0
- package/lib/core/services/license.service.d.ts +33 -0
- package/lib/core/services/markdown.service.d.ts +8 -0
- package/lib/core/services/theme.service.d.ts +40 -0
- package/lib/core/types/auth-mode.type.d.ts +4 -0
- package/lib/core/types/auth-status.type.d.ts +4 -0
- package/lib/core/types/chat-stream.type.d.ts +4 -0
- package/lib/core/types/message-role.type.d.ts +4 -0
- package/lib/core/types/prompt-mode.type.d.ts +4 -0
- package/lib/core/types/snackbar-error.type.d.ts +4 -0
- package/lib/core/types/tenant-resolution-strategy.type.d.ts +4 -0
- package/lib/core/utils/logger.service.d.ts +11 -0
- package/lib/doohbot-input.d.ts +19 -0
- package/lib/doohbot.component.d.ts +108 -0
- package/lib/predefined_messages.d.ts +2 -0
- package/lib/shared/chips/chips.component.d.ts +10 -0
- package/lib/shared/dialog/dialog.component.d.ts +19 -0
- package/lib/shared/dialog/dialog.service.d.ts +29 -0
- package/lib/shared/dialog/dialog.utils.d.ts +41 -0
- package/lib/shared/dropdown-menu/dropdown-menu.component.d.ts +11 -0
- package/lib/shared/input-dialog/input-dialog.component.d.ts +20 -0
- package/lib/shared/menu-item/menu-item.component.d.ts +9 -0
- package/lib/shared/pipes/simple-markdown.pipe.d.ts +10 -0
- package/lib/shared/snackbar/snackbar.component.d.ts +14 -0
- package/lib/shared/snackbar/snackbar.service.d.ts +19 -0
- package/lib/shared/snackbar/snackbar.utils.d.ts +33 -0
- package/package.json +4 -2
- package/public-api.d.ts +11 -0
|
@@ -0,0 +1,194 @@
|
|
|
1
|
+
import { Component, inject } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { FormsModule } from '@angular/forms';
|
|
4
|
+
import { MatDialogModule, MatDialogRef } from '@angular/material/dialog';
|
|
5
|
+
import { MatButtonModule } from '@angular/material/button';
|
|
6
|
+
import { MatIconModule } from '@angular/material/icon';
|
|
7
|
+
import { MatTabsModule } from '@angular/material/tabs';
|
|
8
|
+
import { MatSlideToggleModule } from '@angular/material/slide-toggle';
|
|
9
|
+
import { MatFormFieldModule } from '@angular/material/form-field';
|
|
10
|
+
import { MatInputModule } from '@angular/material/input';
|
|
11
|
+
import { MatSelectModule } from '@angular/material/select';
|
|
12
|
+
import { MatTooltipModule } from '@angular/material/tooltip';
|
|
13
|
+
import { PreferencesSectionComponent } from './sections/preferences/preferences-section.component';
|
|
14
|
+
import { ProfileSectionComponent } from './sections/profile/profile-section.component';
|
|
15
|
+
import { TermsSectionComponent } from './sections/terms/terms-section.component';
|
|
16
|
+
import { SettingSectionComponent } from './sections/settings/setting-section.component';
|
|
17
|
+
import { AccountSectionComponent } from './sections/account/account-section.component';
|
|
18
|
+
import { ChatHistoryService } from '../chat/services/chat-history.service';
|
|
19
|
+
import { AuthService } from '../../core/auth/auth.service';
|
|
20
|
+
import { PersonalizationService } from './personalization.service';
|
|
21
|
+
import { DialogService } from '../../shared/dialog/dialog.service';
|
|
22
|
+
import { DialogUtils } from '../../shared/dialog/dialog.utils';
|
|
23
|
+
import { SnackbarService } from '../../shared/snackbar/snackbar.service';
|
|
24
|
+
import { ThemeService } from '../../core/services/theme.service';
|
|
25
|
+
import * as i0 from "@angular/core";
|
|
26
|
+
import * as i1 from "@angular/material/dialog";
|
|
27
|
+
import * as i2 from "@angular/material/button";
|
|
28
|
+
import * as i3 from "@angular/material/icon";
|
|
29
|
+
import * as i4 from "@angular/material/tooltip";
|
|
30
|
+
export class PersonalizationDialogComponent {
|
|
31
|
+
constructor() {
|
|
32
|
+
this.dialogRef = inject((MatDialogRef));
|
|
33
|
+
this.dialogService = inject(DialogService);
|
|
34
|
+
this.personalization = inject(PersonalizationService);
|
|
35
|
+
this.authService = inject(AuthService);
|
|
36
|
+
this.chatHistory = inject(ChatHistoryService);
|
|
37
|
+
this.snackbar = inject(SnackbarService);
|
|
38
|
+
// Local copy of settings for editing
|
|
39
|
+
this.settings = JSON.parse(JSON.stringify(this.personalization.settings()));
|
|
40
|
+
this.activeSection = 'profile';
|
|
41
|
+
this.sections = [
|
|
42
|
+
{ id: 'account', label: 'Account', type: 'header' },
|
|
43
|
+
{ id: 'profile', label: 'My profile', icon: 'person_outline', type: 'item' },
|
|
44
|
+
{ id: 'account_settings', label: 'Manage Account', icon: 'manage_accounts', type: 'item' },
|
|
45
|
+
{ id: 'preferences', label: 'Preferences', icon: 'tune', type: 'item' },
|
|
46
|
+
{ id: 'workspace', label: 'Workspace', type: 'header' },
|
|
47
|
+
{ id: 'general', label: 'Settings', icon: 'settings', type: 'item' },
|
|
48
|
+
{ id: 'info', label: 'Support', type: 'header' },
|
|
49
|
+
{ id: 'terms', label: 'Terms & Conditions', icon: 'description', type: 'item' },
|
|
50
|
+
];
|
|
51
|
+
this.colorTokens = [
|
|
52
|
+
{
|
|
53
|
+
name: 'Standard',
|
|
54
|
+
value: 'default',
|
|
55
|
+
},
|
|
56
|
+
{ name: 'Purple', value: '#7c3aed' },
|
|
57
|
+
{ name: 'Pink', value: '#db2777' },
|
|
58
|
+
{ name: 'Red', value: '#dc2626' },
|
|
59
|
+
{ name: 'Orange', value: '#ea580c' },
|
|
60
|
+
{ name: 'Green', value: '#16a34a' },
|
|
61
|
+
{ name: 'Teal', value: '#0d9488' },
|
|
62
|
+
{ name: 'Blue', value: '#2563eb' },
|
|
63
|
+
];
|
|
64
|
+
this.themeService = inject(ThemeService);
|
|
65
|
+
this.originalColor = '';
|
|
66
|
+
this.isSaved = false;
|
|
67
|
+
this.originalColor = this.personalization.settings().appearance.primaryColor;
|
|
68
|
+
}
|
|
69
|
+
onSave() {
|
|
70
|
+
if (this.activeSection === 'preferences') {
|
|
71
|
+
this.isSaved = true;
|
|
72
|
+
this.personalization.updateAppearance(this.settings.appearance);
|
|
73
|
+
// Re-apply to ensure consistency (effect handles it, but just in case)
|
|
74
|
+
this.snackbar.success('Appearance settings saved');
|
|
75
|
+
}
|
|
76
|
+
else if (this.activeSection === 'general') {
|
|
77
|
+
this.personalization.updateInstructions(this.settings.customInstructions);
|
|
78
|
+
this.snackbar.success('Settings saved');
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
onCancel() {
|
|
82
|
+
this.dialogRef.close(false);
|
|
83
|
+
// ngOnDestroy will handle revert
|
|
84
|
+
}
|
|
85
|
+
ngOnDestroy() {
|
|
86
|
+
if (!this.isSaved) {
|
|
87
|
+
// Revert to original color if not saved
|
|
88
|
+
this.themeService.applyPrimaryColorToGlobal(this.originalColor);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
onReset() {
|
|
92
|
+
DialogUtils.confirmReset(this.dialogService).subscribe((confirmed) => {
|
|
93
|
+
if (confirmed) {
|
|
94
|
+
// Only reset the current section
|
|
95
|
+
if (this.activeSection === 'preferences') {
|
|
96
|
+
// Reset only appearance settings to defaults
|
|
97
|
+
this.personalization.resetToDefaults();
|
|
98
|
+
// Re-clone settings to update UI
|
|
99
|
+
this.settings = JSON.parse(JSON.stringify(this.personalization.settings()));
|
|
100
|
+
// Live preview the reset
|
|
101
|
+
this.themeService.applyPrimaryColorToGlobal(this.settings.appearance.primaryColor);
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
});
|
|
105
|
+
}
|
|
106
|
+
selectTheme(theme) {
|
|
107
|
+
this.settings.appearance.theme = theme;
|
|
108
|
+
// Optional: Preview theme? User only asked for color.
|
|
109
|
+
}
|
|
110
|
+
selectColor(color) {
|
|
111
|
+
this.settings.appearance.primaryColor = color;
|
|
112
|
+
//! Live Preview for changed color
|
|
113
|
+
// this.themeService.applyPrimaryColorToGlobal(color);
|
|
114
|
+
}
|
|
115
|
+
selectButtonStyle(style) {
|
|
116
|
+
this.settings.appearance.buttonStyle = style;
|
|
117
|
+
}
|
|
118
|
+
selectSection(sectionId) {
|
|
119
|
+
if (this.sections.find((s) => s.id === sectionId && s.type === 'item')) {
|
|
120
|
+
this.activeSection = sectionId;
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
get hasChanges() {
|
|
124
|
+
if (this.activeSection === 'preferences')
|
|
125
|
+
return this.hasPreferencesChanges();
|
|
126
|
+
if (this.activeSection === 'general')
|
|
127
|
+
return this.hasInstructionChanges();
|
|
128
|
+
return false;
|
|
129
|
+
}
|
|
130
|
+
hasPreferencesChanges() {
|
|
131
|
+
const original = this.personalization.settings().appearance;
|
|
132
|
+
const current = this.settings.appearance;
|
|
133
|
+
return JSON.stringify(original) !== JSON.stringify(current);
|
|
134
|
+
}
|
|
135
|
+
hasInstructionChanges() {
|
|
136
|
+
const original = this.personalization.settings().customInstructions;
|
|
137
|
+
const current = this.settings.customInstructions;
|
|
138
|
+
return JSON.stringify(original) !== JSON.stringify(current);
|
|
139
|
+
}
|
|
140
|
+
get username() {
|
|
141
|
+
return this.authService.getLoggedInUserName();
|
|
142
|
+
}
|
|
143
|
+
get userImageUrl() {
|
|
144
|
+
return this.authService.getLoggedInUserImageUrl();
|
|
145
|
+
}
|
|
146
|
+
get userRole() {
|
|
147
|
+
return this.authService.getLoggedInUserRole();
|
|
148
|
+
}
|
|
149
|
+
isCustomColorSelected() {
|
|
150
|
+
return !this.colorTokens.some((token) => token.value.toLowerCase() === this.settings.appearance.primaryColor.toLowerCase());
|
|
151
|
+
}
|
|
152
|
+
onCustomColorChange(event) {
|
|
153
|
+
const color = event.target.value;
|
|
154
|
+
this.selectColor(color);
|
|
155
|
+
}
|
|
156
|
+
deleteAllHistory() {
|
|
157
|
+
const userId = this.authService.getUserId() || '';
|
|
158
|
+
this.chatHistory.deleteAllHistory(userId.toString()).subscribe(() => {
|
|
159
|
+
// History cleared
|
|
160
|
+
});
|
|
161
|
+
}
|
|
162
|
+
logoutAllDevices() {
|
|
163
|
+
console.log('Logging out of all devices...');
|
|
164
|
+
// Implement service call when API is ready
|
|
165
|
+
}
|
|
166
|
+
deleteAccount() {
|
|
167
|
+
console.log('Deleting account...');
|
|
168
|
+
// Implement service call when API is ready
|
|
169
|
+
}
|
|
170
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PersonalizationDialogComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
171
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: PersonalizationDialogComponent, isStandalone: true, selector: "app-personalization-dialog", ngImport: i0, template: "<div class=\"personalization-container\">\r\n <div class=\"header\">\r\n <div class=\"title-section\">\r\n <mat-icon class=\"header-icon\">settings_suggest</mat-icon>\r\n <h2>Personalization</h2>\r\n </div>\r\n <button mat-icon-button (click)=\"onCancel()\" matTooltip=\"Close\">\r\n <mat-icon>close</mat-icon>\r\n </button>\r\n </div>\r\n\r\n <mat-dialog-content class=\"dialog-layout no-padding\">\r\n <!-- Sidebar Navigation -->\r\n <aside class=\"sidebar\">\r\n @for (section of sections; track section.id) {\r\n @if (section.type === 'header') {\r\n <div class=\"nav-header\">{{ section.label }}</div>\r\n } @else {\r\n <div\r\n class=\"nav-item\"\r\n [class.active]=\"activeSection === section.id\"\r\n (click)=\"selectSection(section.id)\"\r\n >\r\n <mat-icon>{{ section.icon }}</mat-icon>\r\n <span>{{ section.label }}</span>\r\n </div>\r\n }\r\n }\r\n </aside>\r\n\r\n <!-- Main Content Area -->\r\n <main class=\"main-content\">\r\n <!-- Appearance / Preferences Section -->\r\n @if (activeSection === 'preferences') {\r\n <app-preferences-section\r\n [settings]=\"settings\"\r\n [colorTokens]=\"colorTokens\"\r\n [isCustomColorSelected]=\"isCustomColorSelected()\"\r\n [defaultColor]=\"personalization.developerDefaults().primaryColor || 'default'\"\r\n (onThemeSelect)=\"selectTheme($event)\"\r\n (onColorSelect)=\"selectColor($event)\"\r\n (onButtonStyleSelect)=\"selectButtonStyle($event)\"\r\n (onSave)=\"onSave()\"\r\n (onReset)=\"onReset()\"\r\n >\r\n </app-preferences-section>\r\n }\r\n\r\n <!-- Account / Profile Section -->\r\n @if (activeSection === 'account_settings') {\r\n <app-account-section\r\n (onDeleteAllHistory)=\"deleteAllHistory()\"\r\n (onLogoutAll)=\"logoutAllDevices()\"\r\n (onDeleteAccount)=\"deleteAccount()\"\r\n ></app-account-section>\r\n }\r\n\r\n @if (activeSection === 'profile') {\r\n <app-profile-section\r\n [username]=\"username\"\r\n [userImageUrl]=\"userImageUrl\"\r\n [userRole]=\"userRole\"\r\n ></app-profile-section>\r\n }\r\n\r\n <!-- General Section -->\r\n @if (activeSection === 'general') {\r\n <app-setting-section\r\n [settings]=\"settings\"\r\n [hasChanges]=\"hasChanges\"\r\n (onSave)=\"onSave()\"\r\n ></app-setting-section>\r\n }\r\n\r\n <!-- Terms and Conditions Section -->\r\n @if (activeSection === 'terms') {\r\n <app-terms-section></app-terms-section>\r\n }\r\n </main>\r\n </mat-dialog-content>\r\n</div>\r\n", styles: [".personalization-container{display:flex;flex-direction:column;width:100%;height:100%;max-width:none;max-height:none;max-height:90vh;background:var(--background-color);color:var(--text-color);border-radius:12px;overflow:hidden;box-shadow:var(--border-shadow-color)}.personalization-container .header{flex:0 0 auto;display:flex;justify-content:space-between;align-items:center;padding:12px 20px;border-bottom:1px solid var(--border-color);background:var(--background-color)}.personalization-container .header .title-section{display:flex;align-items:center;gap:10px}.personalization-container .header .title-section .header-icon{color:var(--secondary-text-color);font-size:20px;width:20px;height:20px}.personalization-container .header .title-section h2{margin:0;font-size:.95rem;font-weight:500;color:var(--secondary-text-color)}.personalization-container .dialog-layout{display:flex;padding:0;margin:0;flex:1;overflow:hidden}.personalization-container .sidebar{width:240px;background:rgba(var(--black-rgb),.02);border-right:1px solid var(--border-color);padding:16px 8px;display:flex;flex-direction:column;gap:2px;overflow-y:auto;scrollbar-gutter:stable}.personalization-container .sidebar::-webkit-scrollbar{width:4px}.personalization-container .sidebar::-webkit-scrollbar-track{background:transparent}.personalization-container .sidebar::-webkit-scrollbar-thumb{background:rgba(var(--black-rgb),.1);border-radius:10px}.personalization-container .sidebar::-webkit-scrollbar-thumb:hover{background:rgba(var(--black-rgb),.2)}.personalization-container .sidebar .nav-header{padding:12px 12px 6px;font-size:.75rem;font-weight:600;text-transform:uppercase;color:var(--secondary-text-color);letter-spacing:.5px}.personalization-container .sidebar .nav-item{display:flex;align-items:center;gap:12px;padding:8px 12px;border-radius:6px;cursor:pointer;transition:all .2s ease;color:var(--text-color)}.personalization-container .sidebar .nav-item mat-icon{font-size:18px;width:18px;height:18px;color:var(--secondary-text-color)}.personalization-container .sidebar .nav-item span{font-size:.9rem;font-weight:400}.personalization-container .sidebar .nav-item:hover{background:rgba(var(--black-rgb),.05)}.personalization-container .sidebar .nav-item.active{background:rgba(var(--primary-color-rgb),.1);font-weight:500;color:var(--primary-color)}.personalization-container .sidebar .nav-item.active mat-icon{color:var(--primary-color)}.personalization-container .main-content{flex:1;overflow-y:auto;background:var(--background-color);position:relative}.personalization-container .main-content::-webkit-scrollbar{width:6px}.personalization-container .main-content::-webkit-scrollbar-track{background:transparent}.personalization-container .main-content::-webkit-scrollbar-thumb{background:rgba(var(--black-rgb),.1);border-radius:10px}.personalization-container .main-content::-webkit-scrollbar-thumb:hover{background:rgba(var(--black-rgb),.2)}.full-width{width:100%}@keyframes slideIn{0%{opacity:0;transform:translateY(4px)}to{opacity:1;transform:translateY(0)}}:host-context(.dark-theme) .personalization-container .sidebar{background:var(--background-color)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: FormsModule }, { kind: "ngmodule", type: MatDialogModule }, { kind: "directive", type: i1.MatDialogContent, selector: "[mat-dialog-content], mat-dialog-content, [matDialogContent]" }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i2.MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatTabsModule }, { kind: "ngmodule", type: MatSlideToggleModule }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: MatInputModule }, { kind: "ngmodule", type: MatSelectModule }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i4.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "component", type: PreferencesSectionComponent, selector: "app-preferences-section", inputs: ["settings", "colorTokens", "isCustomColorSelected", "defaultColor"], outputs: ["onThemeSelect", "onColorSelect", "onButtonStyleSelect", "onSave", "onReset"] }, { kind: "component", type: ProfileSectionComponent, selector: "app-profile-section", inputs: ["username", "userImageUrl", "userRole"] }, { kind: "component", type: SettingSectionComponent, selector: "app-setting-section", inputs: ["settings", "hasChanges"], outputs: ["onSave"] }, { kind: "component", type: TermsSectionComponent, selector: "app-terms-section" }, { kind: "component", type: AccountSectionComponent, selector: "app-account-section", inputs: ["organizationId"], outputs: ["onDeleteAllHistory", "onLogoutAll", "onLogoutSession", "onDeleteAccount"] }] }); }
|
|
172
|
+
}
|
|
173
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PersonalizationDialogComponent, decorators: [{
|
|
174
|
+
type: Component,
|
|
175
|
+
args: [{ selector: 'app-personalization-dialog', standalone: true, imports: [
|
|
176
|
+
CommonModule,
|
|
177
|
+
FormsModule,
|
|
178
|
+
MatDialogModule,
|
|
179
|
+
MatButtonModule,
|
|
180
|
+
MatIconModule,
|
|
181
|
+
MatTabsModule,
|
|
182
|
+
MatSlideToggleModule,
|
|
183
|
+
MatFormFieldModule,
|
|
184
|
+
MatInputModule,
|
|
185
|
+
MatSelectModule,
|
|
186
|
+
MatTooltipModule,
|
|
187
|
+
PreferencesSectionComponent,
|
|
188
|
+
ProfileSectionComponent,
|
|
189
|
+
SettingSectionComponent,
|
|
190
|
+
TermsSectionComponent,
|
|
191
|
+
AccountSectionComponent,
|
|
192
|
+
], template: "<div class=\"personalization-container\">\r\n <div class=\"header\">\r\n <div class=\"title-section\">\r\n <mat-icon class=\"header-icon\">settings_suggest</mat-icon>\r\n <h2>Personalization</h2>\r\n </div>\r\n <button mat-icon-button (click)=\"onCancel()\" matTooltip=\"Close\">\r\n <mat-icon>close</mat-icon>\r\n </button>\r\n </div>\r\n\r\n <mat-dialog-content class=\"dialog-layout no-padding\">\r\n <!-- Sidebar Navigation -->\r\n <aside class=\"sidebar\">\r\n @for (section of sections; track section.id) {\r\n @if (section.type === 'header') {\r\n <div class=\"nav-header\">{{ section.label }}</div>\r\n } @else {\r\n <div\r\n class=\"nav-item\"\r\n [class.active]=\"activeSection === section.id\"\r\n (click)=\"selectSection(section.id)\"\r\n >\r\n <mat-icon>{{ section.icon }}</mat-icon>\r\n <span>{{ section.label }}</span>\r\n </div>\r\n }\r\n }\r\n </aside>\r\n\r\n <!-- Main Content Area -->\r\n <main class=\"main-content\">\r\n <!-- Appearance / Preferences Section -->\r\n @if (activeSection === 'preferences') {\r\n <app-preferences-section\r\n [settings]=\"settings\"\r\n [colorTokens]=\"colorTokens\"\r\n [isCustomColorSelected]=\"isCustomColorSelected()\"\r\n [defaultColor]=\"personalization.developerDefaults().primaryColor || 'default'\"\r\n (onThemeSelect)=\"selectTheme($event)\"\r\n (onColorSelect)=\"selectColor($event)\"\r\n (onButtonStyleSelect)=\"selectButtonStyle($event)\"\r\n (onSave)=\"onSave()\"\r\n (onReset)=\"onReset()\"\r\n >\r\n </app-preferences-section>\r\n }\r\n\r\n <!-- Account / Profile Section -->\r\n @if (activeSection === 'account_settings') {\r\n <app-account-section\r\n (onDeleteAllHistory)=\"deleteAllHistory()\"\r\n (onLogoutAll)=\"logoutAllDevices()\"\r\n (onDeleteAccount)=\"deleteAccount()\"\r\n ></app-account-section>\r\n }\r\n\r\n @if (activeSection === 'profile') {\r\n <app-profile-section\r\n [username]=\"username\"\r\n [userImageUrl]=\"userImageUrl\"\r\n [userRole]=\"userRole\"\r\n ></app-profile-section>\r\n }\r\n\r\n <!-- General Section -->\r\n @if (activeSection === 'general') {\r\n <app-setting-section\r\n [settings]=\"settings\"\r\n [hasChanges]=\"hasChanges\"\r\n (onSave)=\"onSave()\"\r\n ></app-setting-section>\r\n }\r\n\r\n <!-- Terms and Conditions Section -->\r\n @if (activeSection === 'terms') {\r\n <app-terms-section></app-terms-section>\r\n }\r\n </main>\r\n </mat-dialog-content>\r\n</div>\r\n", styles: [".personalization-container{display:flex;flex-direction:column;width:100%;height:100%;max-width:none;max-height:none;max-height:90vh;background:var(--background-color);color:var(--text-color);border-radius:12px;overflow:hidden;box-shadow:var(--border-shadow-color)}.personalization-container .header{flex:0 0 auto;display:flex;justify-content:space-between;align-items:center;padding:12px 20px;border-bottom:1px solid var(--border-color);background:var(--background-color)}.personalization-container .header .title-section{display:flex;align-items:center;gap:10px}.personalization-container .header .title-section .header-icon{color:var(--secondary-text-color);font-size:20px;width:20px;height:20px}.personalization-container .header .title-section h2{margin:0;font-size:.95rem;font-weight:500;color:var(--secondary-text-color)}.personalization-container .dialog-layout{display:flex;padding:0;margin:0;flex:1;overflow:hidden}.personalization-container .sidebar{width:240px;background:rgba(var(--black-rgb),.02);border-right:1px solid var(--border-color);padding:16px 8px;display:flex;flex-direction:column;gap:2px;overflow-y:auto;scrollbar-gutter:stable}.personalization-container .sidebar::-webkit-scrollbar{width:4px}.personalization-container .sidebar::-webkit-scrollbar-track{background:transparent}.personalization-container .sidebar::-webkit-scrollbar-thumb{background:rgba(var(--black-rgb),.1);border-radius:10px}.personalization-container .sidebar::-webkit-scrollbar-thumb:hover{background:rgba(var(--black-rgb),.2)}.personalization-container .sidebar .nav-header{padding:12px 12px 6px;font-size:.75rem;font-weight:600;text-transform:uppercase;color:var(--secondary-text-color);letter-spacing:.5px}.personalization-container .sidebar .nav-item{display:flex;align-items:center;gap:12px;padding:8px 12px;border-radius:6px;cursor:pointer;transition:all .2s ease;color:var(--text-color)}.personalization-container .sidebar .nav-item mat-icon{font-size:18px;width:18px;height:18px;color:var(--secondary-text-color)}.personalization-container .sidebar .nav-item span{font-size:.9rem;font-weight:400}.personalization-container .sidebar .nav-item:hover{background:rgba(var(--black-rgb),.05)}.personalization-container .sidebar .nav-item.active{background:rgba(var(--primary-color-rgb),.1);font-weight:500;color:var(--primary-color)}.personalization-container .sidebar .nav-item.active mat-icon{color:var(--primary-color)}.personalization-container .main-content{flex:1;overflow-y:auto;background:var(--background-color);position:relative}.personalization-container .main-content::-webkit-scrollbar{width:6px}.personalization-container .main-content::-webkit-scrollbar-track{background:transparent}.personalization-container .main-content::-webkit-scrollbar-thumb{background:rgba(var(--black-rgb),.1);border-radius:10px}.personalization-container .main-content::-webkit-scrollbar-thumb:hover{background:rgba(var(--black-rgb),.2)}.full-width{width:100%}@keyframes slideIn{0%{opacity:0;transform:translateY(4px)}to{opacity:1;transform:translateY(0)}}:host-context(.dark-theme) .personalization-container .sidebar{background:var(--background-color)}\n"] }]
|
|
193
|
+
}], ctorParameters: () => [] });
|
|
194
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"personalization-dialog.component.js","sourceRoot":"","sources":["../../../../../../projects/doohbot/src/lib/app/personalization/personalization-dialog.component.ts","../../../../../../projects/doohbot/src/lib/app/personalization/personalization-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, OnDestroy } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { FormsModule } from '@angular/forms';\r\nimport { MatDialogModule, MatDialogRef } from '@angular/material/dialog';\r\nimport { MatButtonModule } from '@angular/material/button';\r\nimport { MatIconModule } from '@angular/material/icon';\r\nimport { MatTabsModule } from '@angular/material/tabs';\r\nimport { MatSlideToggleModule } from '@angular/material/slide-toggle';\r\nimport { MatFormFieldModule } from '@angular/material/form-field';\r\nimport { MatInputModule } from '@angular/material/input';\r\nimport { MatSelectModule } from '@angular/material/select';\r\nimport { MatTooltipModule } from '@angular/material/tooltip';\r\nimport { PreferencesSectionComponent } from './sections/preferences/preferences-section.component';\r\nimport { ProfileSectionComponent } from './sections/profile/profile-section.component';\r\nimport { TermsSectionComponent } from './sections/terms/terms-section.component';\r\nimport { SettingSectionComponent } from './sections/settings/setting-section.component';\r\nimport { AccountSectionComponent } from './sections/account/account-section.component';\r\nimport { ChatHistoryService } from '../chat/services/chat-history.service';\r\nimport { AuthService } from '../../core/auth/auth.service';\r\nimport { PersonalizationService, PersonalizationSettings } from './personalization.service';\r\nimport { DialogService } from '../../shared/dialog/dialog.service';\r\nimport { DialogUtils } from '../../shared/dialog/dialog.utils';\r\nimport { SnackbarService } from '../../shared/snackbar/snackbar.service';\r\nimport { ThemeService } from '../../core/services/theme.service';\r\n\r\n@Component({\r\n  selector: 'app-personalization-dialog',\r\n  standalone: true,\r\n  imports: [\r\n    CommonModule,\r\n    FormsModule,\r\n    MatDialogModule,\r\n    MatButtonModule,\r\n    MatIconModule,\r\n    MatTabsModule,\r\n    MatSlideToggleModule,\r\n    MatFormFieldModule,\r\n    MatInputModule,\r\n    MatSelectModule,\r\n    MatTooltipModule,\r\n    PreferencesSectionComponent,\r\n    ProfileSectionComponent,\r\n    SettingSectionComponent,\r\n    TermsSectionComponent,\r\n    AccountSectionComponent,\r\n  ],\r\n  templateUrl: './personalization-dialog.component.html',\r\n  styleUrls: ['./personalization-dialog.component.scss'],\r\n})\r\nexport class PersonalizationDialogComponent implements OnDestroy {\r\n  private dialogRef = inject(MatDialogRef<PersonalizationDialogComponent>);\r\n  private dialogService = inject(DialogService);\r\n  public personalization = inject(PersonalizationService);\r\n  private authService = inject(AuthService);\r\n  private chatHistory = inject(ChatHistoryService);\r\n  private snackbar = inject(SnackbarService);\r\n\r\n  // Local copy of settings for editing\r\n  public settings: PersonalizationSettings = JSON.parse(\r\n    JSON.stringify(this.personalization.settings()),\r\n  );\r\n\r\n  public activeSection = 'profile';\r\n\r\n  public sections = [\r\n    { id: 'account', label: 'Account', type: 'header' },\r\n    { id: 'profile', label: 'My profile', icon: 'person_outline', type: 'item' },\r\n    { id: 'account_settings', label: 'Manage Account', icon: 'manage_accounts', type: 'item' },\r\n    { id: 'preferences', label: 'Preferences', icon: 'tune', type: 'item' },\r\n    { id: 'workspace', label: 'Workspace', type: 'header' },\r\n    { id: 'general', label: 'Settings', icon: 'settings', type: 'item' },\r\n    { id: 'info', label: 'Support', type: 'header' },\r\n    { id: 'terms', label: 'Terms & Conditions', icon: 'description', type: 'item' },\r\n  ];\r\n\r\n  public colorTokens = [\r\n    {\r\n      name: 'Standard',\r\n      value: 'default',\r\n    },\r\n    { name: 'Purple', value: '#7c3aed' },\r\n    { name: 'Pink', value: '#db2777' },\r\n    { name: 'Red', value: '#dc2626' },\r\n    { name: 'Orange', value: '#ea580c' },\r\n    { name: 'Green', value: '#16a34a' },\r\n    { name: 'Teal', value: '#0d9488' },\r\n    { name: 'Blue', value: '#2563eb' },\r\n  ];\r\n\r\n  private themeService = inject(ThemeService);\r\n  private originalColor: string = '';\r\n  private isSaved = false;\r\n\r\n  constructor() {\r\n    this.originalColor = this.personalization.settings().appearance.primaryColor;\r\n  }\r\n\r\n  onSave() {\r\n    if (this.activeSection === 'preferences') {\r\n      this.isSaved = true;\r\n      this.personalization.updateAppearance(this.settings.appearance);\r\n      // Re-apply to ensure consistency (effect handles it, but just in case)\r\n      this.snackbar.success('Appearance settings saved');\r\n    } else if (this.activeSection === 'general') {\r\n      this.personalization.updateInstructions(this.settings.customInstructions);\r\n      this.snackbar.success('Settings saved');\r\n    }\r\n  }\r\n\r\n  onCancel() {\r\n    this.dialogRef.close(false);\r\n    // ngOnDestroy will handle revert\r\n  }\r\n\r\n  ngOnDestroy() {\r\n    if (!this.isSaved) {\r\n      // Revert to original color if not saved\r\n      this.themeService.applyPrimaryColorToGlobal(this.originalColor);\r\n    }\r\n  }\r\n\r\n  onReset() {\r\n    DialogUtils.confirmReset(this.dialogService).subscribe((confirmed) => {\r\n      if (confirmed) {\r\n        // Only reset the current section\r\n        if (this.activeSection === 'preferences') {\r\n          // Reset only appearance settings to defaults\r\n          this.personalization.resetToDefaults();\r\n          // Re-clone settings to update UI\r\n          this.settings = JSON.parse(JSON.stringify(this.personalization.settings()));\r\n\r\n          // Live preview the reset\r\n          this.themeService.applyPrimaryColorToGlobal(this.settings.appearance.primaryColor);\r\n        }\r\n      }\r\n    });\r\n  }\r\n\r\n  selectTheme(theme: 'light' | 'dark' | 'auto') {\r\n    this.settings.appearance.theme = theme;\r\n    // Optional: Preview theme? User only asked for color.\r\n  }\r\n\r\n  selectColor(color: string) {\r\n    this.settings.appearance.primaryColor = color;\r\n    //! Live Preview for changed color\r\n    // this.themeService.applyPrimaryColorToGlobal(color);\r\n  }\r\n\r\n  selectButtonStyle(style: 'fab' | 'sidebar' | 'sidebar-top' | 'sidebar-bottom') {\r\n    this.settings.appearance.buttonStyle = style;\r\n  }\r\n\r\n  selectSection(sectionId: string) {\r\n    if (this.sections.find((s) => s.id === sectionId && s.type === 'item')) {\r\n      this.activeSection = sectionId;\r\n    }\r\n  }\r\n\r\n  get hasChanges(): boolean {\r\n    if (this.activeSection === 'preferences') return this.hasPreferencesChanges();\r\n    if (this.activeSection === 'general') return this.hasInstructionChanges();\r\n    return false;\r\n  }\r\n\r\n  hasPreferencesChanges(): boolean {\r\n    const original = this.personalization.settings().appearance;\r\n    const current = this.settings.appearance;\r\n    return JSON.stringify(original) !== JSON.stringify(current);\r\n  }\r\n\r\n  hasInstructionChanges(): boolean {\r\n    const original = this.personalization.settings().customInstructions;\r\n    const current = this.settings.customInstructions;\r\n    return JSON.stringify(original) !== JSON.stringify(current);\r\n  }\r\n\r\n  get username(): string {\r\n    return this.authService.getLoggedInUserName();\r\n  }\r\n\r\n  get userImageUrl(): string | null {\r\n    return this.authService.getLoggedInUserImageUrl();\r\n  }\r\n\r\n  get userRole(): string {\r\n    return this.authService.getLoggedInUserRole();\r\n  }\r\n\r\n  isCustomColorSelected(): boolean {\r\n    return !this.colorTokens.some(\r\n      (token) => token.value.toLowerCase() === this.settings.appearance.primaryColor.toLowerCase(),\r\n    );\r\n  }\r\n\r\n  onCustomColorChange(event: Event) {\r\n    const color = (event.target as HTMLInputElement).value;\r\n    this.selectColor(color);\r\n  }\r\n\r\n  deleteAllHistory() {\r\n    const userId = this.authService.getUserId() || '';\r\n    this.chatHistory.deleteAllHistory(userId.toString()).subscribe(() => {\r\n      // History cleared\r\n    });\r\n  }\r\n\r\n  logoutAllDevices() {\r\n    console.log('Logging out of all devices...');\r\n    // Implement service call when API is ready\r\n  }\r\n\r\n  deleteAccount() {\r\n    console.log('Deleting account...');\r\n    // Implement service call when API is ready\r\n  }\r\n}\r\n","<div class=\"personalization-container\">\r\n  <div class=\"header\">\r\n    <div class=\"title-section\">\r\n      <mat-icon class=\"header-icon\">settings_suggest</mat-icon>\r\n      <h2>Personalization</h2>\r\n    </div>\r\n    <button mat-icon-button (click)=\"onCancel()\" matTooltip=\"Close\">\r\n      <mat-icon>close</mat-icon>\r\n    </button>\r\n  </div>\r\n\r\n  <mat-dialog-content class=\"dialog-layout no-padding\">\r\n    <!-- Sidebar Navigation -->\r\n    <aside class=\"sidebar\">\r\n      @for (section of sections; track section.id) {\r\n        @if (section.type === 'header') {\r\n          <div class=\"nav-header\">{{ section.label }}</div>\r\n        } @else {\r\n          <div\r\n            class=\"nav-item\"\r\n            [class.active]=\"activeSection === section.id\"\r\n            (click)=\"selectSection(section.id)\"\r\n          >\r\n            <mat-icon>{{ section.icon }}</mat-icon>\r\n            <span>{{ section.label }}</span>\r\n          </div>\r\n        }\r\n      }\r\n    </aside>\r\n\r\n    <!-- Main Content Area -->\r\n    <main class=\"main-content\">\r\n      <!-- Appearance / Preferences Section -->\r\n      @if (activeSection === 'preferences') {\r\n        <app-preferences-section\r\n          [settings]=\"settings\"\r\n          [colorTokens]=\"colorTokens\"\r\n          [isCustomColorSelected]=\"isCustomColorSelected()\"\r\n          [defaultColor]=\"personalization.developerDefaults().primaryColor || 'default'\"\r\n          (onThemeSelect)=\"selectTheme($event)\"\r\n          (onColorSelect)=\"selectColor($event)\"\r\n          (onButtonStyleSelect)=\"selectButtonStyle($event)\"\r\n          (onSave)=\"onSave()\"\r\n          (onReset)=\"onReset()\"\r\n        >\r\n        </app-preferences-section>\r\n      }\r\n\r\n      <!-- Account / Profile Section -->\r\n      @if (activeSection === 'account_settings') {\r\n        <app-account-section\r\n          (onDeleteAllHistory)=\"deleteAllHistory()\"\r\n          (onLogoutAll)=\"logoutAllDevices()\"\r\n          (onDeleteAccount)=\"deleteAccount()\"\r\n        ></app-account-section>\r\n      }\r\n\r\n      @if (activeSection === 'profile') {\r\n        <app-profile-section\r\n          [username]=\"username\"\r\n          [userImageUrl]=\"userImageUrl\"\r\n          [userRole]=\"userRole\"\r\n        ></app-profile-section>\r\n      }\r\n\r\n      <!-- General Section -->\r\n      @if (activeSection === 'general') {\r\n        <app-setting-section\r\n          [settings]=\"settings\"\r\n          [hasChanges]=\"hasChanges\"\r\n          (onSave)=\"onSave()\"\r\n        ></app-setting-section>\r\n      }\r\n\r\n      <!-- Terms and Conditions Section -->\r\n      @if (activeSection === 'terms') {\r\n        <app-terms-section></app-terms-section>\r\n      }\r\n    </main>\r\n  </mat-dialog-content>\r\n</div>\r\n"]}
|
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
import { Injectable, signal, inject, effect } from '@angular/core';
|
|
2
|
+
import { StorageService } from '../../core/auth/storage.service';
|
|
3
|
+
import { AuthService } from '../../core/auth/auth.service';
|
|
4
|
+
import { toSignal } from '@angular/core/rxjs-interop';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
const BASE_STORAGE_KEY = 'app.personalization.v1';
|
|
7
|
+
const DEFAULT_SETTINGS = {
|
|
8
|
+
customInstructions: {
|
|
9
|
+
enabled: false,
|
|
10
|
+
aboutUser: '',
|
|
11
|
+
responseStyle: '',
|
|
12
|
+
characteristics: {
|
|
13
|
+
warm: 'Default',
|
|
14
|
+
enthusiastic: 'Default',
|
|
15
|
+
headersAndLists: 'Default',
|
|
16
|
+
emoji: 'Default',
|
|
17
|
+
},
|
|
18
|
+
aboutYou: {
|
|
19
|
+
nickname: '',
|
|
20
|
+
occupation: '',
|
|
21
|
+
},
|
|
22
|
+
},
|
|
23
|
+
appearance: {
|
|
24
|
+
theme: 'auto',
|
|
25
|
+
primaryColor: 'default',
|
|
26
|
+
buttonStyle: 'fab',
|
|
27
|
+
},
|
|
28
|
+
meta: {
|
|
29
|
+
version: 1,
|
|
30
|
+
lastUpdated: Date.now(),
|
|
31
|
+
},
|
|
32
|
+
};
|
|
33
|
+
export class PersonalizationService {
|
|
34
|
+
constructor() {
|
|
35
|
+
this.storage = inject(StorageService);
|
|
36
|
+
this.authService = inject(AuthService);
|
|
37
|
+
// Reactive Authentication Status
|
|
38
|
+
this.authStatus = toSignal(this.authService.authStatus$);
|
|
39
|
+
// State Signals
|
|
40
|
+
this._settings = signal(DEFAULT_SETTINGS);
|
|
41
|
+
this.settings = this._settings.asReadonly();
|
|
42
|
+
// Developer-provided defaults (from DoohbotInput)
|
|
43
|
+
this._developerDefaults = signal({});
|
|
44
|
+
this.developerDefaults = this._developerDefaults.asReadonly();
|
|
45
|
+
// Reload settings when auth status changes (user logins or logouts)
|
|
46
|
+
effect(() => {
|
|
47
|
+
const status = this.authStatus();
|
|
48
|
+
this._settings.set(this.loadSettings());
|
|
49
|
+
}, { allowSignalWrites: true });
|
|
50
|
+
// Automatically persist changes when settings signal updates
|
|
51
|
+
effect(() => {
|
|
52
|
+
const currentSettings = this._settings();
|
|
53
|
+
this.saveSettings(currentSettings);
|
|
54
|
+
});
|
|
55
|
+
}
|
|
56
|
+
getStorageKey() {
|
|
57
|
+
const userId = this.authService.getUserId();
|
|
58
|
+
return userId ? `${BASE_STORAGE_KEY}_${userId}` : BASE_STORAGE_KEY;
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Update custom instructions
|
|
62
|
+
*/
|
|
63
|
+
updateInstructions(instructions) {
|
|
64
|
+
this._settings.update((s) => ({
|
|
65
|
+
...s,
|
|
66
|
+
customInstructions: { ...s.customInstructions, ...instructions },
|
|
67
|
+
meta: { ...s.meta, lastUpdated: Date.now() },
|
|
68
|
+
}));
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Update appearance settings
|
|
72
|
+
*/
|
|
73
|
+
updateAppearance(appearance) {
|
|
74
|
+
this._settings.update((s) => ({
|
|
75
|
+
...s,
|
|
76
|
+
appearance: { ...s.appearance, ...appearance },
|
|
77
|
+
meta: { ...s.meta, lastUpdated: Date.now() },
|
|
78
|
+
}));
|
|
79
|
+
}
|
|
80
|
+
/**
|
|
81
|
+
* Set initial developer defaults (called from DoohbotComponent)
|
|
82
|
+
*/
|
|
83
|
+
setDeveloperDefaults(defaults) {
|
|
84
|
+
this._developerDefaults.set(defaults);
|
|
85
|
+
// Refresh settings to apply defaults if no user settings are saved
|
|
86
|
+
this._settings.set(this.loadSettings());
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Reset all personalization to defaults
|
|
90
|
+
*/
|
|
91
|
+
resetToDefaults() {
|
|
92
|
+
this._settings.set({
|
|
93
|
+
...DEFAULT_SETTINGS,
|
|
94
|
+
meta: { ...DEFAULT_SETTINGS.meta, lastUpdated: Date.now() },
|
|
95
|
+
});
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* Get the formatted instructions block for chat requests
|
|
99
|
+
*/
|
|
100
|
+
getFormattedInstructions() {
|
|
101
|
+
const s = this._settings().customInstructions;
|
|
102
|
+
if (!s.enabled)
|
|
103
|
+
return '';
|
|
104
|
+
let block = '--- CUSTOM INSTRUCTIONS ---\n';
|
|
105
|
+
if (s.aboutUser) {
|
|
106
|
+
block += `USER CONTEXT: ${s.aboutUser}\n`;
|
|
107
|
+
}
|
|
108
|
+
if (s.responseStyle) {
|
|
109
|
+
block += `RESPONSE STYLE: ${s.responseStyle}\n`;
|
|
110
|
+
}
|
|
111
|
+
block += '---------------------------';
|
|
112
|
+
return s.aboutUser || s.responseStyle ? block : '';
|
|
113
|
+
}
|
|
114
|
+
loadSettings() {
|
|
115
|
+
try {
|
|
116
|
+
const key = this.getStorageKey();
|
|
117
|
+
// StorageService.get handles decryption/parsing
|
|
118
|
+
const saved = this.storage.get(key);
|
|
119
|
+
if (saved) {
|
|
120
|
+
return { ...DEFAULT_SETTINGS, ...saved };
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
catch (e) {
|
|
124
|
+
console.error('Failed to load personalization settings', e);
|
|
125
|
+
}
|
|
126
|
+
// Return defaults merged with developer overrides
|
|
127
|
+
return {
|
|
128
|
+
...DEFAULT_SETTINGS,
|
|
129
|
+
appearance: {
|
|
130
|
+
...DEFAULT_SETTINGS.appearance,
|
|
131
|
+
...this._developerDefaults(),
|
|
132
|
+
primaryColor: 'default', //! Force 'default' for fresh loads so it resolves dynamically
|
|
133
|
+
},
|
|
134
|
+
};
|
|
135
|
+
}
|
|
136
|
+
saveSettings(settings) {
|
|
137
|
+
const key = this.getStorageKey();
|
|
138
|
+
this.storage.set(key, settings);
|
|
139
|
+
}
|
|
140
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PersonalizationService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
141
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PersonalizationService, providedIn: 'root' }); }
|
|
142
|
+
}
|
|
143
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PersonalizationService, decorators: [{
|
|
144
|
+
type: Injectable,
|
|
145
|
+
args: [{
|
|
146
|
+
providedIn: 'root',
|
|
147
|
+
}]
|
|
148
|
+
}], ctorParameters: () => [] });
|
|
149
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"personalization.service.js","sourceRoot":"","sources":["../../../../../../projects/doohbot/src/lib/app/personalization/personalization.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, signal, inject, effect } from '@angular/core';\r\nimport { StorageService } from '../../core/auth/storage.service';\r\nimport { AuthService } from '../../core/auth/auth.service';\r\nimport { toSignal } from '@angular/core/rxjs-interop';\r\n\r\nexport interface Characteristics {\r\n  warm: string;\r\n  enthusiastic: string;\r\n  headersAndLists: string;\r\n  emoji: string;\r\n}\r\n\r\nexport interface AboutYou {\r\n  nickname: string;\r\n  occupation: string;\r\n}\r\n\r\nexport interface CustomInstructions {\r\n  enabled: boolean;\r\n  aboutUser: string;\r\n  responseStyle: string;\r\n  characteristics: Characteristics;\r\n  aboutYou: AboutYou;\r\n}\r\n\r\nexport interface AppearanceSettings {\r\n  theme: 'light' | 'dark' | 'auto';\r\n  primaryColor: string;\r\n  buttonStyle: 'fab' | 'sidebar' | 'sidebar-top' | 'sidebar-bottom';\r\n}\r\n\r\nexport interface PersonalizationSettings {\r\n  customInstructions: CustomInstructions;\r\n  appearance: AppearanceSettings;\r\n  meta: {\r\n    version: number;\r\n    lastUpdated: number;\r\n  };\r\n}\r\n\r\nconst BASE_STORAGE_KEY = 'app.personalization.v1';\r\n\r\nconst DEFAULT_SETTINGS: PersonalizationSettings = {\r\n  customInstructions: {\r\n    enabled: false,\r\n    aboutUser: '',\r\n    responseStyle: '',\r\n    characteristics: {\r\n      warm: 'Default',\r\n      enthusiastic: 'Default',\r\n      headersAndLists: 'Default',\r\n      emoji: 'Default',\r\n    },\r\n    aboutYou: {\r\n      nickname: '',\r\n      occupation: '',\r\n    },\r\n  },\r\n  appearance: {\r\n    theme: 'auto',\r\n    primaryColor: 'default', // Standard primary color (resolves to developer default)\r\n    buttonStyle: 'fab',\r\n  },\r\n  meta: {\r\n    version: 1,\r\n    lastUpdated: Date.now(),\r\n  },\r\n};\r\n\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class PersonalizationService {\r\n  private storage = inject(StorageService);\r\n  private authService = inject(AuthService);\r\n\r\n  // Reactive Authentication Status\r\n  private authStatus = toSignal(this.authService.authStatus$);\r\n\r\n  // State Signals\r\n  private _settings = signal<PersonalizationSettings>(DEFAULT_SETTINGS);\r\n  public settings = this._settings.asReadonly();\r\n\r\n  // Developer-provided defaults (from DoohbotInput)\r\n  private _developerDefaults = signal<Partial<AppearanceSettings>>({});\r\n  public readonly developerDefaults = this._developerDefaults.asReadonly();\r\n\r\n  constructor() {\r\n    // Reload settings when auth status changes (user logins or logouts)\r\n    effect(\r\n      () => {\r\n        const status = this.authStatus();\r\n        this._settings.set(this.loadSettings());\r\n      },\r\n      { allowSignalWrites: true }, // TODO\r\n    );\r\n\r\n    // Automatically persist changes when settings signal updates\r\n    effect(() => {\r\n      const currentSettings = this._settings();\r\n      this.saveSettings(currentSettings);\r\n    });\r\n  }\r\n\r\n  private getStorageKey(): string {\r\n    const userId = this.authService.getUserId();\r\n    return userId ? `${BASE_STORAGE_KEY}_${userId}` : BASE_STORAGE_KEY;\r\n  }\r\n\r\n  /**\r\n   * Update custom instructions\r\n   */\r\n  updateInstructions(instructions: Partial<CustomInstructions>) {\r\n    this._settings.update((s) => ({\r\n      ...s,\r\n      customInstructions: { ...s.customInstructions, ...instructions },\r\n      meta: { ...s.meta, lastUpdated: Date.now() },\r\n    }));\r\n  }\r\n\r\n  /**\r\n   * Update appearance settings\r\n   */\r\n  updateAppearance(appearance: Partial<AppearanceSettings>) {\r\n    this._settings.update((s) => ({\r\n      ...s,\r\n      appearance: { ...s.appearance, ...appearance },\r\n      meta: { ...s.meta, lastUpdated: Date.now() },\r\n    }));\r\n  }\r\n\r\n  /**\r\n   * Set initial developer defaults (called from DoohbotComponent)\r\n   */\r\n  setDeveloperDefaults(defaults: Partial<AppearanceSettings>) {\r\n    this._developerDefaults.set(defaults);\r\n    // Refresh settings to apply defaults if no user settings are saved\r\n    this._settings.set(this.loadSettings());\r\n  }\r\n\r\n  /**\r\n   * Reset all personalization to defaults\r\n   */\r\n  resetToDefaults() {\r\n    this._settings.set({\r\n      ...DEFAULT_SETTINGS,\r\n      meta: { ...DEFAULT_SETTINGS.meta, lastUpdated: Date.now() },\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Get the formatted instructions block for chat requests\r\n   */\r\n  getFormattedInstructions(): string {\r\n    const s = this._settings().customInstructions;\r\n    if (!s.enabled) return '';\r\n\r\n    let block = '--- CUSTOM INSTRUCTIONS ---\\n';\r\n    if (s.aboutUser) {\r\n      block += `USER CONTEXT: ${s.aboutUser}\\n`;\r\n    }\r\n    if (s.responseStyle) {\r\n      block += `RESPONSE STYLE: ${s.responseStyle}\\n`;\r\n    }\r\n    block += '---------------------------';\r\n\r\n    return s.aboutUser || s.responseStyle ? block : '';\r\n  }\r\n\r\n  private loadSettings(): PersonalizationSettings {\r\n    try {\r\n      const key = this.getStorageKey();\r\n      // StorageService.get handles decryption/parsing\r\n      const saved = this.storage.get(key);\r\n      if (saved) {\r\n        return { ...DEFAULT_SETTINGS, ...saved };\r\n      }\r\n    } catch (e) {\r\n      console.error('Failed to load personalization settings', e);\r\n    }\r\n\r\n    // Return defaults merged with developer overrides\r\n    return {\r\n      ...DEFAULT_SETTINGS,\r\n      appearance: {\r\n        ...DEFAULT_SETTINGS.appearance,\r\n        ...this._developerDefaults(),\r\n        primaryColor: 'default', //! Force 'default'  for fresh loads so it resolves dynamically\r\n      },\r\n    };\r\n  }\r\n\r\n  private saveSettings(settings: PersonalizationSettings) {\r\n    const key = this.getStorageKey();\r\n    this.storage.set(key, settings);\r\n  }\r\n}\r\n"]}
|