hazo_auth 1.4.2 → 1.6.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +65 -19
- package/SETUP_CHECKLIST.md +779 -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_photo/[category]/[filename]/route.d.ts +9 -0
- package/dist/app/api/hazo_auth/library_photo/[category]/[filename]/route.d.ts.map +1 -0
- package/dist/app/api/hazo_auth/library_photo/[category]/[filename]/route.js +82 -0
- package/dist/app/api/hazo_auth/library_photos/route.d.ts +22 -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 +80 -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 +7 -0
- package/dist/cli/generate.d.ts.map +1 -0
- package/dist/cli/generate.js +184 -0
- package/dist/cli/index.d.ts +3 -0
- package/dist/cli/index.d.ts.map +1 -0
- package/dist/cli/index.js +173 -0
- package/dist/cli/init.d.ts +2 -0
- package/dist/cli/init.d.ts.map +1 -0
- package/dist/cli/init.js +201 -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/lib/services/profile_picture_service.d.ts +34 -2
- package/dist/lib/services/profile_picture_service.d.ts.map +1 -1
- package/dist/lib/services/profile_picture_service.js +157 -15
- package/dist/page_components/forgot_password.d.ts +19 -0
- package/dist/page_components/forgot_password.d.ts.map +1 -0
- package/dist/page_components/forgot_password.js +36 -0
- package/dist/page_components/index.d.ts +7 -0
- package/dist/page_components/index.d.ts.map +1 -0
- package/dist/page_components/index.js +9 -0
- package/dist/page_components/login.d.ts +26 -0
- package/dist/page_components/login.d.ts.map +1 -0
- package/dist/page_components/login.js +40 -0
- package/dist/page_components/my_settings.d.ts +64 -0
- package/dist/page_components/my_settings.d.ts.map +1 -0
- package/dist/page_components/my_settings.js +67 -0
- package/dist/page_components/register.d.ts +25 -0
- package/dist/page_components/register.d.ts.map +1 -0
- package/dist/page_components/register.js +43 -0
- package/dist/page_components/reset_password.d.ts +25 -0
- package/dist/page_components/reset_password.d.ts.map +1 -0
- package/dist/page_components/reset_password.js +43 -0
- package/dist/page_components/verify_email.d.ts +21 -0
- package/dist/page_components/verify_email.d.ts.map +1 -0
- package/dist/page_components/verify_email.js +36 -0
- 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 +19 -0
- package/dist/server/routes/index.d.ts.map +1 -0
- package/dist/server/routes/index.js +25 -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_photo.d.ts +2 -0
- package/dist/server/routes/library_photo.d.ts.map +1 -0
- package/dist/server/routes/library_photo.js +3 -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 +40 -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,276 +0,0 @@
|
|
|
1
|
-
// file_description: encapsulate reset password form state, validation, and data interactions
|
|
2
|
-
// section: imports
|
|
3
|
-
import { useCallback, useMemo, useState, useEffect } from "react";
|
|
4
|
-
import { useSearchParams, useRouter } from "next/navigation";
|
|
5
|
-
import { toast } from "sonner";
|
|
6
|
-
import type { LayoutDataClient } from "../../shared/data/layout_data_client";
|
|
7
|
-
import type { PasswordRequirementOptions } from "../../shared/config/layout_customization";
|
|
8
|
-
import { RESET_PASSWORD_FIELD_IDS, type ResetPasswordFieldId } from "../config/reset_password_field_config";
|
|
9
|
-
import { validatePassword } from "../../shared/utils/validation";
|
|
10
|
-
|
|
11
|
-
// section: constants
|
|
12
|
-
const PASSWORD_FIELDS: Array<ResetPasswordFieldId> = [
|
|
13
|
-
RESET_PASSWORD_FIELD_IDS.PASSWORD,
|
|
14
|
-
RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD,
|
|
15
|
-
];
|
|
16
|
-
|
|
17
|
-
// section: types
|
|
18
|
-
export type ResetPasswordFormValues = Record<ResetPasswordFieldId, string>;
|
|
19
|
-
export type ResetPasswordFormErrors = Partial<Record<ResetPasswordFieldId, string | string[]>>;
|
|
20
|
-
export type PasswordVisibilityState = Record<
|
|
21
|
-
Extract<ResetPasswordFieldId, "password" | "confirm_password">,
|
|
22
|
-
boolean
|
|
23
|
-
>;
|
|
24
|
-
|
|
25
|
-
export type UseResetPasswordFormParams<TClient = unknown> = {
|
|
26
|
-
passwordRequirements: PasswordRequirementOptions;
|
|
27
|
-
dataClient: LayoutDataClient<TClient>;
|
|
28
|
-
loginPath?: string;
|
|
29
|
-
};
|
|
30
|
-
|
|
31
|
-
export type UseResetPasswordFormResult = {
|
|
32
|
-
values: ResetPasswordFormValues;
|
|
33
|
-
errors: ResetPasswordFormErrors;
|
|
34
|
-
passwordVisibility: PasswordVisibilityState;
|
|
35
|
-
isSubmitDisabled: boolean;
|
|
36
|
-
isSubmitting: boolean;
|
|
37
|
-
isSuccess: boolean;
|
|
38
|
-
token: string | null;
|
|
39
|
-
isValidatingToken: boolean;
|
|
40
|
-
tokenError: string | null;
|
|
41
|
-
handleFieldChange: (fieldId: ResetPasswordFieldId, value: string) => void;
|
|
42
|
-
togglePasswordVisibility: (fieldId: "password" | "confirm_password") => void;
|
|
43
|
-
handleSubmit: (event: React.FormEvent<HTMLFormElement>) => void;
|
|
44
|
-
handleCancel: () => void;
|
|
45
|
-
};
|
|
46
|
-
|
|
47
|
-
// section: helpers
|
|
48
|
-
const buildInitialValues = (): ResetPasswordFormValues => ({
|
|
49
|
-
[RESET_PASSWORD_FIELD_IDS.PASSWORD]: "",
|
|
50
|
-
[RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD]: "",
|
|
51
|
-
});
|
|
52
|
-
|
|
53
|
-
// section: hook
|
|
54
|
-
export const use_reset_password_form = <TClient,>({
|
|
55
|
-
passwordRequirements,
|
|
56
|
-
dataClient,
|
|
57
|
-
loginPath = "/hazo_auth/login",
|
|
58
|
-
}: UseResetPasswordFormParams<TClient>): UseResetPasswordFormResult => {
|
|
59
|
-
const router = useRouter();
|
|
60
|
-
const searchParams = useSearchParams();
|
|
61
|
-
const tokenParam = searchParams.get("token");
|
|
62
|
-
|
|
63
|
-
const [values, setValues] = useState<ResetPasswordFormValues>(buildInitialValues);
|
|
64
|
-
const [errors, setErrors] = useState<ResetPasswordFormErrors>({});
|
|
65
|
-
const [passwordVisibility, setPasswordVisibility] = useState<PasswordVisibilityState>({
|
|
66
|
-
password: false,
|
|
67
|
-
confirm_password: false,
|
|
68
|
-
});
|
|
69
|
-
const [isSubmitting, setIsSubmitting] = useState<boolean>(false);
|
|
70
|
-
const [isSuccess, setIsSuccess] = useState<boolean>(false);
|
|
71
|
-
const [token, setToken] = useState<string | null>(tokenParam);
|
|
72
|
-
const [isValidatingToken, setIsValidatingToken] = useState<boolean>(false);
|
|
73
|
-
const [tokenError, setTokenError] = useState<string | null>(null);
|
|
74
|
-
|
|
75
|
-
// Validate token on mount
|
|
76
|
-
useEffect(() => {
|
|
77
|
-
if (!tokenParam) {
|
|
78
|
-
setTokenError("Reset password link invalid or has expired. Please go to Reset Password page to get a new link.");
|
|
79
|
-
return;
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
// Validate token by calling validation API
|
|
83
|
-
const validateToken = async () => {
|
|
84
|
-
setIsValidatingToken(true);
|
|
85
|
-
setTokenError(null);
|
|
86
|
-
|
|
87
|
-
try {
|
|
88
|
-
const response = await fetch(`/api/hazo_auth/validate_reset_token?token=${encodeURIComponent(tokenParam)}`, {
|
|
89
|
-
method: "GET",
|
|
90
|
-
});
|
|
91
|
-
|
|
92
|
-
const data = await response.json();
|
|
93
|
-
|
|
94
|
-
if (!response.ok || !data.success) {
|
|
95
|
-
const errorMessage = data.error || "Reset password link invalid or has expired. Please go to Reset Password page to get a new link.";
|
|
96
|
-
setTokenError(errorMessage);
|
|
97
|
-
setToken(null);
|
|
98
|
-
} else {
|
|
99
|
-
// Token is valid
|
|
100
|
-
setToken(tokenParam);
|
|
101
|
-
setTokenError(null);
|
|
102
|
-
}
|
|
103
|
-
} catch (error) {
|
|
104
|
-
const errorMessage = error instanceof Error ? error.message : "An error occurred while validating the token";
|
|
105
|
-
setTokenError("Reset password link invalid or has expired. Please go to Reset Password page to get a new link.");
|
|
106
|
-
setToken(null);
|
|
107
|
-
} finally {
|
|
108
|
-
setIsValidatingToken(false);
|
|
109
|
-
}
|
|
110
|
-
};
|
|
111
|
-
|
|
112
|
-
void validateToken();
|
|
113
|
-
}, [tokenParam]);
|
|
114
|
-
|
|
115
|
-
const isSubmitDisabled = useMemo(() => {
|
|
116
|
-
if (isSubmitting || isSuccess || !token || tokenError) {
|
|
117
|
-
return true;
|
|
118
|
-
}
|
|
119
|
-
|
|
120
|
-
const hasEmptyField = Object.values(values).some((value) => value.trim() === "");
|
|
121
|
-
if (hasEmptyField) {
|
|
122
|
-
return true;
|
|
123
|
-
}
|
|
124
|
-
|
|
125
|
-
return Object.keys(errors).length > 0;
|
|
126
|
-
}, [isSubmitting, isSuccess, token, tokenError, values, errors]);
|
|
127
|
-
|
|
128
|
-
const handleFieldChange = useCallback((fieldId: ResetPasswordFieldId, value: string) => {
|
|
129
|
-
setValues((prev) => ({ ...prev, [fieldId]: value }));
|
|
130
|
-
|
|
131
|
-
// Clear error for this field when user starts typing
|
|
132
|
-
if (errors[fieldId]) {
|
|
133
|
-
setErrors((prev) => {
|
|
134
|
-
const next = { ...prev };
|
|
135
|
-
delete next[fieldId];
|
|
136
|
-
return next;
|
|
137
|
-
});
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
// Validate password fields in real-time
|
|
141
|
-
if (fieldId === RESET_PASSWORD_FIELD_IDS.PASSWORD) {
|
|
142
|
-
const passwordError = validatePassword(value, passwordRequirements);
|
|
143
|
-
if (passwordError) {
|
|
144
|
-
setErrors((prev) => ({
|
|
145
|
-
...prev,
|
|
146
|
-
[RESET_PASSWORD_FIELD_IDS.PASSWORD]: passwordError,
|
|
147
|
-
}));
|
|
148
|
-
} else {
|
|
149
|
-
setErrors((prev) => {
|
|
150
|
-
const next = { ...prev };
|
|
151
|
-
delete next[RESET_PASSWORD_FIELD_IDS.PASSWORD];
|
|
152
|
-
return next;
|
|
153
|
-
});
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
// Also validate confirm password if it has a value
|
|
157
|
-
if (values[RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD]) {
|
|
158
|
-
if (value !== values[RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD]) {
|
|
159
|
-
setErrors((prev) => ({
|
|
160
|
-
...prev,
|
|
161
|
-
[RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD]: "Passwords do not match",
|
|
162
|
-
}));
|
|
163
|
-
} else {
|
|
164
|
-
setErrors((prev) => {
|
|
165
|
-
const next = { ...prev };
|
|
166
|
-
delete next[RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD];
|
|
167
|
-
return next;
|
|
168
|
-
});
|
|
169
|
-
}
|
|
170
|
-
}
|
|
171
|
-
} else if (fieldId === RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD) {
|
|
172
|
-
if (value !== values[RESET_PASSWORD_FIELD_IDS.PASSWORD]) {
|
|
173
|
-
setErrors((prev) => ({
|
|
174
|
-
...prev,
|
|
175
|
-
[RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD]: "Passwords do not match",
|
|
176
|
-
}));
|
|
177
|
-
} else {
|
|
178
|
-
setErrors((prev) => {
|
|
179
|
-
const next = { ...prev };
|
|
180
|
-
delete next[RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD];
|
|
181
|
-
return next;
|
|
182
|
-
});
|
|
183
|
-
}
|
|
184
|
-
}
|
|
185
|
-
}, [errors, passwordRequirements, values]);
|
|
186
|
-
|
|
187
|
-
const togglePasswordVisibility = useCallback((fieldId: "password" | "confirm_password") => {
|
|
188
|
-
setPasswordVisibility((prev) => ({
|
|
189
|
-
...prev,
|
|
190
|
-
[fieldId]: !prev[fieldId],
|
|
191
|
-
}));
|
|
192
|
-
}, []);
|
|
193
|
-
|
|
194
|
-
const handleSubmit = useCallback(async (event: React.FormEvent<HTMLFormElement>) => {
|
|
195
|
-
event.preventDefault();
|
|
196
|
-
|
|
197
|
-
if (!token) {
|
|
198
|
-
toast.error("Reset token is missing");
|
|
199
|
-
return;
|
|
200
|
-
}
|
|
201
|
-
|
|
202
|
-
// Validate all fields
|
|
203
|
-
const passwordError = validatePassword(values[RESET_PASSWORD_FIELD_IDS.PASSWORD], passwordRequirements);
|
|
204
|
-
const confirmPasswordError =
|
|
205
|
-
values[RESET_PASSWORD_FIELD_IDS.PASSWORD] !== values[RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD]
|
|
206
|
-
? "Passwords do not match"
|
|
207
|
-
: undefined;
|
|
208
|
-
|
|
209
|
-
if (passwordError || confirmPasswordError) {
|
|
210
|
-
setErrors({
|
|
211
|
-
...(passwordError ? { [RESET_PASSWORD_FIELD_IDS.PASSWORD]: passwordError } : {}),
|
|
212
|
-
...(confirmPasswordError ? { [RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD]: confirmPasswordError } : {}),
|
|
213
|
-
});
|
|
214
|
-
return;
|
|
215
|
-
}
|
|
216
|
-
|
|
217
|
-
setIsSubmitting(true);
|
|
218
|
-
|
|
219
|
-
try {
|
|
220
|
-
const response = await fetch("/api/hazo_auth/reset_password", {
|
|
221
|
-
method: "POST",
|
|
222
|
-
headers: {
|
|
223
|
-
"Content-Type": "application/json",
|
|
224
|
-
},
|
|
225
|
-
body: JSON.stringify({
|
|
226
|
-
token,
|
|
227
|
-
new_password: values[RESET_PASSWORD_FIELD_IDS.PASSWORD],
|
|
228
|
-
}),
|
|
229
|
-
});
|
|
230
|
-
|
|
231
|
-
const data = await response.json();
|
|
232
|
-
|
|
233
|
-
if (!response.ok || !data.success) {
|
|
234
|
-
const errorMessage = data.error || "Failed to reset password";
|
|
235
|
-
toast.error(errorMessage);
|
|
236
|
-
setTokenError(errorMessage);
|
|
237
|
-
setIsSubmitting(false);
|
|
238
|
-
return;
|
|
239
|
-
}
|
|
240
|
-
|
|
241
|
-
toast.success(data.message || "Password reset successfully");
|
|
242
|
-
setIsSuccess(true);
|
|
243
|
-
|
|
244
|
-
// Redirect to login after a short delay
|
|
245
|
-
setTimeout(() => {
|
|
246
|
-
router.push(loginPath);
|
|
247
|
-
}, 2000);
|
|
248
|
-
} catch (error) {
|
|
249
|
-
const errorMessage = error instanceof Error ? error.message : "An error occurred";
|
|
250
|
-
toast.error(errorMessage);
|
|
251
|
-
setTokenError(errorMessage);
|
|
252
|
-
setIsSubmitting(false);
|
|
253
|
-
}
|
|
254
|
-
}, [token, values, passwordRequirements, router, loginPath]);
|
|
255
|
-
|
|
256
|
-
const handleCancel = useCallback(() => {
|
|
257
|
-
router.push(loginPath);
|
|
258
|
-
}, [router, loginPath]);
|
|
259
|
-
|
|
260
|
-
return {
|
|
261
|
-
values,
|
|
262
|
-
errors,
|
|
263
|
-
passwordVisibility,
|
|
264
|
-
isSubmitDisabled,
|
|
265
|
-
isSubmitting,
|
|
266
|
-
isSuccess,
|
|
267
|
-
token,
|
|
268
|
-
isValidatingToken,
|
|
269
|
-
tokenError,
|
|
270
|
-
handleFieldChange,
|
|
271
|
-
togglePasswordVisibility,
|
|
272
|
-
handleSubmit,
|
|
273
|
-
handleCancel,
|
|
274
|
-
};
|
|
275
|
-
};
|
|
276
|
-
|
|
@@ -1,294 +0,0 @@
|
|
|
1
|
-
// file_description: reset password layout component built atop shared layout utilities
|
|
2
|
-
// section: client_directive
|
|
3
|
-
"use client";
|
|
4
|
-
|
|
5
|
-
// section: imports
|
|
6
|
-
import { PasswordField } from "../shared/components/password_field";
|
|
7
|
-
import { FormFieldWrapper } from "../shared/components/form_field_wrapper";
|
|
8
|
-
import { FormHeader } from "../shared/components/form_header";
|
|
9
|
-
import { FormActionButtons } from "../shared/components/form_action_buttons";
|
|
10
|
-
import { TwoColumnAuthLayout } from "../shared/components/two_column_auth_layout";
|
|
11
|
-
import { CheckCircle, XCircle, Loader2 } from "lucide-react";
|
|
12
|
-
import { AlreadyLoggedInGuard } from "../shared/components/already_logged_in_guard";
|
|
13
|
-
import {
|
|
14
|
-
type ButtonPaletteOverrides,
|
|
15
|
-
type LayoutFieldMapOverrides,
|
|
16
|
-
type LayoutLabelOverrides,
|
|
17
|
-
type PasswordRequirementOverrides,
|
|
18
|
-
} from "../shared/config/layout_customization";
|
|
19
|
-
import {
|
|
20
|
-
RESET_PASSWORD_FIELD_IDS,
|
|
21
|
-
createResetPasswordFieldDefinitions,
|
|
22
|
-
resolveResetPasswordButtonPalette,
|
|
23
|
-
resolveResetPasswordLabels,
|
|
24
|
-
resolveResetPasswordPasswordRequirements,
|
|
25
|
-
} from "./config/reset_password_field_config";
|
|
26
|
-
import {
|
|
27
|
-
use_reset_password_form,
|
|
28
|
-
type UseResetPasswordFormResult,
|
|
29
|
-
} from "./hooks/use_reset_password_form";
|
|
30
|
-
import { type LayoutDataClient } from "../shared/data/layout_data_client";
|
|
31
|
-
import Link from "next/link";
|
|
32
|
-
|
|
33
|
-
// section: types
|
|
34
|
-
export type ResetPasswordLayoutProps<TClient = unknown> = {
|
|
35
|
-
image_src: string;
|
|
36
|
-
image_alt: string;
|
|
37
|
-
image_background_color?: string;
|
|
38
|
-
field_overrides?: LayoutFieldMapOverrides;
|
|
39
|
-
labels?: LayoutLabelOverrides;
|
|
40
|
-
button_colors?: ButtonPaletteOverrides;
|
|
41
|
-
password_requirements?: PasswordRequirementOverrides;
|
|
42
|
-
data_client: LayoutDataClient<TClient>;
|
|
43
|
-
alreadyLoggedInMessage?: string;
|
|
44
|
-
showLogoutButton?: boolean;
|
|
45
|
-
showReturnHomeButton?: boolean;
|
|
46
|
-
returnHomeButtonLabel?: string;
|
|
47
|
-
returnHomePath?: string;
|
|
48
|
-
errorMessage?: string;
|
|
49
|
-
successMessage?: string;
|
|
50
|
-
loginPath?: string;
|
|
51
|
-
forgotPasswordPath?: string;
|
|
52
|
-
};
|
|
53
|
-
|
|
54
|
-
const ORDERED_FIELDS: ResetPasswordFieldId[] = [
|
|
55
|
-
RESET_PASSWORD_FIELD_IDS.PASSWORD,
|
|
56
|
-
RESET_PASSWORD_FIELD_IDS.CONFIRM_PASSWORD,
|
|
57
|
-
];
|
|
58
|
-
|
|
59
|
-
type ResetPasswordFieldId = (typeof RESET_PASSWORD_FIELD_IDS)[keyof typeof RESET_PASSWORD_FIELD_IDS];
|
|
60
|
-
|
|
61
|
-
// section: component
|
|
62
|
-
export default function reset_password_layout<TClient>({
|
|
63
|
-
image_src,
|
|
64
|
-
image_alt,
|
|
65
|
-
image_background_color = "#f1f5f9",
|
|
66
|
-
field_overrides,
|
|
67
|
-
labels,
|
|
68
|
-
button_colors,
|
|
69
|
-
password_requirements,
|
|
70
|
-
data_client,
|
|
71
|
-
alreadyLoggedInMessage,
|
|
72
|
-
showLogoutButton = true,
|
|
73
|
-
showReturnHomeButton = false,
|
|
74
|
-
returnHomeButtonLabel = "Return home",
|
|
75
|
-
returnHomePath = "/",
|
|
76
|
-
errorMessage = "Reset password link invalid or has expired. Please go to Reset Password page to get a new link.",
|
|
77
|
-
successMessage = "Password reset successfully. Redirecting to login...",
|
|
78
|
-
loginPath = "/hazo_auth/login",
|
|
79
|
-
forgotPasswordPath = "/hazo_auth/forgot_password",
|
|
80
|
-
}: ResetPasswordLayoutProps<TClient>) {
|
|
81
|
-
const fieldDefinitions = createResetPasswordFieldDefinitions(field_overrides);
|
|
82
|
-
const resolvedLabels = resolveResetPasswordLabels(labels);
|
|
83
|
-
const resolvedButtonPalette = resolveResetPasswordButtonPalette(button_colors);
|
|
84
|
-
const resolvedPasswordRequirements = resolveResetPasswordPasswordRequirements(
|
|
85
|
-
password_requirements,
|
|
86
|
-
);
|
|
87
|
-
|
|
88
|
-
const form = use_reset_password_form({
|
|
89
|
-
passwordRequirements: resolvedPasswordRequirements,
|
|
90
|
-
dataClient: data_client,
|
|
91
|
-
loginPath,
|
|
92
|
-
});
|
|
93
|
-
|
|
94
|
-
const renderFields = (formState: UseResetPasswordFormResult) => {
|
|
95
|
-
return ORDERED_FIELDS.map((fieldId) => {
|
|
96
|
-
const fieldDefinition = fieldDefinitions[fieldId];
|
|
97
|
-
const fieldValue = formState.values[fieldId];
|
|
98
|
-
const fieldError = formState.errors[fieldId];
|
|
99
|
-
|
|
100
|
-
const inputElement = (
|
|
101
|
-
<PasswordField
|
|
102
|
-
inputId={fieldDefinition.id}
|
|
103
|
-
ariaLabel={fieldDefinition.ariaLabel}
|
|
104
|
-
value={fieldValue}
|
|
105
|
-
placeholder={fieldDefinition.placeholder}
|
|
106
|
-
autoComplete={fieldDefinition.autoComplete}
|
|
107
|
-
isVisible={formState.passwordVisibility[fieldDefinition.id as "password" | "confirm_password"]}
|
|
108
|
-
onChange={(nextValue) => formState.handleFieldChange(fieldId, nextValue)}
|
|
109
|
-
onToggleVisibility={() =>
|
|
110
|
-
formState.togglePasswordVisibility(fieldDefinition.id as "password" | "confirm_password")
|
|
111
|
-
}
|
|
112
|
-
errorMessage={fieldError as string | string[] | undefined}
|
|
113
|
-
/>
|
|
114
|
-
);
|
|
115
|
-
|
|
116
|
-
return (
|
|
117
|
-
<FormFieldWrapper
|
|
118
|
-
key={fieldId}
|
|
119
|
-
fieldId={fieldDefinition.id}
|
|
120
|
-
label={fieldDefinition.label}
|
|
121
|
-
input={inputElement}
|
|
122
|
-
errorMessage={fieldError as string | string[] | undefined}
|
|
123
|
-
/>
|
|
124
|
-
);
|
|
125
|
-
});
|
|
126
|
-
};
|
|
127
|
-
|
|
128
|
-
// Show success message if password reset was successful
|
|
129
|
-
if (form.isSuccess) {
|
|
130
|
-
return (
|
|
131
|
-
<AlreadyLoggedInGuard
|
|
132
|
-
image_src={image_src}
|
|
133
|
-
image_alt={image_alt}
|
|
134
|
-
image_background_color={image_background_color}
|
|
135
|
-
message={alreadyLoggedInMessage}
|
|
136
|
-
showLogoutButton={showLogoutButton}
|
|
137
|
-
showReturnHomeButton={showReturnHomeButton}
|
|
138
|
-
returnHomeButtonLabel={returnHomeButtonLabel}
|
|
139
|
-
returnHomePath={returnHomePath}
|
|
140
|
-
>
|
|
141
|
-
<TwoColumnAuthLayout
|
|
142
|
-
imageSrc={image_src}
|
|
143
|
-
imageAlt={image_alt}
|
|
144
|
-
imageBackgroundColor={image_background_color}
|
|
145
|
-
formContent={
|
|
146
|
-
<>
|
|
147
|
-
<FormHeader
|
|
148
|
-
heading={resolvedLabels.heading}
|
|
149
|
-
subHeading={resolvedLabels.subHeading}
|
|
150
|
-
/>
|
|
151
|
-
<div className="cls_reset_password_layout_success flex flex-col items-center justify-center gap-4 p-8 text-center">
|
|
152
|
-
<CheckCircle
|
|
153
|
-
className="cls_reset_password_layout_success_icon h-16 w-16 text-green-600"
|
|
154
|
-
aria-hidden="true"
|
|
155
|
-
/>
|
|
156
|
-
<p className="cls_reset_password_layout_success_message text-lg font-medium text-slate-900">
|
|
157
|
-
{successMessage}
|
|
158
|
-
</p>
|
|
159
|
-
</div>
|
|
160
|
-
</>
|
|
161
|
-
}
|
|
162
|
-
/>
|
|
163
|
-
</AlreadyLoggedInGuard>
|
|
164
|
-
);
|
|
165
|
-
}
|
|
166
|
-
|
|
167
|
-
// Show loading state while validating token
|
|
168
|
-
if (form.isValidatingToken) {
|
|
169
|
-
return (
|
|
170
|
-
<AlreadyLoggedInGuard
|
|
171
|
-
image_src={image_src}
|
|
172
|
-
image_alt={image_alt}
|
|
173
|
-
image_background_color={image_background_color}
|
|
174
|
-
message={alreadyLoggedInMessage}
|
|
175
|
-
showLogoutButton={showLogoutButton}
|
|
176
|
-
showReturnHomeButton={showReturnHomeButton}
|
|
177
|
-
returnHomeButtonLabel={returnHomeButtonLabel}
|
|
178
|
-
returnHomePath={returnHomePath}
|
|
179
|
-
>
|
|
180
|
-
<TwoColumnAuthLayout
|
|
181
|
-
imageSrc={image_src}
|
|
182
|
-
imageAlt={image_alt}
|
|
183
|
-
imageBackgroundColor={image_background_color}
|
|
184
|
-
formContent={
|
|
185
|
-
<div className="cls_reset_password_layout_validating flex flex-col items-center justify-center gap-4 py-8">
|
|
186
|
-
<Loader2 className="h-12 w-12 animate-spin text-slate-600" aria-hidden="true" />
|
|
187
|
-
<div className="cls_reset_password_layout_validating_text text-center">
|
|
188
|
-
<h1 className="cls_reset_password_layout_validating_heading text-2xl font-semibold text-slate-900">
|
|
189
|
-
{resolvedLabels.heading}
|
|
190
|
-
</h1>
|
|
191
|
-
<p className="cls_reset_password_layout_validating_subheading mt-2 text-sm text-slate-600">
|
|
192
|
-
Validating reset token...
|
|
193
|
-
</p>
|
|
194
|
-
</div>
|
|
195
|
-
</div>
|
|
196
|
-
}
|
|
197
|
-
/>
|
|
198
|
-
</AlreadyLoggedInGuard>
|
|
199
|
-
);
|
|
200
|
-
}
|
|
201
|
-
|
|
202
|
-
// Show error message if token is invalid or missing
|
|
203
|
-
if (form.tokenError || !form.token) {
|
|
204
|
-
return (
|
|
205
|
-
<AlreadyLoggedInGuard
|
|
206
|
-
image_src={image_src}
|
|
207
|
-
image_alt={image_alt}
|
|
208
|
-
image_background_color={image_background_color}
|
|
209
|
-
message={alreadyLoggedInMessage}
|
|
210
|
-
showLogoutButton={showLogoutButton}
|
|
211
|
-
showReturnHomeButton={showReturnHomeButton}
|
|
212
|
-
returnHomeButtonLabel={returnHomeButtonLabel}
|
|
213
|
-
returnHomePath={returnHomePath}
|
|
214
|
-
>
|
|
215
|
-
<TwoColumnAuthLayout
|
|
216
|
-
imageSrc={image_src}
|
|
217
|
-
imageAlt={image_alt}
|
|
218
|
-
imageBackgroundColor={image_background_color}
|
|
219
|
-
formContent={
|
|
220
|
-
<div className="cls_reset_password_layout_error flex flex-col items-center justify-center gap-4 p-8 text-center">
|
|
221
|
-
<XCircle
|
|
222
|
-
className="cls_reset_password_layout_error_icon h-16 w-16 text-red-600"
|
|
223
|
-
aria-hidden="true"
|
|
224
|
-
/>
|
|
225
|
-
<div className="cls_reset_password_layout_error_text">
|
|
226
|
-
<h1 className="cls_reset_password_layout_error_heading text-2xl font-semibold text-slate-900">
|
|
227
|
-
Invalid Reset Link
|
|
228
|
-
</h1>
|
|
229
|
-
<p className="cls_reset_password_layout_error_message mt-2 text-sm text-slate-600">
|
|
230
|
-
{form.tokenError || errorMessage}
|
|
231
|
-
</p>
|
|
232
|
-
</div>
|
|
233
|
-
<Link
|
|
234
|
-
href={forgotPasswordPath}
|
|
235
|
-
className="cls_reset_password_layout_forgot_password_link mt-4 text-sm text-blue-600 hover:text-blue-800 underline"
|
|
236
|
-
>
|
|
237
|
-
Go to Reset Password page
|
|
238
|
-
</Link>
|
|
239
|
-
</div>
|
|
240
|
-
}
|
|
241
|
-
/>
|
|
242
|
-
</AlreadyLoggedInGuard>
|
|
243
|
-
);
|
|
244
|
-
}
|
|
245
|
-
|
|
246
|
-
return (
|
|
247
|
-
<AlreadyLoggedInGuard
|
|
248
|
-
image_src={image_src}
|
|
249
|
-
image_alt={image_alt}
|
|
250
|
-
image_background_color={image_background_color}
|
|
251
|
-
message={alreadyLoggedInMessage}
|
|
252
|
-
showLogoutButton={showLogoutButton}
|
|
253
|
-
showReturnHomeButton={showReturnHomeButton}
|
|
254
|
-
returnHomeButtonLabel={returnHomeButtonLabel}
|
|
255
|
-
returnHomePath={returnHomePath}
|
|
256
|
-
>
|
|
257
|
-
<TwoColumnAuthLayout
|
|
258
|
-
imageSrc={image_src}
|
|
259
|
-
imageAlt={image_alt}
|
|
260
|
-
imageBackgroundColor={image_background_color}
|
|
261
|
-
formContent={
|
|
262
|
-
<>
|
|
263
|
-
<FormHeader
|
|
264
|
-
heading={resolvedLabels.heading}
|
|
265
|
-
subHeading={resolvedLabels.subHeading}
|
|
266
|
-
/>
|
|
267
|
-
<form
|
|
268
|
-
className="cls_reset_password_layout_form_fields flex flex-col gap-5"
|
|
269
|
-
onSubmit={form.handleSubmit}
|
|
270
|
-
aria-label="Reset password form"
|
|
271
|
-
>
|
|
272
|
-
{renderFields(form)}
|
|
273
|
-
<FormActionButtons
|
|
274
|
-
submitLabel={resolvedLabels.submitButton}
|
|
275
|
-
cancelLabel={resolvedLabels.cancelButton}
|
|
276
|
-
buttonPalette={resolvedButtonPalette}
|
|
277
|
-
isSubmitDisabled={form.isSubmitDisabled}
|
|
278
|
-
onCancel={form.handleCancel}
|
|
279
|
-
submitAriaLabel="Submit reset password form"
|
|
280
|
-
cancelAriaLabel="Cancel reset password form"
|
|
281
|
-
/>
|
|
282
|
-
{form.isSubmitting && (
|
|
283
|
-
<div className="cls_reset_password_layout_submitting_indicator text-sm text-slate-600 text-center">
|
|
284
|
-
Resetting password...
|
|
285
|
-
</div>
|
|
286
|
-
)}
|
|
287
|
-
</form>
|
|
288
|
-
</>
|
|
289
|
-
}
|
|
290
|
-
/>
|
|
291
|
-
</AlreadyLoggedInGuard>
|
|
292
|
-
);
|
|
293
|
-
}
|
|
294
|
-
|
|
@@ -1,95 +0,0 @@
|
|
|
1
|
-
// file_description: reusable component to show "already logged in" message when user is authenticated
|
|
2
|
-
// section: client_directive
|
|
3
|
-
"use client";
|
|
4
|
-
|
|
5
|
-
// section: imports
|
|
6
|
-
import { use_auth_status } from "../hooks/use_auth_status";
|
|
7
|
-
import { LogoutButton } from "./logout_button";
|
|
8
|
-
import { Button } from "../../../ui/button";
|
|
9
|
-
import { TwoColumnAuthLayout } from "./two_column_auth_layout";
|
|
10
|
-
import { useRouter } from "next/navigation";
|
|
11
|
-
import { Home } from "lucide-react";
|
|
12
|
-
|
|
13
|
-
// section: types
|
|
14
|
-
export type AlreadyLoggedInGuardProps = {
|
|
15
|
-
image_src: string;
|
|
16
|
-
image_alt: string;
|
|
17
|
-
image_background_color?: string;
|
|
18
|
-
message?: string;
|
|
19
|
-
showLogoutButton?: boolean;
|
|
20
|
-
showReturnHomeButton?: boolean;
|
|
21
|
-
returnHomeButtonLabel?: string;
|
|
22
|
-
returnHomePath?: string;
|
|
23
|
-
requireEmailVerified?: boolean;
|
|
24
|
-
children: React.ReactNode;
|
|
25
|
-
};
|
|
26
|
-
|
|
27
|
-
// section: component
|
|
28
|
-
/**
|
|
29
|
-
* Guard component that shows "already logged in" message if user is authenticated
|
|
30
|
-
* Otherwise renders children
|
|
31
|
-
* @param props - Component props including layout config and message customization
|
|
32
|
-
* @returns Either the "already logged in" UI or the children
|
|
33
|
-
*/
|
|
34
|
-
export function AlreadyLoggedInGuard({
|
|
35
|
-
image_src,
|
|
36
|
-
image_alt,
|
|
37
|
-
image_background_color = "#f1f5f9",
|
|
38
|
-
message = "You're already logged in.",
|
|
39
|
-
showLogoutButton = true,
|
|
40
|
-
showReturnHomeButton = false,
|
|
41
|
-
returnHomeButtonLabel = "Return home",
|
|
42
|
-
returnHomePath = "/",
|
|
43
|
-
requireEmailVerified = false,
|
|
44
|
-
children,
|
|
45
|
-
}: AlreadyLoggedInGuardProps) {
|
|
46
|
-
const router = useRouter();
|
|
47
|
-
const authStatus = use_auth_status();
|
|
48
|
-
|
|
49
|
-
// Check if user should see "already logged in" message
|
|
50
|
-
// If requireEmailVerified is true, only show if email is verified
|
|
51
|
-
// If requireEmailVerified is false, show if authenticated
|
|
52
|
-
const shouldShowAlreadyLoggedIn =
|
|
53
|
-
authStatus.authenticated &&
|
|
54
|
-
!authStatus.loading &&
|
|
55
|
-
(!requireEmailVerified || authStatus.email_verified === true);
|
|
56
|
-
|
|
57
|
-
if (shouldShowAlreadyLoggedIn) {
|
|
58
|
-
return (
|
|
59
|
-
<TwoColumnAuthLayout
|
|
60
|
-
imageSrc={image_src}
|
|
61
|
-
imageAlt={image_alt}
|
|
62
|
-
imageBackgroundColor={image_background_color}
|
|
63
|
-
formContent={
|
|
64
|
-
<div className="cls_already_logged_in_guard flex flex-col items-center justify-center gap-4 p-8 text-center">
|
|
65
|
-
<p className="cls_already_logged_in_guard_message text-lg font-medium text-slate-900">
|
|
66
|
-
{message}
|
|
67
|
-
</p>
|
|
68
|
-
<div className="cls_already_logged_in_guard_actions flex flex-col gap-3 items-center mt-4">
|
|
69
|
-
{showLogoutButton && (
|
|
70
|
-
<LogoutButton
|
|
71
|
-
className="cls_already_logged_in_guard_logout_button"
|
|
72
|
-
variant="default"
|
|
73
|
-
/>
|
|
74
|
-
)}
|
|
75
|
-
{showReturnHomeButton && (
|
|
76
|
-
<Button
|
|
77
|
-
onClick={() => router.push(returnHomePath)}
|
|
78
|
-
variant="outline"
|
|
79
|
-
className="cls_already_logged_in_guard_return_home_button"
|
|
80
|
-
aria-label={returnHomeButtonLabel}
|
|
81
|
-
>
|
|
82
|
-
<Home className="h-4 w-4 mr-2" aria-hidden="true" />
|
|
83
|
-
{returnHomeButtonLabel}
|
|
84
|
-
</Button>
|
|
85
|
-
)}
|
|
86
|
-
</div>
|
|
87
|
-
</div>
|
|
88
|
-
}
|
|
89
|
-
/>
|
|
90
|
-
);
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
return <>{children}</>;
|
|
94
|
-
}
|
|
95
|
-
|
|
@@ -1,36 +0,0 @@
|
|
|
1
|
-
// file_description: server component that chooses between sidebar shell and standalone shell
|
|
2
|
-
// section: imports
|
|
3
|
-
import type { ReactNode } from "react";
|
|
4
|
-
import { SidebarLayoutWrapper } from "./sidebar_layout_wrapper";
|
|
5
|
-
import { StandaloneLayoutWrapper } from "./standalone_layout_wrapper";
|
|
6
|
-
import { get_ui_shell_config } from "../../../../lib/ui_shell_config.server";
|
|
7
|
-
|
|
8
|
-
// section: types
|
|
9
|
-
type AuthPageShellProps = {
|
|
10
|
-
children: ReactNode;
|
|
11
|
-
};
|
|
12
|
-
|
|
13
|
-
// section: component
|
|
14
|
-
export function AuthPageShell({ children }: AuthPageShellProps) {
|
|
15
|
-
const uiShellConfig = get_ui_shell_config();
|
|
16
|
-
|
|
17
|
-
if (uiShellConfig.layout_mode === "standalone") {
|
|
18
|
-
return (
|
|
19
|
-
<StandaloneLayoutWrapper
|
|
20
|
-
heading={uiShellConfig.standalone_heading}
|
|
21
|
-
description={uiShellConfig.standalone_description}
|
|
22
|
-
wrapperClassName={uiShellConfig.standalone_wrapper_class}
|
|
23
|
-
contentClassName={uiShellConfig.standalone_content_class}
|
|
24
|
-
showHeading={uiShellConfig.standalone_show_heading}
|
|
25
|
-
showDescription={uiShellConfig.standalone_show_description}
|
|
26
|
-
>
|
|
27
|
-
{children}
|
|
28
|
-
</StandaloneLayoutWrapper>
|
|
29
|
-
);
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
return <SidebarLayoutWrapper>{children}</SidebarLayoutWrapper>;
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|