diff --git a/Projects/App/Sources/Application/DI/AppComponent.swift b/Projects/App/Sources/Application/DI/AppComponent.swift index 2690d572..d713a765 100644 --- a/Projects/App/Sources/Application/DI/AppComponent.swift +++ b/Projects/App/Sources/Application/DI/AppComponent.swift @@ -42,6 +42,8 @@ public extension AppComponent { var signupProfileImageComponent: SignupProfileImageComponent { SignupProfileImageComponent(parent: self) } + var signupPasswordComponent: SignupPasswordComponent { + SignupPasswordComponent(parent: self) var signupTermsComponent: SignupTermsComponent { SignupTermsComponent(parent: self) } diff --git a/Projects/App/Sources/Application/DMSApp.swift b/Projects/App/Sources/Application/DMSApp.swift index 69188092..14074981 100644 --- a/Projects/App/Sources/Application/DMSApp.swift +++ b/Projects/App/Sources/Application/DMSApp.swift @@ -12,7 +12,7 @@ struct DMSApp: App { var body: some Scene { WindowGroup { - AppComponent().signupTermsComponent.makeView() + AppComponent().mainTabComponent.makeView() } } } diff --git a/Projects/App/Sources/Application/NeedleGenerated.swift b/Projects/App/Sources/Application/NeedleGenerated.swift index e55f2baf..ac907491 100644 --- a/Projects/App/Sources/Application/NeedleGenerated.swift +++ b/Projects/App/Sources/Application/NeedleGenerated.swift @@ -51,6 +51,17 @@ private class SchoolCodeDependencyc0114744c1c8c7843672Provider: SchoolCodeDepend private func factoryb65c1efbf06b87162473f47b58f8f304c97af4d5(_ component: NeedleFoundation.Scope) -> AnyObject { return SchoolCodeDependencyc0114744c1c8c7843672Provider(appComponent: parent1(component) as! AppComponent) } +private class SignupPasswordDependency778bf5389a70d7df6152Provider: SignupPasswordDependency { + + + init() { + + } +} +/// ^->AppComponent->SignupPasswordComponent +private func factorye93d1d56840ff97c674ae3b0c44298fc1c149afb(_ component: NeedleFoundation.Scope) -> AnyObject { + return SignupPasswordDependency778bf5389a70d7df6152Provider() +} private class SignupEmailAuthCodeVerifyDependencyaf9da1ebf0e9e5f1b708Provider: SignupEmailAuthCodeVerifyDependency { var sendAuthCodeUseCase: any SendAuthCodeUseCase { return appComponent.sendAuthCodeUseCase @@ -182,6 +193,7 @@ extension AppComponent: Registration { localTable["signupEmailAuthCodeVerifyComponent-SignupEmailAuthCodeVerifyComponent"] = { self.signupEmailAuthCodeVerifyComponent as Any } localTable["signupTermsComponent-SignupTermsComponent"] = { self.signupTermsComponent as Any } localTable["signupProfileImageComponent-SignupProfileImageComponent"] = { self.signupProfileImageComponent as Any } + localTable["signupPasswordComponent-SignupPasswordComponent"] = { self.signupPasswordComponent as Any } localTable["mainTabComponent-MainTabComponent"] = { self.mainTabComponent as Any } localTable["homeComponent-HomeComponent"] = { self.homeComponent as Any } localTable["remoteStudentsDataSource-any RemoteStudentsDataSource"] = { self.remoteStudentsDataSource as Any } @@ -216,6 +228,11 @@ extension SchoolCodeComponent: Registration { keyPathToName[\SchoolCodeDependency.checkSchoolCodeUseCase] = "checkSchoolCodeUseCase-any CheckSchoolCodeUseCase" } } +extension SignupPasswordComponent: Registration { + public func registerItems() { + + } +} extension SignupEmailAuthCodeVerifyComponent: Registration { public func registerItems() { keyPathToName[\SignupEmailAuthCodeVerifyDependency.sendAuthCodeUseCase] = "sendAuthCodeUseCase-any SendAuthCodeUseCase" @@ -278,6 +295,7 @@ private func register1() { registerProviderFactory("^->AppComponent", factoryEmptyDependencyProvider) registerProviderFactory("^->AppComponent->SchoolConfirmationQuestionsComponent", factoryd462667f0418a53210fcf47b58f8f304c97af4d5) registerProviderFactory("^->AppComponent->SchoolCodeComponent", factoryb65c1efbf06b87162473f47b58f8f304c97af4d5) + registerProviderFactory("^->AppComponent->SignupPasswordComponent", factorye93d1d56840ff97c674ae3b0c44298fc1c149afb) registerProviderFactory("^->AppComponent->SignupEmailAuthCodeVerifyComponent", factoryb06be35aa893adde971bf47b58f8f304c97af4d5) registerProviderFactory("^->AppComponent->SignupTermsComponent", factoryf84223c07d964abc9b0ee3b0c44298fc1c149afb) registerProviderFactory("^->AppComponent->SignupEmailVerifyComponent", factory3b1904c76335d70151ebf47b58f8f304c97af4d5) diff --git a/Projects/Features/SignupFeature/Sources/SignupPassword/SignupPasswordComponent.swift b/Projects/Features/SignupFeature/Sources/SignupPassword/SignupPasswordComponent.swift new file mode 100644 index 00000000..cd4ad441 --- /dev/null +++ b/Projects/Features/SignupFeature/Sources/SignupPassword/SignupPasswordComponent.swift @@ -0,0 +1,12 @@ +import SwiftUI +import NeedleFoundation + +public protocol SignupPasswordDependency: Dependency {} + +public final class SignupPasswordComponent: Component { + public func makeView() -> some View { + SignupPasswordView( + viewModel: .init() + ) + } +} diff --git a/Projects/Features/SignupFeature/Sources/SignupPassword/SignupPasswordView.swift b/Projects/Features/SignupFeature/Sources/SignupPassword/SignupPasswordView.swift new file mode 100644 index 00000000..e1170365 --- /dev/null +++ b/Projects/Features/SignupFeature/Sources/SignupPassword/SignupPasswordView.swift @@ -0,0 +1,69 @@ +import DesignSystem +import SwiftUI + +struct SignupPasswordView: View { + private enum FocusField { + case password + case passwordCheck + } + @StateObject var viewModel: SignupPasswordViewModel + @FocusState private var focusField: FocusField? + + public init( + viewModel: SignupPasswordViewModel + ) { + _viewModel = StateObject(wrappedValue: viewModel) + } + + var body: some View { + VStack(spacing: 4) { + HStack { + VStack(alignment: .leading, spacing: 8) { + Text("DMS") + .dmsFont(.title(.extraLarge), color: .PrimaryVariant.primary) + + Text("학교 인증코드 입력") + .dmsFont(.text(.medium), color: .GrayScale.gray6) + + Text("비밀번호는 영문, 숫자, 기호를 포함한 8~20자이어야 합니다.") + .dmsFont(.text(.extraSmall), color: .GrayScale.gray5) + } + + Spacer() + } + .padding(.top, 24) + + VStack(spacing: 56) { + SecureDMSFloatingTextField( + "비밀번호", + text: $viewModel.password, + isError: viewModel.isPasswordRegexError, + errorMessage: "비밀번호 형식이 올바르지 않습니다." + ) { + focusField = .passwordCheck + } + .padding(.top, 56) + .focused($focusField, equals: .password) + + SecureDMSFloatingTextField( + "비밀번호 확인", + text: $viewModel.passwordCheck, + isError: viewModel.isPasswordMismatchedError, + errorMessage: "비밀번호가 일치하지 않습니다." + ) { + viewModel.nextButtonDidTap() + } + .focused($focusField, equals: .passwordCheck) + } + + Spacer() + + DMSWideButton(text: "다음", color: .PrimaryVariant.primary) { + viewModel.nextButtonDidTap() + } + .padding(.bottom, 40) + } + .dmsBackground() + .padding(.horizontal, 24) + } +} diff --git a/Projects/Features/SignupFeature/Sources/SignupPassword/SignupPasswordViewModel.swift b/Projects/Features/SignupFeature/Sources/SignupPassword/SignupPasswordViewModel.swift new file mode 100644 index 00000000..ab31756d --- /dev/null +++ b/Projects/Features/SignupFeature/Sources/SignupPassword/SignupPasswordViewModel.swift @@ -0,0 +1,40 @@ +import BaseFeature +import Combine +import Utility + +final class SignupPasswordViewModel: BaseViewModel { + @Published var password = "" { + didSet { resettingError() } + } + @Published var passwordCheck = "" { + didSet { resettingError() } + } + @Published var isPasswordRegexError = false + @Published var isPasswordMismatchedError = false + + var isEnabledNextStep: Bool { + !password.isEmpty && !passwordCheck.isEmpty + } + + func nextButtonDidTap() { + guard isEnabledNextStep else { + return + } + + let passwordExpression = "^(?=.*[A-Za-z])(?=.*[0-9])(?=.*[!@#$%^&*()_+=-]).{8,20}" + guard password ~= passwordExpression else { + isPasswordRegexError = true + return + } + + guard password == passwordCheck else { + isPasswordMismatchedError = true + return + } + } + + func resettingError() { + isPasswordRegexError = false + isPasswordMismatchedError = false + } +}