hazo_auth 1.4.2 → 1.6.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/SETUP_CHECKLIST.md +708 -0
- package/dist/app/api/hazo_auth/change_password/route.d.ts +8 -0
- package/dist/app/api/hazo_auth/change_password/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/change_password/route.js +98 -0
- package/dist/app/api/hazo_auth/forgot_password/route.d.ts +8 -0
- package/dist/app/api/hazo_auth/forgot_password/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/forgot_password/route.js +78 -0
- package/dist/app/api/hazo_auth/get_auth/route.d.ts +10 -0
- package/dist/app/api/hazo_auth/get_auth/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/get_auth/route.js +63 -0
- package/dist/app/api/hazo_auth/invalidate_cache/route.d.ts +14 -0
- package/dist/app/api/hazo_auth/invalidate_cache/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/invalidate_cache/route.js +96 -0
- package/dist/app/api/hazo_auth/library_photos/route.d.ts +13 -0
- package/dist/app/api/hazo_auth/library_photos/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/library_photos/route.js +55 -0
- package/dist/app/api/hazo_auth/login/route.d.ts +12 -0
- package/dist/app/api/hazo_auth/login/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/login/route.js +140 -0
- package/dist/app/api/hazo_auth/logout/route.d.ts +8 -0
- package/dist/app/api/hazo_auth/logout/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/logout/route.js +71 -0
- package/dist/app/api/hazo_auth/me/route.d.ts +3 -0
- package/dist/app/api/hazo_auth/me/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/me/route.js +34 -0
- package/dist/app/api/hazo_auth/profile_picture/[filename]/route.d.ts +7 -0
- package/dist/app/api/hazo_auth/profile_picture/[filename]/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/profile_picture/[filename]/route.js +43 -0
- package/dist/app/api/hazo_auth/register/route.d.ts +9 -0
- package/dist/app/api/hazo_auth/register/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/register/route.js +80 -0
- package/dist/app/api/hazo_auth/remove_profile_picture/route.d.ts +8 -0
- package/dist/app/api/hazo_auth/remove_profile_picture/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/remove_profile_picture/route.js +64 -0
- package/dist/app/api/hazo_auth/resend_verification/route.d.ts +8 -0
- package/dist/app/api/hazo_auth/resend_verification/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/resend_verification/route.js +79 -0
- package/dist/app/api/hazo_auth/reset_password/route.d.ts +8 -0
- package/dist/app/api/hazo_auth/reset_password/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/reset_password/route.js +76 -0
- package/dist/app/api/hazo_auth/update_user/route.d.ts +9 -0
- package/dist/app/api/hazo_auth/update_user/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/update_user/route.js +95 -0
- package/dist/app/api/hazo_auth/upload_profile_picture/route.d.ts +9 -0
- package/dist/app/api/hazo_auth/upload_profile_picture/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/upload_profile_picture/route.js +204 -0
- package/dist/app/api/hazo_auth/validate_reset_token/route.d.ts +6 -0
- package/dist/app/api/hazo_auth/validate_reset_token/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/validate_reset_token/route.js +58 -0
- package/dist/app/api/hazo_auth/verify_email/route.d.ts +11 -0
- package/dist/app/api/hazo_auth/verify_email/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/verify_email/route.js +63 -0
- package/dist/cli/generate.d.ts +2 -0
- package/dist/cli/generate.d.ts.map +1 -0
- package/dist/cli/generate.js +117 -0
- package/dist/cli/index.d.ts +3 -0
- package/dist/cli/index.d.ts.map +1 -0
- package/dist/cli/index.js +120 -0
- package/dist/cli/validate.d.ts +15 -0
- package/dist/cli/validate.d.ts.map +1 -0
- package/dist/cli/validate.js +509 -0
- package/dist/components/ui/card.d.ts +9 -0
- package/dist/components/ui/card.d.ts.map +1 -0
- package/dist/components/ui/card.js +45 -0
- package/dist/hooks/use-mobile.d.ts.map +1 -1
- package/dist/hooks/use-mobile.js +17 -3
- package/dist/server/routes/change_password.d.ts +2 -0
- package/dist/server/routes/change_password.d.ts.map +1 -0
- package/dist/server/routes/change_password.js +2 -0
- package/dist/server/routes/forgot_password.d.ts +2 -0
- package/dist/server/routes/forgot_password.d.ts.map +1 -0
- package/dist/server/routes/forgot_password.js +2 -0
- package/dist/server/routes/get_auth.d.ts +2 -0
- package/dist/server/routes/get_auth.d.ts.map +1 -0
- package/dist/server/routes/get_auth.js +2 -0
- package/dist/server/routes/index.d.ts +18 -0
- package/dist/server/routes/index.d.ts.map +1 -0
- package/dist/server/routes/index.js +24 -0
- package/dist/server/routes/invalidate_cache.d.ts +2 -0
- package/dist/server/routes/invalidate_cache.d.ts.map +1 -0
- package/dist/server/routes/invalidate_cache.js +2 -0
- package/dist/server/routes/library_photos.d.ts +2 -0
- package/dist/server/routes/library_photos.d.ts.map +1 -0
- package/dist/server/routes/library_photos.js +2 -0
- package/dist/server/routes/login.d.ts +2 -0
- package/dist/server/routes/login.d.ts.map +1 -0
- package/dist/server/routes/login.js +2 -0
- package/dist/server/routes/logout.d.ts +2 -0
- package/dist/server/routes/logout.d.ts.map +1 -0
- package/dist/server/routes/logout.js +2 -0
- package/dist/server/routes/me.d.ts +2 -0
- package/dist/server/routes/me.d.ts.map +1 -0
- package/dist/server/routes/me.js +2 -0
- package/dist/server/routes/profile_picture_filename.d.ts +2 -0
- package/dist/server/routes/profile_picture_filename.d.ts.map +1 -0
- package/dist/server/routes/profile_picture_filename.js +3 -0
- package/dist/server/routes/register.d.ts +2 -0
- package/dist/server/routes/register.d.ts.map +1 -0
- package/dist/server/routes/register.js +2 -0
- package/dist/server/routes/remove_profile_picture.d.ts +2 -0
- package/dist/server/routes/remove_profile_picture.d.ts.map +1 -0
- package/dist/server/routes/remove_profile_picture.js +2 -0
- package/dist/server/routes/resend_verification.d.ts +2 -0
- package/dist/server/routes/resend_verification.d.ts.map +1 -0
- package/dist/server/routes/resend_verification.js +2 -0
- package/dist/server/routes/reset_password.d.ts +2 -0
- package/dist/server/routes/reset_password.d.ts.map +1 -0
- package/dist/server/routes/reset_password.js +2 -0
- package/dist/server/routes/update_user.d.ts +2 -0
- package/dist/server/routes/update_user.d.ts.map +1 -0
- package/dist/server/routes/update_user.js +2 -0
- package/dist/server/routes/upload_profile_picture.d.ts +2 -0
- package/dist/server/routes/upload_profile_picture.d.ts.map +1 -0
- package/dist/server/routes/upload_profile_picture.js +2 -0
- package/dist/server/routes/validate_reset_token.d.ts +2 -0
- package/dist/server/routes/validate_reset_token.d.ts.map +1 -0
- package/dist/server/routes/validate_reset_token.js +2 -0
- package/dist/server/routes/verify_email.d.ts +2 -0
- package/dist/server/routes/verify_email.d.ts.map +1 -0
- package/dist/server/routes/verify_email.js +2 -0
- package/package.json +12 -17
- package/components.json +0 -22
- package/instrumentation.ts +0 -32
- package/migrations/001_add_token_type_to_refresh_tokens.sql +0 -14
- package/migrations/002_add_name_to_hazo_users.sql +0 -7
- package/migrations/003_add_url_on_logon_to_hazo_users.sql +0 -8
- package/next.config.mjs +0 -67
- package/postcss.config.mjs +0 -8
- package/public/file.svg +0 -1
- package/public/globe.svg +0 -1
- package/public/next.svg +0 -1
- package/public/vercel.svg +0 -1
- package/public/window.svg +0 -1
- package/scripts/apply_migration.ts +0 -118
- package/scripts/init_users.ts +0 -378
- package/src/app/api/hazo_auth/auth/upload_profile_picture/route.ts +0 -268
- package/src/app/api/hazo_auth/change_password/route.ts +0 -132
- package/src/app/api/hazo_auth/forgot_password/route.ts +0 -107
- package/src/app/api/hazo_auth/get_auth/route.ts +0 -89
- package/src/app/api/hazo_auth/invalidate_cache/route.ts +0 -139
- package/src/app/api/hazo_auth/library_photos/route.ts +0 -73
- package/src/app/api/hazo_auth/login/route.ts +0 -181
- package/src/app/api/hazo_auth/logout/route.ts +0 -89
- package/src/app/api/hazo_auth/me/route.ts +0 -47
- package/src/app/api/hazo_auth/profile_picture/[filename]/route.ts +0 -67
- package/src/app/api/hazo_auth/register/route.ts +0 -109
- package/src/app/api/hazo_auth/remove_profile_picture/route.ts +0 -86
- package/src/app/api/hazo_auth/resend_verification/route.ts +0 -108
- package/src/app/api/hazo_auth/reset_password/route.ts +0 -107
- package/src/app/api/hazo_auth/update_user/route.ts +0 -126
- package/src/app/api/hazo_auth/upload_profile_picture/route.ts +0 -268
- package/src/app/api/hazo_auth/user_management/permissions/route.ts +0 -367
- package/src/app/api/hazo_auth/user_management/roles/route.ts +0 -442
- package/src/app/api/hazo_auth/user_management/users/roles/route.ts +0 -367
- package/src/app/api/hazo_auth/user_management/users/route.ts +0 -239
- package/src/app/api/hazo_auth/validate_reset_token/route.ts +0 -83
- package/src/app/api/hazo_auth/verify_email/route.ts +0 -88
- package/src/app/api/migrations/apply/route.ts +0 -91
- package/src/app/favicon.ico +0 -0
- package/src/app/fonts/GeistMonoVF.woff +0 -0
- package/src/app/fonts/GeistVF.woff +0 -0
- package/src/app/globals.css +0 -89
- package/src/app/hazo_auth/forgot_password/forgot_password_page_client.tsx +0 -60
- package/src/app/hazo_auth/forgot_password/page.tsx +0 -24
- package/src/app/hazo_auth/login/login_page_client.tsx +0 -86
- package/src/app/hazo_auth/login/page.tsx +0 -38
- package/src/app/hazo_auth/my_settings/my_settings_page_client.tsx +0 -120
- package/src/app/hazo_auth/my_settings/page.tsx +0 -40
- package/src/app/hazo_auth/register/page.tsx +0 -36
- package/src/app/hazo_auth/register/register_page_client.tsx +0 -81
- package/src/app/hazo_auth/reset_password/page.tsx +0 -29
- package/src/app/hazo_auth/reset_password/reset_password_page_client.tsx +0 -81
- package/src/app/hazo_auth/user_management/page.tsx +0 -14
- package/src/app/hazo_auth/user_management/user_management_page_client.tsx +0 -16
- package/src/app/hazo_auth/verify_email/page.tsx +0 -24
- package/src/app/hazo_auth/verify_email/verify_email_page_client.tsx +0 -60
- package/src/app/hazo_connect/api/sqlite/data/route.ts +0 -203
- package/src/app/hazo_connect/api/sqlite/schema/route.ts +0 -45
- package/src/app/hazo_connect/api/sqlite/tables/route.ts +0 -36
- package/src/app/hazo_connect/sqlite_admin/page.tsx +0 -51
- package/src/app/hazo_connect/sqlite_admin/sqlite-admin-client.tsx +0 -984
- package/src/app/layout.tsx +0 -43
- package/src/app/page.tsx +0 -170
- package/src/components/index.ts +0 -7
- package/src/components/layouts/email_verification/config/email_verification_field_config.ts +0 -86
- package/src/components/layouts/email_verification/hooks/use_email_verification.ts +0 -297
- package/src/components/layouts/email_verification/index.tsx +0 -297
- package/src/components/layouts/forgot_password/config/forgot_password_field_config.ts +0 -58
- package/src/components/layouts/forgot_password/hooks/use_forgot_password_form.ts +0 -179
- package/src/components/layouts/forgot_password/index.tsx +0 -168
- package/src/components/layouts/index.ts +0 -26
- package/src/components/layouts/login/config/login_field_config.ts +0 -67
- package/src/components/layouts/login/hooks/use_login_form.ts +0 -286
- package/src/components/layouts/login/index.tsx +0 -252
- package/src/components/layouts/my_settings/components/editable_field.tsx +0 -177
- package/src/components/layouts/my_settings/components/password_change_dialog.tsx +0 -301
- package/src/components/layouts/my_settings/components/profile_picture_dialog.tsx +0 -385
- package/src/components/layouts/my_settings/components/profile_picture_display.tsx +0 -66
- package/src/components/layouts/my_settings/components/profile_picture_gravatar_tab.tsx +0 -143
- package/src/components/layouts/my_settings/components/profile_picture_library_tab.tsx +0 -311
- package/src/components/layouts/my_settings/components/profile_picture_upload_tab.tsx +0 -341
- package/src/components/layouts/my_settings/config/my_settings_field_config.ts +0 -61
- package/src/components/layouts/my_settings/hooks/use_my_settings.ts +0 -458
- package/src/components/layouts/my_settings/index.tsx +0 -351
- package/src/components/layouts/register/config/register_field_config.ts +0 -101
- package/src/components/layouts/register/hooks/use_register_form.ts +0 -275
- package/src/components/layouts/register/index.tsx +0 -226
- package/src/components/layouts/reset_password/config/reset_password_field_config.ts +0 -86
- package/src/components/layouts/reset_password/hooks/use_reset_password_form.ts +0 -276
- package/src/components/layouts/reset_password/index.tsx +0 -294
- package/src/components/layouts/shared/components/already_logged_in_guard.tsx +0 -95
- package/src/components/layouts/shared/components/auth_page_shell.tsx +0 -36
- package/src/components/layouts/shared/components/field_error_message.tsx +0 -29
- package/src/components/layouts/shared/components/form_action_buttons.tsx +0 -64
- package/src/components/layouts/shared/components/form_field_wrapper.tsx +0 -44
- package/src/components/layouts/shared/components/form_header.tsx +0 -36
- package/src/components/layouts/shared/components/logout_button.tsx +0 -76
- package/src/components/layouts/shared/components/password_field.tsx +0 -72
- package/src/components/layouts/shared/components/profile_pic_menu.tsx +0 -321
- package/src/components/layouts/shared/components/profile_pic_menu_wrapper.tsx +0 -40
- package/src/components/layouts/shared/components/sidebar_layout_wrapper.tsx +0 -214
- package/src/components/layouts/shared/components/standalone_layout_wrapper.tsx +0 -53
- package/src/components/layouts/shared/components/two_column_auth_layout.tsx +0 -44
- package/src/components/layouts/shared/components/unauthorized_guard.tsx +0 -78
- package/src/components/layouts/shared/components/visual_panel.tsx +0 -41
- package/src/components/layouts/shared/config/layout_customization.ts +0 -95
- package/src/components/layouts/shared/data/layout_data_client.ts +0 -19
- package/src/components/layouts/shared/hooks/use_auth_status.ts +0 -103
- package/src/components/layouts/shared/hooks/use_hazo_auth.ts +0 -158
- package/src/components/layouts/shared/index.ts +0 -34
- package/src/components/layouts/shared/utils/ip_address.ts +0 -37
- package/src/components/layouts/shared/utils/validation.ts +0 -66
- package/src/components/layouts/user_management/components/roles_matrix.tsx +0 -607
- package/src/components/layouts/user_management/index.tsx +0 -1295
- package/src/components/ui/alert-dialog.tsx +0 -141
- package/src/components/ui/avatar.tsx +0 -50
- package/src/components/ui/button.tsx +0 -57
- package/src/components/ui/checkbox.tsx +0 -30
- package/src/components/ui/dialog.tsx +0 -122
- package/src/components/ui/dropdown-menu.tsx +0 -201
- package/src/components/ui/hazo_ui_tooltip.tsx +0 -67
- package/src/components/ui/index.ts +0 -22
- package/src/components/ui/input.tsx +0 -22
- package/src/components/ui/label.tsx +0 -26
- package/src/components/ui/separator.tsx +0 -31
- package/src/components/ui/sheet.tsx +0 -139
- package/src/components/ui/sidebar.tsx +0 -773
- package/src/components/ui/skeleton.tsx +0 -15
- package/src/components/ui/sonner.tsx +0 -31
- package/src/components/ui/switch.tsx +0 -29
- package/src/components/ui/table.tsx +0 -120
- package/src/components/ui/tabs.tsx +0 -55
- package/src/components/ui/tooltip.tsx +0 -32
- package/src/components/ui/vertical-tabs.tsx +0 -59
- package/src/hooks/use-mobile.tsx +0 -19
- package/src/index.ts +0 -7
- package/src/lib/already_logged_in_config.server.ts +0 -46
- package/src/lib/app_logger.ts +0 -24
- package/src/lib/auth/auth_cache.ts +0 -220
- package/src/lib/auth/auth_rate_limiter.ts +0 -121
- package/src/lib/auth/auth_types.ts +0 -65
- package/src/lib/auth/auth_utils.server.ts +0 -196
- package/src/lib/auth/hazo_get_auth.server.ts +0 -333
- package/src/lib/auth/index.ts +0 -23
- package/src/lib/auth/server_auth.ts +0 -88
- package/src/lib/auth_utility_config.server.ts +0 -136
- package/src/lib/config/config_loader.server.ts +0 -164
- package/src/lib/email_verification_config.server.ts +0 -32
- package/src/lib/file_types_config.server.ts +0 -25
- package/src/lib/forgot_password_config.server.ts +0 -32
- package/src/lib/hazo_connect_instance.server.ts +0 -101
- package/src/lib/hazo_connect_setup.server.ts +0 -194
- package/src/lib/hazo_connect_setup.ts +0 -54
- package/src/lib/index.ts +0 -44
- package/src/lib/login_config.server.ts +0 -71
- package/src/lib/messages_config.server.ts +0 -45
- package/src/lib/migrations/apply_migration.ts +0 -105
- package/src/lib/my_settings_config.server.ts +0 -135
- package/src/lib/password_requirements_config.server.ts +0 -39
- package/src/lib/profile_pic_menu_config.server.ts +0 -138
- package/src/lib/profile_picture_config.server.ts +0 -56
- package/src/lib/register_config.server.ts +0 -73
- package/src/lib/reset_password_config.server.ts +0 -75
- package/src/lib/services/email_service.ts +0 -581
- package/src/lib/services/email_verification_service.ts +0 -270
- package/src/lib/services/index.ts +0 -15
- package/src/lib/services/login_service.ts +0 -134
- package/src/lib/services/password_change_service.ts +0 -154
- package/src/lib/services/password_reset_service.ts +0 -405
- package/src/lib/services/profile_picture_remove_service.ts +0 -120
- package/src/lib/services/profile_picture_service.ts +0 -215
- package/src/lib/services/profile_picture_source_mapper.ts +0 -62
- package/src/lib/services/registration_service.ts +0 -184
- package/src/lib/services/token_service.ts +0 -240
- package/src/lib/services/user_profiles_service.ts +0 -143
- package/src/lib/services/user_update_service.ts +0 -141
- package/src/lib/ui_shell_config.server.ts +0 -73
- package/src/lib/ui_sizes_config.server.ts +0 -37
- package/src/lib/user_fields_config.server.ts +0 -31
- package/src/lib/user_management_config.server.ts +0 -39
- package/src/lib/utils/api_route_helpers.ts +0 -60
- package/src/lib/utils/error_sanitizer.ts +0 -75
- package/src/lib/utils.ts +0 -11
- package/src/middleware.ts +0 -94
- package/src/routes/index.ts +0 -34
- package/src/server/config/config_loader.ts +0 -496
- package/src/server/index.ts +0 -38
- package/src/server/logging/logger_service.ts +0 -56
- package/src/server/routes/root_router.ts +0 -16
- package/src/server/server.ts +0 -28
- package/src/server/types/app_types.ts +0 -74
- package/src/server/types/express.d.ts +0 -16
- package/src/stories/email_verification_layout.stories.tsx +0 -137
- package/src/stories/forgot_password_layout.stories.tsx +0 -85
- package/src/stories/login_layout.stories.tsx +0 -85
- package/src/stories/project_overview.stories.tsx +0 -33
- package/src/stories/register_layout.stories.tsx +0 -107
- package/tailwind.config.ts +0 -77
- package/tsconfig.build.json +0 -36
- package/tsconfig.json +0 -28
|
@@ -1,61 +0,0 @@
|
|
|
1
|
-
// file_description: my settings layout specific configuration helpers
|
|
2
|
-
// section: imports
|
|
3
|
-
import type { LayoutFieldMap } from "../../shared/config/layout_customization";
|
|
4
|
-
import {
|
|
5
|
-
resolveButtonPalette,
|
|
6
|
-
type ButtonPaletteDefaults,
|
|
7
|
-
type ButtonPaletteOverrides,
|
|
8
|
-
type PasswordRequirementOptions,
|
|
9
|
-
} from "../../shared/config/layout_customization";
|
|
10
|
-
|
|
11
|
-
// section: field_identifiers
|
|
12
|
-
export const MY_SETTINGS_FIELD_IDS = {
|
|
13
|
-
NAME: "name",
|
|
14
|
-
EMAIL: "email_address",
|
|
15
|
-
PASSWORD: "password",
|
|
16
|
-
} as const;
|
|
17
|
-
|
|
18
|
-
export type MySettingsFieldId = (typeof MY_SETTINGS_FIELD_IDS)[keyof typeof MY_SETTINGS_FIELD_IDS];
|
|
19
|
-
|
|
20
|
-
// section: label_defaults
|
|
21
|
-
export type MySettingsLabelDefaults = {
|
|
22
|
-
heading: string;
|
|
23
|
-
profileTab: string;
|
|
24
|
-
securityTab: string;
|
|
25
|
-
lastLoggedInLabel: string;
|
|
26
|
-
profilePictureLabel: string;
|
|
27
|
-
changePasswordButton: string;
|
|
28
|
-
};
|
|
29
|
-
|
|
30
|
-
const MY_SETTINGS_LABEL_DEFAULTS: MySettingsLabelDefaults = {
|
|
31
|
-
heading: "My Settings",
|
|
32
|
-
profileTab: "Profile",
|
|
33
|
-
securityTab: "Security",
|
|
34
|
-
lastLoggedInLabel: "Last logged in",
|
|
35
|
-
profilePictureLabel: "Profile Picture",
|
|
36
|
-
changePasswordButton: "Change Password",
|
|
37
|
-
};
|
|
38
|
-
|
|
39
|
-
export type MySettingsLabelOverrides = Partial<MySettingsLabelDefaults>;
|
|
40
|
-
|
|
41
|
-
export const resolveMySettingsLabels = (
|
|
42
|
-
overrides?: MySettingsLabelOverrides,
|
|
43
|
-
): MySettingsLabelDefaults => {
|
|
44
|
-
return {
|
|
45
|
-
...MY_SETTINGS_LABEL_DEFAULTS,
|
|
46
|
-
...overrides,
|
|
47
|
-
};
|
|
48
|
-
};
|
|
49
|
-
|
|
50
|
-
// section: button_palette_defaults
|
|
51
|
-
const MY_SETTINGS_BUTTON_PALETTE_DEFAULTS: ButtonPaletteDefaults = {
|
|
52
|
-
submitBackground: "#0f172a",
|
|
53
|
-
submitText: "#ffffff",
|
|
54
|
-
cancelBorder: "#cbd5f5",
|
|
55
|
-
cancelText: "#0f172a",
|
|
56
|
-
};
|
|
57
|
-
|
|
58
|
-
export const resolveMySettingsButtonPalette = (
|
|
59
|
-
overrides?: ButtonPaletteOverrides,
|
|
60
|
-
) => resolveButtonPalette(MY_SETTINGS_BUTTON_PALETTE_DEFAULTS, overrides);
|
|
61
|
-
|
|
@@ -1,458 +0,0 @@
|
|
|
1
|
-
// file_description: hook for managing my settings state and API calls
|
|
2
|
-
// section: client_directive
|
|
3
|
-
"use client";
|
|
4
|
-
|
|
5
|
-
// section: imports
|
|
6
|
-
import { useState, useEffect, useCallback } from "react";
|
|
7
|
-
import { use_auth_status, trigger_auth_status_refresh } from "../../shared/hooks/use_auth_status";
|
|
8
|
-
import { toast } from "sonner";
|
|
9
|
-
import type { PasswordRequirementOptions } from "../../shared/config/layout_customization";
|
|
10
|
-
|
|
11
|
-
// section: types
|
|
12
|
-
export type PasswordFields = {
|
|
13
|
-
currentPassword: string;
|
|
14
|
-
newPassword: string;
|
|
15
|
-
confirmPassword: string;
|
|
16
|
-
currentPasswordVisible: boolean;
|
|
17
|
-
newPasswordVisible: boolean;
|
|
18
|
-
confirmPasswordVisible: boolean;
|
|
19
|
-
errors: {
|
|
20
|
-
currentPassword?: string;
|
|
21
|
-
newPassword?: string | string[];
|
|
22
|
-
confirmPassword?: string;
|
|
23
|
-
};
|
|
24
|
-
};
|
|
25
|
-
|
|
26
|
-
export type UseMySettingsResult = {
|
|
27
|
-
// User data
|
|
28
|
-
name: string;
|
|
29
|
-
email: string;
|
|
30
|
-
profilePictureUrl?: string;
|
|
31
|
-
profileSource?: "upload" | "library" | "gravatar" | "custom";
|
|
32
|
-
lastLogon?: string;
|
|
33
|
-
loading: boolean;
|
|
34
|
-
|
|
35
|
-
// Password fields
|
|
36
|
-
passwordFields?: PasswordFields;
|
|
37
|
-
handlePasswordFieldChange: (field: "currentPassword" | "newPassword" | "confirmPassword", value: string) => void;
|
|
38
|
-
togglePasswordVisibility: (field: "currentPassword" | "newPassword" | "confirmPassword") => void;
|
|
39
|
-
handlePasswordSave: () => Promise<void>;
|
|
40
|
-
isPasswordSaveDisabled: boolean;
|
|
41
|
-
|
|
42
|
-
// Profile picture
|
|
43
|
-
profilePictureDialogOpen: boolean;
|
|
44
|
-
handleProfilePictureEdit: () => void;
|
|
45
|
-
handleProfilePictureDialogClose: () => void;
|
|
46
|
-
handleProfilePictureSave: (profilePictureUrl: string, profileSource: "upload" | "library" | "gravatar") => Promise<void>;
|
|
47
|
-
handleProfilePictureRemove: () => Promise<void>;
|
|
48
|
-
|
|
49
|
-
// Actions
|
|
50
|
-
handleNameSave: (value: string) => Promise<void>;
|
|
51
|
-
handleEmailSave: (value: string) => Promise<void>;
|
|
52
|
-
|
|
53
|
-
// Refresh
|
|
54
|
-
refreshUserData: () => Promise<void>;
|
|
55
|
-
};
|
|
56
|
-
|
|
57
|
-
export type UseMySettingsParams = {
|
|
58
|
-
passwordRequirements: PasswordRequirementOptions;
|
|
59
|
-
};
|
|
60
|
-
|
|
61
|
-
// section: helpers
|
|
62
|
-
/**
|
|
63
|
-
* Validates email format
|
|
64
|
-
*/
|
|
65
|
-
const validateEmail = (email: string): string | null => {
|
|
66
|
-
if (!email || email.trim() === "") {
|
|
67
|
-
return "Email is required";
|
|
68
|
-
}
|
|
69
|
-
const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
|
|
70
|
-
if (!emailRegex.test(email)) {
|
|
71
|
-
return "Invalid email address format";
|
|
72
|
-
}
|
|
73
|
-
return null;
|
|
74
|
-
};
|
|
75
|
-
|
|
76
|
-
/**
|
|
77
|
-
* Validates name (optional, but if provided should not be empty)
|
|
78
|
-
*/
|
|
79
|
-
const validateName = (name: string): string | null => {
|
|
80
|
-
if (name.trim() === "") {
|
|
81
|
-
return "Name cannot be empty";
|
|
82
|
-
}
|
|
83
|
-
return null;
|
|
84
|
-
};
|
|
85
|
-
|
|
86
|
-
// section: hook
|
|
87
|
-
/**
|
|
88
|
-
* Hook for managing my settings state and API calls
|
|
89
|
-
* Handles user data loading, field editing, and API calls for updates
|
|
90
|
-
* @param params - Hook parameters including password requirements
|
|
91
|
-
* @returns My settings hook result with state and actions
|
|
92
|
-
*/
|
|
93
|
-
export function use_my_settings({
|
|
94
|
-
passwordRequirements,
|
|
95
|
-
}: UseMySettingsParams): UseMySettingsResult {
|
|
96
|
-
const authStatus = use_auth_status();
|
|
97
|
-
|
|
98
|
-
// Password fields state
|
|
99
|
-
const [passwordFields, setPasswordFields] = useState<PasswordFields>({
|
|
100
|
-
currentPassword: "",
|
|
101
|
-
newPassword: "",
|
|
102
|
-
confirmPassword: "",
|
|
103
|
-
currentPasswordVisible: false,
|
|
104
|
-
newPasswordVisible: false,
|
|
105
|
-
confirmPasswordVisible: false,
|
|
106
|
-
errors: {},
|
|
107
|
-
});
|
|
108
|
-
|
|
109
|
-
// Get user data from auth status
|
|
110
|
-
const name = authStatus.name || "";
|
|
111
|
-
const email = authStatus.email || "";
|
|
112
|
-
const profilePictureUrl = authStatus.profile_picture_url;
|
|
113
|
-
const profileSource = authStatus.profile_source;
|
|
114
|
-
const lastLogon = authStatus.last_logon;
|
|
115
|
-
const loading = authStatus.loading;
|
|
116
|
-
|
|
117
|
-
/**
|
|
118
|
-
* Refreshes user data by triggering auth status refresh
|
|
119
|
-
*/
|
|
120
|
-
const refreshUserData = useCallback(async () => {
|
|
121
|
-
trigger_auth_status_refresh();
|
|
122
|
-
// Also call the refresh method directly
|
|
123
|
-
await authStatus.refresh();
|
|
124
|
-
}, [authStatus]);
|
|
125
|
-
|
|
126
|
-
/**
|
|
127
|
-
* Updates user name
|
|
128
|
-
*/
|
|
129
|
-
const handleNameSave = useCallback(async (value: string) => {
|
|
130
|
-
const validationError = validateName(value);
|
|
131
|
-
if (validationError) {
|
|
132
|
-
toast.error(validationError);
|
|
133
|
-
throw new Error(validationError);
|
|
134
|
-
}
|
|
135
|
-
|
|
136
|
-
try {
|
|
137
|
-
const response = await fetch("/api/hazo_auth/update_user", {
|
|
138
|
-
method: "PATCH",
|
|
139
|
-
headers: {
|
|
140
|
-
"Content-Type": "application/json",
|
|
141
|
-
},
|
|
142
|
-
credentials: "include",
|
|
143
|
-
body: JSON.stringify({ name: value }),
|
|
144
|
-
});
|
|
145
|
-
|
|
146
|
-
const data = await response.json();
|
|
147
|
-
|
|
148
|
-
if (!response.ok || !data.success) {
|
|
149
|
-
const errorMessage = data.error || "Failed to update name";
|
|
150
|
-
toast.error(errorMessage);
|
|
151
|
-
throw new Error(errorMessage);
|
|
152
|
-
}
|
|
153
|
-
|
|
154
|
-
toast.success("Name updated successfully");
|
|
155
|
-
await refreshUserData();
|
|
156
|
-
} catch (error) {
|
|
157
|
-
const errorMessage = error instanceof Error ? error.message : "Failed to update name";
|
|
158
|
-
toast.error(errorMessage);
|
|
159
|
-
throw error;
|
|
160
|
-
}
|
|
161
|
-
}, [refreshUserData]);
|
|
162
|
-
|
|
163
|
-
/**
|
|
164
|
-
* Updates user email
|
|
165
|
-
*/
|
|
166
|
-
const handleEmailSave = useCallback(async (value: string) => {
|
|
167
|
-
const validationError = validateEmail(value);
|
|
168
|
-
if (validationError) {
|
|
169
|
-
toast.error(validationError);
|
|
170
|
-
throw new Error(validationError);
|
|
171
|
-
}
|
|
172
|
-
|
|
173
|
-
try {
|
|
174
|
-
const response = await fetch("/api/hazo_auth/update_user", {
|
|
175
|
-
method: "PATCH",
|
|
176
|
-
headers: {
|
|
177
|
-
"Content-Type": "application/json",
|
|
178
|
-
},
|
|
179
|
-
credentials: "include",
|
|
180
|
-
body: JSON.stringify({ email: value }),
|
|
181
|
-
});
|
|
182
|
-
|
|
183
|
-
const data = await response.json();
|
|
184
|
-
|
|
185
|
-
if (!response.ok || !data.success) {
|
|
186
|
-
const errorMessage = data.error || "Failed to update email";
|
|
187
|
-
toast.error(errorMessage);
|
|
188
|
-
throw new Error(errorMessage);
|
|
189
|
-
}
|
|
190
|
-
|
|
191
|
-
if (data.email_changed) {
|
|
192
|
-
toast.success("Email updated successfully. Please verify your new email address.");
|
|
193
|
-
} else {
|
|
194
|
-
toast.success("Email updated successfully");
|
|
195
|
-
}
|
|
196
|
-
|
|
197
|
-
await refreshUserData();
|
|
198
|
-
} catch (error) {
|
|
199
|
-
const errorMessage = error instanceof Error ? error.message : "Failed to update email";
|
|
200
|
-
toast.error(errorMessage);
|
|
201
|
-
throw error;
|
|
202
|
-
}
|
|
203
|
-
}, [refreshUserData]);
|
|
204
|
-
|
|
205
|
-
/**
|
|
206
|
-
* Validates password requirements
|
|
207
|
-
*/
|
|
208
|
-
const validatePassword = useCallback((password: string): string | null => {
|
|
209
|
-
if (!password || password.length < passwordRequirements.minimum_length) {
|
|
210
|
-
return `Password must be at least ${passwordRequirements.minimum_length} characters long`;
|
|
211
|
-
}
|
|
212
|
-
|
|
213
|
-
const errors: string[] = [];
|
|
214
|
-
|
|
215
|
-
if (passwordRequirements.require_uppercase && !/[A-Z]/.test(password)) {
|
|
216
|
-
errors.push("uppercase letter");
|
|
217
|
-
}
|
|
218
|
-
|
|
219
|
-
if (passwordRequirements.require_lowercase && !/[a-z]/.test(password)) {
|
|
220
|
-
errors.push("lowercase letter");
|
|
221
|
-
}
|
|
222
|
-
|
|
223
|
-
if (passwordRequirements.require_number && !/[0-9]/.test(password)) {
|
|
224
|
-
errors.push("number");
|
|
225
|
-
}
|
|
226
|
-
|
|
227
|
-
if (passwordRequirements.require_special && !/[^A-Za-z0-9]/.test(password)) {
|
|
228
|
-
errors.push("special character");
|
|
229
|
-
}
|
|
230
|
-
|
|
231
|
-
if (errors.length > 0) {
|
|
232
|
-
return `Password must contain at least one: ${errors.join(", ")}`;
|
|
233
|
-
}
|
|
234
|
-
|
|
235
|
-
return null;
|
|
236
|
-
}, [passwordRequirements]);
|
|
237
|
-
|
|
238
|
-
/**
|
|
239
|
-
* Handles password field change
|
|
240
|
-
*/
|
|
241
|
-
const handlePasswordFieldChange = useCallback((field: "currentPassword" | "newPassword" | "confirmPassword", value: string) => {
|
|
242
|
-
setPasswordFields((prev) => {
|
|
243
|
-
const newFields = { ...prev, [field]: value };
|
|
244
|
-
// Clear errors for this field when user types
|
|
245
|
-
if (newFields.errors[field as keyof typeof newFields.errors]) {
|
|
246
|
-
newFields.errors = { ...newFields.errors, [field]: undefined };
|
|
247
|
-
}
|
|
248
|
-
return newFields;
|
|
249
|
-
});
|
|
250
|
-
}, []);
|
|
251
|
-
|
|
252
|
-
/**
|
|
253
|
-
* Toggles password visibility
|
|
254
|
-
*/
|
|
255
|
-
const togglePasswordVisibility = useCallback((field: "currentPassword" | "newPassword" | "confirmPassword") => {
|
|
256
|
-
setPasswordFields((prev) => {
|
|
257
|
-
const fieldKey = `${field}Visible` as keyof PasswordFields;
|
|
258
|
-
const currentValue = prev[fieldKey] as boolean;
|
|
259
|
-
return {
|
|
260
|
-
...prev,
|
|
261
|
-
[fieldKey]: !currentValue,
|
|
262
|
-
};
|
|
263
|
-
});
|
|
264
|
-
}, []);
|
|
265
|
-
|
|
266
|
-
/**
|
|
267
|
-
* Validates password form
|
|
268
|
-
*/
|
|
269
|
-
const validatePasswordForm = useCallback((): boolean => {
|
|
270
|
-
const errors: PasswordFields["errors"] = {};
|
|
271
|
-
|
|
272
|
-
if (!passwordFields.currentPassword) {
|
|
273
|
-
errors.currentPassword = "Current password is required";
|
|
274
|
-
}
|
|
275
|
-
|
|
276
|
-
const newPasswordError = validatePassword(passwordFields.newPassword);
|
|
277
|
-
if (newPasswordError) {
|
|
278
|
-
errors.newPassword = newPasswordError;
|
|
279
|
-
}
|
|
280
|
-
|
|
281
|
-
if (!passwordFields.confirmPassword) {
|
|
282
|
-
errors.confirmPassword = "Please confirm your new password";
|
|
283
|
-
} else if (passwordFields.newPassword !== passwordFields.confirmPassword) {
|
|
284
|
-
errors.confirmPassword = "Passwords do not match";
|
|
285
|
-
}
|
|
286
|
-
|
|
287
|
-
setPasswordFields((prev) => ({ ...prev, errors }));
|
|
288
|
-
return Object.keys(errors).length === 0;
|
|
289
|
-
}, [passwordFields, validatePassword]);
|
|
290
|
-
|
|
291
|
-
/**
|
|
292
|
-
* Checks if password save should be disabled
|
|
293
|
-
*/
|
|
294
|
-
const isPasswordSaveDisabled = useCallback((): boolean => {
|
|
295
|
-
return !passwordFields.currentPassword || !passwordFields.newPassword || !passwordFields.confirmPassword;
|
|
296
|
-
}, [passwordFields]);
|
|
297
|
-
|
|
298
|
-
/**
|
|
299
|
-
* Saves password changes
|
|
300
|
-
*/
|
|
301
|
-
const handlePasswordSave = useCallback(async () => {
|
|
302
|
-
if (!validatePasswordForm()) {
|
|
303
|
-
return;
|
|
304
|
-
}
|
|
305
|
-
|
|
306
|
-
try {
|
|
307
|
-
const response = await fetch("/api/hazo_auth/change_password", {
|
|
308
|
-
method: "POST",
|
|
309
|
-
headers: {
|
|
310
|
-
"Content-Type": "application/json",
|
|
311
|
-
},
|
|
312
|
-
credentials: "include",
|
|
313
|
-
body: JSON.stringify({
|
|
314
|
-
current_password: passwordFields.currentPassword,
|
|
315
|
-
new_password: passwordFields.newPassword,
|
|
316
|
-
}),
|
|
317
|
-
});
|
|
318
|
-
|
|
319
|
-
const data = await response.json();
|
|
320
|
-
|
|
321
|
-
if (!response.ok || !data.success) {
|
|
322
|
-
const errorMessage = data.error || "Failed to change password";
|
|
323
|
-
setPasswordFields((prev) => ({
|
|
324
|
-
...prev,
|
|
325
|
-
errors: { currentPassword: errorMessage },
|
|
326
|
-
}));
|
|
327
|
-
toast.error(errorMessage);
|
|
328
|
-
throw new Error(errorMessage);
|
|
329
|
-
}
|
|
330
|
-
|
|
331
|
-
toast.success("Password changed successfully");
|
|
332
|
-
// Reset password fields
|
|
333
|
-
setPasswordFields({
|
|
334
|
-
currentPassword: "",
|
|
335
|
-
newPassword: "",
|
|
336
|
-
confirmPassword: "",
|
|
337
|
-
currentPasswordVisible: false,
|
|
338
|
-
newPasswordVisible: false,
|
|
339
|
-
confirmPasswordVisible: false,
|
|
340
|
-
errors: {},
|
|
341
|
-
});
|
|
342
|
-
} catch (error) {
|
|
343
|
-
const errorMessage = error instanceof Error ? error.message : "Failed to change password";
|
|
344
|
-
toast.error(errorMessage);
|
|
345
|
-
throw error;
|
|
346
|
-
}
|
|
347
|
-
}, [passwordFields, validatePasswordForm]);
|
|
348
|
-
|
|
349
|
-
/**
|
|
350
|
-
* Profile picture dialog state
|
|
351
|
-
*/
|
|
352
|
-
const [profilePictureDialogOpen, setProfilePictureDialogOpen] = useState(false);
|
|
353
|
-
|
|
354
|
-
/**
|
|
355
|
-
* Opens profile picture dialog
|
|
356
|
-
*/
|
|
357
|
-
const handleProfilePictureEdit = useCallback(() => {
|
|
358
|
-
setProfilePictureDialogOpen(true);
|
|
359
|
-
}, []);
|
|
360
|
-
|
|
361
|
-
/**
|
|
362
|
-
* Closes profile picture dialog
|
|
363
|
-
*/
|
|
364
|
-
const handleProfilePictureDialogClose = useCallback(() => {
|
|
365
|
-
setProfilePictureDialogOpen(false);
|
|
366
|
-
}, []);
|
|
367
|
-
|
|
368
|
-
/**
|
|
369
|
-
* Saves profile picture changes
|
|
370
|
-
* Note: profilePictureUrl is already a fully-formed URL (Gravatar URL is generated in the dialog)
|
|
371
|
-
*/
|
|
372
|
-
const handleProfilePictureSave = useCallback(async (profilePictureUrl: string, profileSource: "upload" | "library" | "gravatar") => {
|
|
373
|
-
try {
|
|
374
|
-
const response = await fetch("/api/hazo_auth/update_user", {
|
|
375
|
-
method: "PATCH",
|
|
376
|
-
headers: {
|
|
377
|
-
"Content-Type": "application/json",
|
|
378
|
-
},
|
|
379
|
-
credentials: "include",
|
|
380
|
-
body: JSON.stringify({
|
|
381
|
-
profile_picture_url: profilePictureUrl,
|
|
382
|
-
profile_source: profileSource,
|
|
383
|
-
}),
|
|
384
|
-
});
|
|
385
|
-
|
|
386
|
-
const data = await response.json();
|
|
387
|
-
|
|
388
|
-
if (!response.ok || !data.success) {
|
|
389
|
-
const errorMessage = data.error || "Failed to update profile picture";
|
|
390
|
-
toast.error(errorMessage);
|
|
391
|
-
throw new Error(errorMessage);
|
|
392
|
-
}
|
|
393
|
-
|
|
394
|
-
toast.success("Profile picture updated successfully");
|
|
395
|
-
await refreshUserData();
|
|
396
|
-
setProfilePictureDialogOpen(false);
|
|
397
|
-
} catch (error) {
|
|
398
|
-
const errorMessage = error instanceof Error ? error.message : "Failed to update profile picture";
|
|
399
|
-
toast.error(errorMessage);
|
|
400
|
-
throw error;
|
|
401
|
-
}
|
|
402
|
-
}, [refreshUserData]);
|
|
403
|
-
|
|
404
|
-
/**
|
|
405
|
-
* Removes profile picture
|
|
406
|
-
* - If upload: deletes the file and clears profile_picture_url and profile_source
|
|
407
|
-
* - If gravatar/library: clears profile_picture_url and profile_source
|
|
408
|
-
*/
|
|
409
|
-
const handleProfilePictureRemove = useCallback(async () => {
|
|
410
|
-
try {
|
|
411
|
-
const response = await fetch("/api/hazo_auth/remove_profile_picture", {
|
|
412
|
-
method: "DELETE",
|
|
413
|
-
headers: {
|
|
414
|
-
"Content-Type": "application/json",
|
|
415
|
-
},
|
|
416
|
-
credentials: "include",
|
|
417
|
-
});
|
|
418
|
-
|
|
419
|
-
const data = await response.json();
|
|
420
|
-
|
|
421
|
-
if (!response.ok || !data.success) {
|
|
422
|
-
const errorMessage = data.error || "Failed to remove profile picture";
|
|
423
|
-
toast.error(errorMessage);
|
|
424
|
-
throw new Error(errorMessage);
|
|
425
|
-
}
|
|
426
|
-
|
|
427
|
-
toast.success("Profile picture removed successfully");
|
|
428
|
-
await refreshUserData();
|
|
429
|
-
} catch (error) {
|
|
430
|
-
const errorMessage = error instanceof Error ? error.message : "Failed to remove profile picture";
|
|
431
|
-
toast.error(errorMessage);
|
|
432
|
-
throw error;
|
|
433
|
-
}
|
|
434
|
-
}, [refreshUserData]);
|
|
435
|
-
|
|
436
|
-
return {
|
|
437
|
-
name,
|
|
438
|
-
email,
|
|
439
|
-
profilePictureUrl,
|
|
440
|
-
profileSource,
|
|
441
|
-
lastLogon,
|
|
442
|
-
loading,
|
|
443
|
-
passwordFields,
|
|
444
|
-
handlePasswordFieldChange,
|
|
445
|
-
togglePasswordVisibility,
|
|
446
|
-
handlePasswordSave,
|
|
447
|
-
isPasswordSaveDisabled: isPasswordSaveDisabled(),
|
|
448
|
-
profilePictureDialogOpen,
|
|
449
|
-
handleProfilePictureEdit,
|
|
450
|
-
handleProfilePictureDialogClose,
|
|
451
|
-
handleProfilePictureSave,
|
|
452
|
-
handleProfilePictureRemove,
|
|
453
|
-
handleNameSave,
|
|
454
|
-
handleEmailSave,
|
|
455
|
-
refreshUserData,
|
|
456
|
-
};
|
|
457
|
-
}
|
|
458
|
-
|