From eef3f20bfd89eeabf1881c75f1ffc878351bc346 Mon Sep 17 00:00:00 2001 From: Jisu Kim Date: Sun, 10 Nov 2024 23:02:59 +0900 Subject: [PATCH 01/11] [WEAV-122] Profile Widget Component Added --- .../ProfileWidget/ProfileWidgetView.swift | 73 +++++++++++++++++++ .../Home/Sources/Profile/ProfileView.swift | 7 ++ 2 files changed, 80 insertions(+) create mode 100644 Projects/DesignSystem/DesignCore/Sources/ProfileWidget/ProfileWidgetView.swift diff --git a/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/ProfileWidgetView.swift b/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/ProfileWidgetView.swift new file mode 100644 index 0000000..b15538a --- /dev/null +++ b/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/ProfileWidgetView.swift @@ -0,0 +1,73 @@ +// +// ProfileWidgetView.swift +// DesignCore +// +// Created by 김지수 on 11/7/24. +// Copyright © 2024 com.weave. All rights reserved. +// + +import SwiftUI + +struct ProfileWidgetView: View { + var body: some View { + ZStack { + RoundedRectangle(cornerRadius: 24) + .fill( + LinearGradient( + colors: [ + .init(hex: 0xEDF7FF), + .init(hex: 0xCDE8FF), + ], + startPoint: .top, + endPoint: .bottom + ) + ) + VStack { + HStack { + Text("취미") + Spacer() + Image(systemName: "plus.circle") + } + Spacer() + ScrollView { + LeftAlignText( + """ + ex. + 공백 포함해서 최소 5글자 이상부터 최대 40자까지 입력 + 우어어어엉 + """ + ) + .typography(.regular_14) + } + } + .scrollIndicators(.hidden) + .padding(.all, 20) + } + } +} + +#Preview { + ScrollView { + VStack { + ProfileWidgetView() + .frame(width: 162, height: 150) + ProfileWidgetView() + .frame(width: 162, height: 150) + ProfileWidgetView() + .frame(width: 162, height: 150) + ProfileWidgetView() + .frame(width: 162, height: 150) + ProfileWidgetView() + .frame(width: 162, height: 150) + ProfileWidgetView() + .frame(width: 162, height: 150) + ProfileWidgetView() + .frame(width: 162, height: 150) + ProfileWidgetView() + .frame(width: 162, height: 150) + ProfileWidgetView() + .frame(width: 162, height: 150) + } + } + +} diff --git a/Projects/Features/Home/Sources/Profile/ProfileView.swift b/Projects/Features/Home/Sources/Profile/ProfileView.swift index 3a5bf94..1445c98 100644 --- a/Projects/Features/Home/Sources/Profile/ProfileView.swift +++ b/Projects/Features/Home/Sources/Profile/ProfileView.swift @@ -15,6 +15,7 @@ import Model public struct ProfileView: View { @StateObject var container: MVIContainer + @State var isShowWidgetSelectionView = false private var intent: ProfileIntent.Intentable { container.intent } private var state: ProfileModel.Stateful { container.model } @@ -97,6 +98,9 @@ public struct ProfileView: View { .frame(height: widgetSize) .shadow(.default) .padding(.bottom, 36) + .onTapGesture { + isShowWidgetSelectionView = true + } } .padding(.horizontal, 18) .padding(.top, 36) @@ -105,6 +109,9 @@ public struct ProfileView: View { ProgressView() } } + .sheet(isPresented: $isShowWidgetSelectionView, content: { + Rectangle() + }) .task { await intent.task() } From c6dfaec0f83dfe06687d939aa0c2400d76457055 Mon Sep 17 00:00:00 2001 From: Jisu Kim Date: Mon, 11 Nov 2024 00:47:29 +0900 Subject: [PATCH 02/11] =?UTF-8?q?[WEAV-122]=20Widget=20Selection=20View=20?= =?UTF-8?q?=EC=83=9D=EC=84=B1?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../CoreKit}/Sources/Color+Ext.swift | 6 +- .../ProfileWidget/ProfileWidgetView.swift | 84 ++++----- .../Home/Sources/Profile/ProfileView.swift | 14 +- .../WidgetSelectionIntent.swift | 51 ++++++ .../WidgetSelectionModel.swift | 78 ++++++++ .../WidgetSelection/WidgetSelectionView.swift | 88 +++++++++ .../Sources/Profile/Widget/WidgetType.swift | 170 ++++++++++++++++++ 7 files changed, 437 insertions(+), 54 deletions(-) rename Projects/{DesignSystem/DesignCore => Core/CoreKit}/Sources/Color+Ext.swift (75%) create mode 100644 Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionIntent.swift create mode 100644 Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionModel.swift create mode 100644 Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionView.swift create mode 100644 Projects/Model/Model/Sources/Profile/Widget/WidgetType.swift diff --git a/Projects/DesignSystem/DesignCore/Sources/Color+Ext.swift b/Projects/Core/CoreKit/Sources/Color+Ext.swift similarity index 75% rename from Projects/DesignSystem/DesignCore/Sources/Color+Ext.swift rename to Projects/Core/CoreKit/Sources/Color+Ext.swift index 45f0459..c33158b 100644 --- a/Projects/DesignSystem/DesignCore/Sources/Color+Ext.swift +++ b/Projects/Core/CoreKit/Sources/Color+Ext.swift @@ -1,9 +1,9 @@ // // Color+Ext.swift -// DesignCore +// CoreKit // -// Created by 김지수 on 9/14/24. -// Copyright © 2024 com.studentcenter. All rights reserved. +// Created by 김지수 on 11/11/24. +// Copyright © 2024 com.weave. All rights reserved. // import SwiftUI diff --git a/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/ProfileWidgetView.swift b/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/ProfileWidgetView.swift index b15538a..f533f58 100644 --- a/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/ProfileWidgetView.swift +++ b/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/ProfileWidgetView.swift @@ -7,37 +7,53 @@ // import SwiftUI +import CoreKit -struct ProfileWidgetView: View { - var body: some View { +public struct ProfileWidgetView: View { + + public let title: String + public let bodyText: String + public let titleColor: Color + public let bodyColor: Color + public let gradientColors: [Color] + + public init( + title: String, + bodyText: String, + titleColor: Color, + bodyColor: Color, + gradientColors: [Color] + ) { + self.title = title + self.bodyText = bodyText + self.titleColor = titleColor + self.bodyColor = bodyColor + self.gradientColors = gradientColors + } + + public var body: some View { ZStack { RoundedRectangle(cornerRadius: 24) .fill( LinearGradient( - colors: [ - .init(hex: 0xEDF7FF), - .init(hex: 0xCDE8FF), - ], + colors: gradientColors, startPoint: .top, endPoint: .bottom ) ) - VStack { - HStack { - Text("취미") - Spacer() - Image(systemName: "plus.circle") - } + VStack { + HStack { + Text(title) + .pretendard(weight: ._600, size: 22) + .foregroundStyle(titleColor) Spacer() - ScrollView { - LeftAlignText( - """ - ex. - 공백 포함해서 최소 5글자 이상부터 최대 40자까지 입력 - 우어어어엉 - """ - ) - .typography(.regular_14) + Image(systemName: "plus.circle") + } + Spacer() + ScrollView { + LeftAlignText(bodyText) + .typography(.regular_14) + .foregroundStyle(bodyColor) } } .scrollIndicators(.hidden) @@ -45,29 +61,3 @@ struct ProfileWidgetView: View { } } } - -#Preview { - ScrollView { - VStack { - ProfileWidgetView() - .frame(width: 162, height: 150) - ProfileWidgetView() - .frame(width: 162, height: 150) - ProfileWidgetView() - .frame(width: 162, height: 150) - ProfileWidgetView() - .frame(width: 162, height: 150) - ProfileWidgetView() - .frame(width: 162, height: 150) - ProfileWidgetView() - .frame(width: 162, height: 150) - ProfileWidgetView() - .frame(width: 162, height: 150) - ProfileWidgetView() - .frame(width: 162, height: 150) - ProfileWidgetView() - .frame(width: 162, height: 150) - } - } - -} diff --git a/Projects/Features/Home/Sources/Profile/ProfileView.swift b/Projects/Features/Home/Sources/Profile/ProfileView.swift index 1445c98..2e293d6 100644 --- a/Projects/Features/Home/Sources/Profile/ProfileView.swift +++ b/Projects/Features/Home/Sources/Profile/ProfileView.swift @@ -15,7 +15,7 @@ import Model public struct ProfileView: View { @StateObject var container: MVIContainer - @State var isShowWidgetSelectionView = false + @State var isPresentWidgetSelectionView = false private var intent: ProfileIntent.Intentable { container.intent } private var state: ProfileModel.Stateful { container.model } @@ -99,7 +99,7 @@ public struct ProfileView: View { .shadow(.default) .padding(.bottom, 36) .onTapGesture { - isShowWidgetSelectionView = true + isPresentWidgetSelectionView = true } } .padding(.horizontal, 18) @@ -109,8 +109,14 @@ public struct ProfileView: View { ProgressView() } } - .sheet(isPresented: $isShowWidgetSelectionView, content: { - Rectangle() + .sheet( + isPresented: $isPresentWidgetSelectionView, + content: { + NavigationView { + WidgetSelectionView( + isPresented: $isPresentWidgetSelectionView + ) + } }) .task { await intent.task() diff --git a/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionIntent.swift b/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionIntent.swift new file mode 100644 index 0000000..2b442bc --- /dev/null +++ b/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionIntent.swift @@ -0,0 +1,51 @@ +// +// WidgetSelectionIntent.swift +// Home +// +// Created by 김지수 on 11/10/24. +// Copyright © 2024 com.weave. All rights reserved. +// + +import Foundation +import CommonKit +import CoreKit + +//MARK: - Intent +class WidgetSelectionIntent { + private weak var model: WidgetSelectionModelActionable? + private let input: DataModel + + // MARK: Life cycle + init( + model: WidgetSelectionModelActionable, + input: DataModel + ) { + self.input = input + self.model = model + } +} + +//MARK: - Intentable +extension WidgetSelectionIntent { + protocol Intentable { + // content + func onTapNextButton() + + // default + func onAppear() + func task() async + } + + struct DataModel {} +} + +//MARK: - Intentable +extension WidgetSelectionIntent: WidgetSelectionIntent.Intentable { + // default + func onAppear() {} + + func task() async {} + + // content + func onTapNextButton() {} +} diff --git a/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionModel.swift b/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionModel.swift new file mode 100644 index 0000000..377846e --- /dev/null +++ b/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionModel.swift @@ -0,0 +1,78 @@ +// +// WidgetSelectionModel.swift +// Home +// +// Created by 김지수 on 11/10/24. +// Copyright © 2024 com.weave. All rights reserved. +// + +import Foundation +import CommonKit +import CoreKit + +final class WidgetSelectionModel: ObservableObject { + + //MARK: Stateful + protocol Stateful { + // content + var isValidated: Bool { get } + + // default + var isLoading: Bool { get } + + // error + var showErrorView: ErrorModel? { get } + var showErrorAlert: ErrorModel? { get } + } + + //MARK: State Properties + // content + @Published var isValidated: Bool = false + + // default + @Published var isLoading: Bool = false + + // error + @Published var showErrorView: ErrorModel? + @Published var showErrorAlert: ErrorModel? +} + +extension WidgetSelectionModel: WidgetSelectionModel.Stateful {} + +//MARK: - Actionable +protocol WidgetSelectionModelActionable: AnyObject { + // content + func setValidation(value: Bool) + + // default + func setLoading(status: Bool) + + // error + func showErrorView(error: ErrorModel) + func showErrorAlert(error: ErrorModel) + func resetError() +} + +extension WidgetSelectionModel: WidgetSelectionModelActionable { + // content + func setValidation(value: Bool) { + isValidated = value + } + + // default + func setLoading(status: Bool) { + isLoading = status + } + + // error + func showErrorView(error: ErrorModel) { + showErrorView = error + } + func showErrorAlert(error: ErrorModel) { + showErrorAlert = error + } + func resetError() { + showErrorView = nil + showErrorAlert = nil + } +} diff --git a/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionView.swift b/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionView.swift new file mode 100644 index 0000000..63dfb99 --- /dev/null +++ b/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionView.swift @@ -0,0 +1,88 @@ +// +// WidgetSelectionView.swift +// Home +// +// Created by 김지수 on 11/10/24. +// Copyright © 2024 com.weave. All rights reserved. +// + +import SwiftUI +import CoreKit +import DesignCore +import CommonKit +import Model + +public struct WidgetSelectionView: View { + + @Binding var isShowSelectionView: Bool + @StateObject var container: MVIContainer + + private var intent: WidgetSelectionIntent.Intentable { container.intent } + private var state: WidgetSelectionModel.Stateful { container.model } + + let columns = [ + GridItem(.flexible(), spacing: 16), + GridItem(.flexible(), spacing: 16) + ] + + public init(isPresented: Binding) { + let model = WidgetSelectionModel() + let intent = WidgetSelectionIntent( + model: model, + input: .init() + ) + let container = MVIContainer( + intent: intent as WidgetSelectionIntent.Intentable, + model: model as WidgetSelectionModel.Stateful, + modelChangePublisher: model.objectWillChange + ) + self._container = StateObject(wrappedValue: container) + self._isShowSelectionView = isPresented + } + + public var body: some View { + VStack { + Text("추가하고 싶은 프로필 위젯을 눌러 소개를 작성해요!") + .typography(.regular_14) + .padding(.top, 20) + ScrollView { + LazyVGrid(columns: columns, spacing: 16) { + ForEach(WidgetType.allCases, id: \.self) { widget in + ProfileWidgetView( + title: widget.title, + bodyText: widget.exampleText, + titleColor: widget.titleColor, + bodyColor: widget.bodyColor, + gradientColors: widget.gradationColors + ) + } + } + .padding() + } + } + .task { + await intent.task() + } + .onAppear { + intent.onAppear() + } + .ignoresSafeArea(.keyboard) + .navigationTitle("프로필 위젯") + .navigationBarTitleDisplayMode(.inline) + .toolbar { + ToolbarItem { + Button("닫기") { + isShowSelectionView = false + } + .typography(.medium_16) + } + } + .setLoading(state.isLoading) + } +} + +#Preview { + NavigationView { + WidgetSelectionView(isPresented: .constant(true)) + } +} diff --git a/Projects/Model/Model/Sources/Profile/Widget/WidgetType.swift b/Projects/Model/Model/Sources/Profile/Widget/WidgetType.swift new file mode 100644 index 0000000..bd7c3e9 --- /dev/null +++ b/Projects/Model/Model/Sources/Profile/Widget/WidgetType.swift @@ -0,0 +1,170 @@ +// +// WidgetType.swift +// Model +// +// Created by 김지수 on 11/10/24. +// Copyright © 2024 com.weave. All rights reserved. +// + +import SwiftUI +import CoreKit + +public enum WidgetType: CaseIterable { + case hobby + case style + case mbti + case music + case body + case food + case movie + case drama + case book + case travel + case alcohol + case marriage + case religion + case smoking +} + +extension WidgetType { + public var title: String { + switch self { + case .hobby: return "취미" + case .style: return "스타일" + case .mbti: return "MBTI" + case .music: return "음악" + case .body: return "키·체형" + case .food: return "음식" + case .movie: return "영화" + case .drama: return "드라마" + case .book: return "책" + case .travel: return "여행" + case .alcohol: return "술" + case .marriage: return "결혼관" + case .religion: return "종교" + case .smoking: return "흡연" + } + } + + public var exampleText: String { + switch self { + case .hobby: + return "ex.\n테니스랑 헬스 즐겨해요! 같이 하실 분?" + case .style: + return "ex.\n옷은 깔끔하게 흰 티에 청바지만 입는 게 진리입니다" + case .mbti: + return "ex.\n저는 INTP지만 연애할 때는 F 100%가 된답니다" + case .music: + return "ex.\nFly to me the moon이 제 인생곡이에요!" + case .body: + return "ex.\n키는 180이구 헬스 하면서 어깨 키우고 있어요☺️" + case .food: + return "ex.\n음식 가리는 거 없이 거의 다 잘 먹어요!" + case .movie: + return "ex.\n제 인생 영화는 비긴 어게인이에용" + case .drama: + return "ex.\n하츠코이, 언내추럴 같은 일드 취향👀" + case .book: + return "ex.\nIT 관련 서적이나 자기계발서 위주로 봐요" + case .travel: + return "ex.\n아이슬란드처럼 대자연의 낭만이 있는 곳으로 가보고 싶어요.." + case .alcohol: + return "ex.\n화이트 와인, 하이볼, 칵테일을 좋아하는 술찌입니다😇" + case .marriage: + return "ex.\n마음만 맞으면 결혼 자금이나 시기는 조율할 수 있다고 생각해요!" + case .religion: + return "ex.\n저와 우리 집안 모두 무교입니다!" + case .smoking: + return "ex.\n전자담배만 피워요 ! 담배냄새는 저도 싫어합니다 ㅜ" + } + } +} + +struct WidgetColorSet { + let gradientColors: [Color] + let titleColor: Color + let bodyColor: Color + + init( + gradientColors: [Color], + titleColor: Color + ) { + self.gradientColors = gradientColors + self.titleColor = titleColor + self.bodyColor = titleColor.opacity(0.6) + } + + static var allColorSets: [WidgetColorSet] { + [skyBlueColorSet, brownColorSet, pinkColorSet, greyColorSet, greenColorSet] + } + + static var skyBlueColorSet: WidgetColorSet { + .init( + gradientColors: [ + .init(hex: 0xEDF7FF), + .init(hex: 0xCDE8FF), + ], + titleColor: .init(hex: 0x15394B) + ) + } + + static var brownColorSet: WidgetColorSet { + .init( + gradientColors: [ + .init(hex: 0xFAF3E5), + .init(hex: 0xEEDCB9), + ], + titleColor: .init(hex: 0x4C3B1C) + ) + } + + static var pinkColorSet: WidgetColorSet { + .init( + gradientColors: [ + .init(hex: 0xFEF0F4), + .init(hex: 0xEFD6E1), + ], + titleColor: .init(hex: 0x6C324A) + ) + } + + static var greyColorSet: WidgetColorSet { + .init( + gradientColors: [ + .init(hex: 0xF9F9F9), + .init(hex: 0xE7E7E7), + ], + titleColor: .init(hex: 0x454545) + ) + } + + static var greenColorSet: WidgetColorSet { + .init( + gradientColors: [ + .init(hex: 0xF2FCEB), + .init(hex: 0xD7E9C8), + ], + titleColor: .init(hex: 0x1D5018) + ) + } +} + +extension WidgetType { + private var colorSet: WidgetColorSet { + let index = WidgetType.allCases.firstIndex(of: self) ?? 0 + let allColors = WidgetColorSet.allColorSets + return allColors[index % allColors.count] + } + + public var titleColor: Color { + colorSet.titleColor + } + + public var bodyColor: Color { + colorSet.bodyColor + } + + public var gradationColors: [Color] { + colorSet.gradientColors + } +} From c567b4b466d3e334f2da2a1df65f30df69263677 Mon Sep 17 00:00:00 2001 From: Jisu Kim Date: Tue, 12 Nov 2024 01:58:59 +0900 Subject: [PATCH 03/11] =?UTF-8?q?[WEAV-122]=20Writable=20Profile=20Widget?= =?UTF-8?q?=20=EC=83=9D=EC=84=B1,=20Write=20=EB=B7=B0=20=EA=B5=AC=ED=98=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../WritableProfileWidgetView.swift | 86 ++++++++++++++ .../WidgetSelection/WidgetSelectionView.swift | 25 +++- .../WidgetWriting/WidgetWritingIntent.swift | 62 ++++++++++ .../WidgetWriting/WidgetWritingModel.swift | 101 ++++++++++++++++ .../WidgetWriting/WidgetWritingView.swift | 111 ++++++++++++++++++ 5 files changed, 381 insertions(+), 4 deletions(-) create mode 100644 Projects/DesignSystem/DesignCore/Sources/ProfileWidget/WritableProfileWidgetView.swift create mode 100644 Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift create mode 100644 Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift create mode 100644 Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift diff --git a/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/WritableProfileWidgetView.swift b/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/WritableProfileWidgetView.swift new file mode 100644 index 0000000..3bc0370 --- /dev/null +++ b/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/WritableProfileWidgetView.swift @@ -0,0 +1,86 @@ +// +// WritableProfileWidgetView.swift +// DesignCore +// +// Created by 김지수 on 11/12/24. +// Copyright © 2024 com.weave. All rights reserved. +// + +import SwiftUI + +public struct WritableProfileWidgetView: View { + @Binding public var bodyText: String + public let title: String + public let placeholder: String + public let titleColor: Color + public let bodyColor: Color + public let gradientColors: [Color] + + public init( + title: String, + placeholder: String, + bodyText: Binding, + titleColor: Color, + bodyColor: Color, + gradientColors: [Color] + ) { + self.title = title + self.placeholder = placeholder + self._bodyText = bodyText + self.titleColor = titleColor + self.bodyColor = bodyColor + self.gradientColors = gradientColors + } + + public var body: some View { + ZStack { + RoundedRectangle(cornerRadius: 24) + .fill( + LinearGradient( + colors: gradientColors, + startPoint: .top, + endPoint: .bottom + ) + ) + VStack { + HStack { + Text(title) + .pretendard(weight: ._600, size: 22) + .foregroundStyle(titleColor) + Spacer() + } + Spacer() + TextEditor( + text: $bodyText + ) + .textEditorStyle( + PlainTextEditorStyle() + ) + .typography(.regular_14) + .foregroundStyle(bodyColor) + } + .scrollIndicators(.hidden) + .padding(.all, 28) + } + } +} + +struct PreviewView: View { + @State var text = "" + + var body: some View { + WritableProfileWidgetView( + title: "Title", + placeholder: "Placeholder", + bodyText: $text, + titleColor: .black, + bodyColor: .red, + gradientColors: [.yellow, .green] + ) + .frame(width: 200, height: 200) + } +} + +#Preview { + PreviewView() +} diff --git a/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionView.swift b/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionView.swift index 63dfb99..25320ba 100644 --- a/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionView.swift +++ b/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionView.swift @@ -14,7 +14,10 @@ import Model public struct WidgetSelectionView: View { - @Binding var isShowSelectionView: Bool + @State private var isPushedWriteView: Bool = false + @State private var selectedWidget: WidgetType? + + @Binding var isPresentedSelectionView: Bool @StateObject var container: MVIContainer private var intent: WidgetSelectionIntent.Intentable { container.intent } @@ -37,7 +40,7 @@ public struct WidgetSelectionView: View { modelChangePublisher: model.objectWillChange ) self._container = StateObject(wrappedValue: container) - self._isShowSelectionView = isPresented + self._isPresentedSelectionView = isPresented } public var body: some View { @@ -55,11 +58,25 @@ public struct WidgetSelectionView: View { bodyColor: widget.bodyColor, gradientColors: widget.gradationColors ) + .onTapGesture { + selectedWidget = widget + isPushedWriteView = true + } } } .padding() } } + .navigationDestination( + isPresented: $isPushedWriteView, + destination: { + WidgetWritingView( + widgetType: selectedWidget ?? .book, + isModalPresented: $isPresentedSelectionView, + isPushed: $isPushedWriteView + ) + } + ) .task { await intent.task() } @@ -72,7 +89,7 @@ public struct WidgetSelectionView: View { .toolbar { ToolbarItem { Button("닫기") { - isShowSelectionView = false + isPresentedSelectionView = false } .typography(.medium_16) } @@ -82,7 +99,7 @@ public struct WidgetSelectionView: View { } #Preview { - NavigationView { + NavigationStack { WidgetSelectionView(isPresented: .constant(true)) } } diff --git a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift new file mode 100644 index 0000000..7af73a8 --- /dev/null +++ b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift @@ -0,0 +1,62 @@ +// +// WidgetWritingIntent.swift +// Home +// +// Created by 김지수 on 11/12/24. +// Copyright © 2024 com.weave. All rights reserved. +// + +import Foundation +import CommonKit +import CoreKit +import Model + +//MARK: - Intent +class WidgetWritingIntent { + private weak var model: WidgetWritingModelActionable? + private let input: DataModel + + // MARK: Life cycle + init( + model: WidgetWritingModelActionable, + input: DataModel + ) { + self.input = input + self.model = model + model.setWidgetType(input.widgetType) + } +} + +//MARK: - Intentable +extension WidgetWritingIntent { + protocol Intentable { + // content + func onChangedBodyText(_ text: String, maxCount: Int) + func onTapNextButton() + + // default + func onAppear() + func task() async + } + + struct DataModel { + let widgetType: WidgetType + } +} + +//MARK: - Intentable +extension WidgetWritingIntent: WidgetWritingIntent.Intentable { + // default + func onChangedBodyText(_ text: String, maxCount: Int) { + if text.count > maxCount { + return + } + model?.setBodyText(text) + } + func onAppear() {} + + func task() async {} + + // content + func onTapNextButton() {} +} diff --git a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift new file mode 100644 index 0000000..4a7c93c --- /dev/null +++ b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift @@ -0,0 +1,101 @@ +// +// WidgetWritingModel.swift +// Home +// +// Created by 김지수 on 11/12/24. +// Copyright © 2024 com.weave. All rights reserved. +// + +import Foundation +import CommonKit +import CoreKit +import Model + +final class WidgetWritingModel: ObservableObject { + + //MARK: Stateful + protocol Stateful { + // content + var selectedWidgetType: WidgetType? { get } + var widgetBodyText: String { get set } + var isValidated: Bool { get } + var textMaxCount: Int { get } + + var isModalPresented: Bool { get } + + // default + var isLoading: Bool { get } + + // error + var showErrorView: ErrorModel? { get } + var showErrorAlert: ErrorModel? { get } + } + + //MARK: State Properties + // content + @Published var selectedWidgetType: WidgetType? + @Published var widgetBodyText = String() + @Published var isModalPresented: Bool = true + var textMaxCount: Int = 40 + + @Published var isValidated: Bool = false + + // default + @Published var isLoading: Bool = false + + // error + @Published var showErrorView: ErrorModel? + @Published var showErrorAlert: ErrorModel? +} + +extension WidgetWritingModel: WidgetWritingModel.Stateful {} + +//MARK: - Actionable +protocol WidgetWritingModelActionable: AnyObject { + // content + func setBodyText(_ text: String) + func setValidation(value: Bool) + func setWidgetType(_ widget: WidgetType) + func modalDismiss() + + // default + func setLoading(status: Bool) + + // error + func showErrorView(error: ErrorModel) + func showErrorAlert(error: ErrorModel) + func resetError() +} + +extension WidgetWritingModel: WidgetWritingModelActionable { + // content + func setBodyText(_ text: String) { + widgetBodyText = text + } + func setValidation(value: Bool) { + isValidated = value + } + func setWidgetType(_ widget: WidgetType) { + selectedWidgetType = widget + } + func modalDismiss() { + isModalPresented = false + } + + // default + func setLoading(status: Bool) { + isLoading = status + } + + // error + func showErrorView(error: ErrorModel) { + showErrorView = error + } + func showErrorAlert(error: ErrorModel) { + showErrorAlert = error + } + func resetError() { + showErrorView = nil + showErrorAlert = nil + } +} diff --git a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift new file mode 100644 index 0000000..f1c2cd9 --- /dev/null +++ b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift @@ -0,0 +1,111 @@ +// +// WidgetWritingView.swift +// Home +// +// Created by 김지수 on 11/12/24. +// Copyright © 2024 com.weave. All rights reserved. +// + +import SwiftUI +import CoreKit +import DesignCore +import CommonKit +import Model + +public struct WidgetWritingView: View { + + @Binding var isPushed: Bool + @Binding var isModalPresented: Bool + + @StateObject var container: MVIContainer + + private var intent: WidgetWritingIntent.Intentable { container.intent } + private var state: WidgetWritingModel.Stateful { container.model } + + private var size: CGFloat { + Device.height * 0.3 + } + + public init( + widgetType: WidgetType, + isModalPresented: Binding, + isPushed: Binding + ) { + let model = WidgetWritingModel() + let intent = WidgetWritingIntent( + model: model, + input: .init(widgetType: widgetType) + ) + let container = MVIContainer( + intent: intent as WidgetWritingIntent.Intentable, + model: model as WidgetWritingModel.Stateful, + modelChangePublisher: model.objectWillChange + ) + self._container = StateObject(wrappedValue: container) + self._isModalPresented = isModalPresented + self._isPushed = isPushed + } + + public var body: some View { + VStack { + if let widget = state.selectedWidgetType { + Text("최대 40자 이내로 자유롭게 작성해주세요!") + .typography(.regular_14) + .padding(.vertical, 20) + + WritableProfileWidgetView( + title: widget.title, + placeholder: widget.exampleText, + bodyText: $container.model.widgetBodyText, + titleColor: widget.titleColor, + bodyColor: widget.bodyColor, + gradientColors: widget.gradationColors + ) + .frame(width: size, height: size) + .onChange(of: state.widgetBodyText) { + intent.onChangedBodyText( + bodyText, + maxCount: state.textMaxCount + ) + } + } + + HStack(spacing: 0) { + Text(String(state.widgetBodyText.count)) + .foregroundStyle(DesignCore.Colors.blue300) + Text("/\(state.textMaxCount)") + .foregroundStyle(DesignCore.Colors.grey300) + } + .typography(.regular_15) + Spacer() + } + .onChange(of: state.isModalPresented) { + isModalPresented = state.isModalPresented + } + .task { + await intent.task() + } + .onAppear { + intent.onAppear() + } + .ignoresSafeArea(.keyboard) + .navigationTitle(state.selectedWidgetType?.title ?? "") + .navigationBarTitleDisplayMode(.inline) + .setNavigation( + showLeftBackButton: true, + handler: { + isPushed = false + }) + .setLoading(state.isLoading) + } +} + +#Preview { + NavigationView { + WidgetWritingView( + widgetType: .body, + isModalPresented: .constant(false), + isPushed: .constant(false) + ) + } +} From 9e8431948e9298927321d601d45dd1ecb5a60a12 Mon Sep 17 00:00:00 2001 From: Jisu Kim Date: Thu, 14 Nov 2024 00:58:19 +0900 Subject: [PATCH 04/11] =?UTF-8?q?[WEAV-122]=20=ED=94=84=EB=A1=9C=ED=95=84?= =?UTF-8?q?=20=EC=9C=84=EC=A0=AF=20=EC=9E=85=EB=A0=A5=20=EB=B7=B0=20?= =?UTF-8?q?=EC=83=81=EC=84=B8=20=EA=B5=AC=ED=98=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Core/CoreKit/Sources/String+Ext.swift | 9 +++++ .../Sources/FlatTextFieldOption.swift | 29 ++++++++++++++ .../WritableProfileWidgetView.swift | 38 +++++++++++++------ .../Home/Sources/Profile/ProfileView.swift | 2 +- .../WidgetWriting/WidgetWritingIntent.swift | 21 ++++++---- .../WidgetWriting/WidgetWritingModel.swift | 11 ++++++ .../WidgetWriting/WidgetWritingView.swift | 20 ++++++++-- .../AuthName/AuthNameInputView.swift | 7 +--- .../Sources/Profile/Widget/WidgetType.swift | 19 ++++++++++ 9 files changed, 127 insertions(+), 29 deletions(-) create mode 100644 Projects/DesignSystem/DesignCore/Sources/FlatTextFieldOption.swift diff --git a/Projects/Core/CoreKit/Sources/String+Ext.swift b/Projects/Core/CoreKit/Sources/String+Ext.swift index 5bd0b62..8334599 100644 --- a/Projects/Core/CoreKit/Sources/String+Ext.swift +++ b/Projects/Core/CoreKit/Sources/String+Ext.swift @@ -34,4 +34,13 @@ extension String { return "\(firstPart)-\(secondPart)-\(thirdPart)" } } + + public func clipMaxCount(_ maxCount: Int) -> String { + if self.count > maxCount { + let index = self.index(self.startIndex, offsetBy: maxCount) + let formattedText = String(self[.. some View { + content + .keyboardType(keyboardType) + .interactiveDismissDisabled() + .textInputAutocapitalization(.never) + .autocorrectionDisabled() + .speechAnnouncementsQueued(false) + .speechSpellsOutCharacters(false) + } +} + +extension View { + public func flatTextFieldOption(keyboardType: UIKeyboardType = .default) -> some View { + modifier(FlatTextFieldOptionModifier(keyboardType: keyboardType)) + } +} diff --git a/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/WritableProfileWidgetView.swift b/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/WritableProfileWidgetView.swift index 3bc0370..99c1926 100644 --- a/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/WritableProfileWidgetView.swift +++ b/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/WritableProfileWidgetView.swift @@ -9,6 +9,7 @@ import SwiftUI public struct WritableProfileWidgetView: View { + @FocusState private var isfocused: Bool @Binding public var bodyText: String public let title: String public let placeholder: String @@ -22,7 +23,8 @@ public struct WritableProfileWidgetView: View { bodyText: Binding, titleColor: Color, bodyColor: Color, - gradientColors: [Color] + gradientColors: [Color], + focusState: FocusState = .init() ) { self.title = title self.placeholder = placeholder @@ -30,6 +32,7 @@ public struct WritableProfileWidgetView: View { self.titleColor = titleColor self.bodyColor = bodyColor self.gradientColors = gradientColors + self._isfocused = focusState } public var body: some View { @@ -50,14 +53,27 @@ public struct WritableProfileWidgetView: View { Spacer() } Spacer() - TextEditor( - text: $bodyText - ) - .textEditorStyle( - PlainTextEditorStyle() - ) - .typography(.regular_14) - .foregroundStyle(bodyColor) + ZStack(alignment: .topLeading) { + if bodyText.isEmpty { + Text(placeholder) + .typography(.regular_14) + .foregroundStyle( + Color(hex: 0x15394B4D).opacity(0.3) + ) + .padding(.all, 8) + } + + TextEditor( + text: $bodyText + ) + .textEditorStyle( + PlainTextEditorStyle() + ) + .focused($isfocused) + .flatTextFieldOption() + .typography(.regular_14) + .foregroundStyle(bodyColor) + } } .scrollIndicators(.hidden) .padding(.all, 28) @@ -71,13 +87,13 @@ struct PreviewView: View { var body: some View { WritableProfileWidgetView( title: "Title", - placeholder: "Placeholder", + placeholder: "this is Placeholder hahaha dhdhdh 바래보아요", bodyText: $text, titleColor: .black, bodyColor: .red, gradientColors: [.yellow, .green] ) - .frame(width: 200, height: 200) + .frame(width: 300, height: 300) } } diff --git a/Projects/Features/Home/Sources/Profile/ProfileView.swift b/Projects/Features/Home/Sources/Profile/ProfileView.swift index 2e293d6..b49abdc 100644 --- a/Projects/Features/Home/Sources/Profile/ProfileView.swift +++ b/Projects/Features/Home/Sources/Profile/ProfileView.swift @@ -112,7 +112,7 @@ public struct ProfileView: View { .sheet( isPresented: $isPresentWidgetSelectionView, content: { - NavigationView { + NavigationStack { WidgetSelectionView( isPresented: $isPresentWidgetSelectionView ) diff --git a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift index 7af73a8..63f7adc 100644 --- a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift +++ b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift @@ -32,7 +32,7 @@ extension WidgetWritingIntent { protocol Intentable { // content func onChangedBodyText(_ text: String, maxCount: Int) - func onTapNextButton() + func onTapNextButton(state: WidgetWritingModel.Stateful) // default func onAppear() @@ -48,15 +48,22 @@ extension WidgetWritingIntent { extension WidgetWritingIntent: WidgetWritingIntent.Intentable { // default func onChangedBodyText(_ text: String, maxCount: Int) { - if text.count > maxCount { - return - } - model?.setBodyText(text) + let formattedText = text.clipMaxCount(maxCount) + model?.setBodyText(formattedText) + } + func onAppear() { + model?.setFocusState(true) } - func onAppear() {} func task() async {} // content - func onTapNextButton() {} + func onTapNextButton(state: any WidgetWritingModel.Stateful) { + guard let selectedWidget = state.selectedWidgetType else { return } + // 창닫기 + model?.modalDismiss() + + } + + } diff --git a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift index 4a7c93c..d5cd728 100644 --- a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift +++ b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift @@ -22,6 +22,8 @@ final class WidgetWritingModel: ObservableObject { var textMaxCount: Int { get } var isModalPresented: Bool { get } + var isFocused: Bool { get } + var isCTAButtonEnabled: Bool { get } // default var isLoading: Bool { get } @@ -36,8 +38,13 @@ final class WidgetWritingModel: ObservableObject { @Published var selectedWidgetType: WidgetType? @Published var widgetBodyText = String() @Published var isModalPresented: Bool = true + @Published var isFocused: Bool = false var textMaxCount: Int = 40 + var isCTAButtonEnabled: Bool { + widgetBodyText.count > 4 + } + @Published var isValidated: Bool = false // default @@ -53,6 +60,7 @@ extension WidgetWritingModel: WidgetWritingModel.Stateful {} //MARK: - Actionable protocol WidgetWritingModelActionable: AnyObject { // content + func setFocusState(_ isFocused: Bool) func setBodyText(_ text: String) func setValidation(value: Bool) func setWidgetType(_ widget: WidgetType) @@ -69,6 +77,9 @@ protocol WidgetWritingModelActionable: AnyObject { extension WidgetWritingModel: WidgetWritingModelActionable { // content + func setFocusState(_ isFocused: Bool) { + self.isFocused = isFocused + } func setBodyText(_ text: String) { widgetBodyText = text } diff --git a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift index f1c2cd9..7109ffd 100644 --- a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift +++ b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift @@ -16,6 +16,7 @@ public struct WidgetWritingView: View { @Binding var isPushed: Bool @Binding var isModalPresented: Bool + @FocusState var isFocused: Bool @StateObject var container: MVIContainer @@ -23,7 +24,7 @@ public struct WidgetWritingView: View { private var state: WidgetWritingModel.Stateful { container.model } private var size: CGFloat { - Device.height * 0.3 + Device.height * 0.25 } public init( @@ -54,20 +55,24 @@ public struct WidgetWritingView: View { .padding(.vertical, 20) WritableProfileWidgetView( - title: widget.title, + title: widget.title + widget.emoji, placeholder: widget.exampleText, bodyText: $container.model.widgetBodyText, titleColor: widget.titleColor, bodyColor: widget.bodyColor, - gradientColors: widget.gradationColors + gradientColors: widget.gradationColors, + focusState: _isFocused ) .frame(width: size, height: size) .onChange(of: state.widgetBodyText) { intent.onChangedBodyText( - bodyText, + state.widgetBodyText, maxCount: state.textMaxCount ) } + .onChange(of: state.isFocused) { + self.isFocused = state.isFocused + } } HStack(spacing: 0) { @@ -78,6 +83,13 @@ public struct WidgetWritingView: View { } .typography(.regular_15) Spacer() + + CTABottomButton( + title: "다 썻어요", + isActive: state.isCTAButtonEnabled + ) { + intent.onTapNextButton(state: state) + } } .onChange(of: state.isModalPresented) { isModalPresented = state.isModalPresented diff --git a/Projects/Features/SignUp/Sources/ProfileInput/AuthName/AuthNameInputView.swift b/Projects/Features/SignUp/Sources/ProfileInput/AuthName/AuthNameInputView.swift index fddf345..2aa4a6a 100644 --- a/Projects/Features/SignUp/Sources/ProfileInput/AuthName/AuthNameInputView.swift +++ b/Projects/Features/SignUp/Sources/ProfileInput/AuthName/AuthNameInputView.swift @@ -66,12 +66,7 @@ public struct AuthNameInputView: View { "김위브", text: $inputText ) - .keyboardType(.namePhonePad) - .interactiveDismissDisabled() - .textInputAutocapitalization(.never) - .autocorrectionDisabled() - .speechAnnouncementsQueued(false) - .speechSpellsOutCharacters(false) + .flatTextFieldOption(keyboardType: .namePhonePad) .multilineTextAlignment(.center) .pretendard(weight: ._400, size: 28) .foregroundStyle(DesignCore.Colors.grey500) diff --git a/Projects/Model/Model/Sources/Profile/Widget/WidgetType.swift b/Projects/Model/Model/Sources/Profile/Widget/WidgetType.swift index bd7c3e9..e570ee4 100644 --- a/Projects/Model/Model/Sources/Profile/Widget/WidgetType.swift +++ b/Projects/Model/Model/Sources/Profile/Widget/WidgetType.swift @@ -46,6 +46,25 @@ extension WidgetType { } } + public var emoji: String { + switch self { + case .hobby: return "🏃" + case .style: return "👖" + case .mbti: return "💭" + case .music: return "🎧" + case .body: return "💪" + case .food: return "🍔" + case .movie: return "🎬" + case .drama: return "📺" + case .book: return "📚" + case .travel: return "✈️" + case .alcohol: return "🍷" + case .marriage: return "💍" + case .religion: return "⛪" + case .smoking: return "🚬" + } + } + public var exampleText: String { switch self { case .hobby: From 7e33230cbdaf08bdc3e685fbb62b77d6355544bf Mon Sep 17 00:00:00 2001 From: Jisu Kim Date: Thu, 14 Nov 2024 01:57:50 +0900 Subject: [PATCH 05/11] =?UTF-8?q?[WEAV-122]=20=ED=94=84=EB=A1=9C=ED=95=84?= =?UTF-8?q?=20=EC=9C=84=EC=A0=AF=20=EC=9E=85=EB=A0=A5=20=EA=B4=80=EB=A0=A8?= =?UTF-8?q?=20=EB=84=A4=ED=8A=B8=EC=9B=8C=ED=81=AC=20=EB=A0=88=EC=9D=B4?= =?UTF-8?q?=EC=96=B4=20=EA=B5=AC=ED=98=84?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Sources/OpenapiGenerated/Client.swift | 434 +++++++++- .../Sources/OpenapiGenerated/Types.swift | 781 +++++++++++++++++- .../Sources/openapi-generator-cli/3days-oas | 2 +- .../CommonKit/Sources/AppCoordinator.swift | 8 + .../Sources/AuthService/AuthServiceMock.swift | 3 +- .../ProfileService/ProfileService.swift | 41 + .../Home/Sources/Profile/ProfileView.swift | 140 +++- .../WidgetWriting/WidgetWritingIntent.swift | 29 +- .../Model/Sources/Auth/Domain/UserInfo.swift | 34 +- .../Sources/Profile/Widget/WidgetType.swift | 41 + 10 files changed, 1450 insertions(+), 63 deletions(-) create mode 100644 Projects/Core/NetworkKit/Sources/ProfileService/ProfileService.swift diff --git a/OpenApiGenerator/Sources/OpenapiGenerated/Client.swift b/OpenApiGenerator/Sources/OpenapiGenerated/Client.swift index 7046fe9..fcd73aa 100644 --- a/OpenApiGenerator/Sources/OpenapiGenerated/Client.swift +++ b/OpenApiGenerator/Sources/OpenapiGenerated/Client.swift @@ -127,7 +127,10 @@ public struct Client: APIProtocol { default: return .undocumented( statusCode: response.status.code, - .init() + .init( + headerFields: response.headerFields, + body: responseBody + ) ) } } @@ -263,7 +266,10 @@ public struct Client: APIProtocol { default: return .undocumented( statusCode: response.status.code, - .init() + .init( + headerFields: response.headerFields, + body: responseBody + ) ) } } @@ -399,7 +405,10 @@ public struct Client: APIProtocol { default: return .undocumented( statusCode: response.status.code, - .init() + .init( + headerFields: response.headerFields, + body: responseBody + ) ) } } @@ -546,7 +555,10 @@ public struct Client: APIProtocol { default: return .undocumented( statusCode: response.status.code, - .init() + .init( + headerFields: response.headerFields, + body: responseBody + ) ) } } @@ -649,7 +661,394 @@ public struct Client: APIProtocol { default: return .undocumented( statusCode: response.status.code, - .init() + .init( + headerFields: response.headerFields, + body: responseBody + ) + ) + } + } + ) + } + /// 내 프로필 수정 + /// + /// 현재 로그인한 사용자의 프로필 정보를 수정합니다. (이름, 직군, 직장, 활동 지역) + /// + /// - Remark: HTTP `PATCH /users/my`. + /// - Remark: Generated from `#/paths//users/my/patch(updateMyUserInfo)`. + public func updateMyUserInfo(_ input: Operations.updateMyUserInfo.Input) async throws -> Operations.updateMyUserInfo.Output { + try await client.send( + input: input, + forOperation: Operations.updateMyUserInfo.id, + serializer: { input in + let path = try converter.renderedPath( + template: "/users/my", + parameters: [] + ) + var request: HTTPTypes.HTTPRequest = .init( + soar_path: path, + method: .patch + ) + suppressMutabilityWarning(&request) + converter.setAcceptHeader( + in: &request.headerFields, + contentTypes: input.headers.accept + ) + let body: OpenAPIRuntime.HTTPBody? + switch input.body { + case let .json(value): + body = try converter.setRequiredRequestBodyAsJSON( + value, + headerFields: &request.headerFields, + contentType: "application/json; charset=utf-8" + ) + } + return (request, body) + }, + deserializer: { response, responseBody in + switch response.status.code { + case 200: + let contentType = converter.extractContentTypeIfPresent(in: response.headerFields) + let body: Operations.updateMyUserInfo.Output.Ok.Body + let chosenContentType = try converter.bestContentType( + received: contentType, + options: [ + "application/json" + ] + ) + switch chosenContentType { + case "application/json": + body = try await converter.getResponseBodyAsJSON( + Components.Schemas.UpdateMyUserInfoResponse.self, + from: responseBody, + transforming: { value in + .json(value) + } + ) + default: + preconditionFailure("bestContentType chose an invalid content type.") + } + return .ok(.init(body: body)) + case 400: + let contentType = converter.extractContentTypeIfPresent(in: response.headerFields) + let body: Components.Responses.BadRequest.Body + let chosenContentType = try converter.bestContentType( + received: contentType, + options: [ + "application/json" + ] + ) + switch chosenContentType { + case "application/json": + body = try await converter.getResponseBodyAsJSON( + Components.Schemas.ErrorResponse.self, + from: responseBody, + transforming: { value in + .json(value) + } + ) + default: + preconditionFailure("bestContentType chose an invalid content type.") + } + return .badRequest(.init(body: body)) + case 401: + let contentType = converter.extractContentTypeIfPresent(in: response.headerFields) + let body: Components.Responses.Unauthorized.Body + let chosenContentType = try converter.bestContentType( + received: contentType, + options: [ + "application/json" + ] + ) + switch chosenContentType { + case "application/json": + body = try await converter.getResponseBodyAsJSON( + Components.Schemas.ErrorResponse.self, + from: responseBody, + transforming: { value in + .json(value) + } + ) + default: + preconditionFailure("bestContentType chose an invalid content type.") + } + return .unauthorized(.init(body: body)) + case 500: + let contentType = converter.extractContentTypeIfPresent(in: response.headerFields) + let body: Components.Responses.InternalServerError.Body + let chosenContentType = try converter.bestContentType( + received: contentType, + options: [ + "application/json" + ] + ) + switch chosenContentType { + case "application/json": + body = try await converter.getResponseBodyAsJSON( + Components.Schemas.ErrorResponse.self, + from: responseBody, + transforming: { value in + .json(value) + } + ) + default: + preconditionFailure("bestContentType chose an invalid content type.") + } + return .internalServerError(.init(body: body)) + default: + return .undocumented( + statusCode: response.status.code, + .init( + headerFields: response.headerFields, + body: responseBody + ) + ) + } + } + ) + } + /// 프로필 위젯 추가 및 수정 + /// + /// 현재 사용자의 프로필 위젯을 추가 및 수정합니다. + /// + /// - Remark: HTTP `PUT /users/profileWidgets`. + /// - Remark: Generated from `#/paths//users/profileWidgets/put(putProfileWidget)`. + public func putProfileWidget(_ input: Operations.putProfileWidget.Input) async throws -> Operations.putProfileWidget.Output { + try await client.send( + input: input, + forOperation: Operations.putProfileWidget.id, + serializer: { input in + let path = try converter.renderedPath( + template: "/users/profileWidgets", + parameters: [] + ) + var request: HTTPTypes.HTTPRequest = .init( + soar_path: path, + method: .put + ) + suppressMutabilityWarning(&request) + converter.setAcceptHeader( + in: &request.headerFields, + contentTypes: input.headers.accept + ) + let body: OpenAPIRuntime.HTTPBody? + switch input.body { + case let .json(value): + body = try converter.setRequiredRequestBodyAsJSON( + value, + headerFields: &request.headerFields, + contentType: "application/json; charset=utf-8" + ) + } + return (request, body) + }, + deserializer: { response, responseBody in + switch response.status.code { + case 200: + let contentType = converter.extractContentTypeIfPresent(in: response.headerFields) + let body: Operations.putProfileWidget.Output.Ok.Body + let chosenContentType = try converter.bestContentType( + received: contentType, + options: [ + "application/json" + ] + ) + switch chosenContentType { + case "application/json": + body = try await converter.getResponseBodyAsJSON( + Components.Schemas.PutProfileWidgetResponse.self, + from: responseBody, + transforming: { value in + .json(value) + } + ) + default: + preconditionFailure("bestContentType chose an invalid content type.") + } + return .ok(.init(body: body)) + case 400: + let contentType = converter.extractContentTypeIfPresent(in: response.headerFields) + let body: Components.Responses.BadRequest.Body + let chosenContentType = try converter.bestContentType( + received: contentType, + options: [ + "application/json" + ] + ) + switch chosenContentType { + case "application/json": + body = try await converter.getResponseBodyAsJSON( + Components.Schemas.ErrorResponse.self, + from: responseBody, + transforming: { value in + .json(value) + } + ) + default: + preconditionFailure("bestContentType chose an invalid content type.") + } + return .badRequest(.init(body: body)) + case 401: + let contentType = converter.extractContentTypeIfPresent(in: response.headerFields) + let body: Components.Responses.Unauthorized.Body + let chosenContentType = try converter.bestContentType( + received: contentType, + options: [ + "application/json" + ] + ) + switch chosenContentType { + case "application/json": + body = try await converter.getResponseBodyAsJSON( + Components.Schemas.ErrorResponse.self, + from: responseBody, + transforming: { value in + .json(value) + } + ) + default: + preconditionFailure("bestContentType chose an invalid content type.") + } + return .unauthorized(.init(body: body)) + case 500: + let contentType = converter.extractContentTypeIfPresent(in: response.headerFields) + let body: Components.Responses.InternalServerError.Body + let chosenContentType = try converter.bestContentType( + received: contentType, + options: [ + "application/json" + ] + ) + switch chosenContentType { + case "application/json": + body = try await converter.getResponseBodyAsJSON( + Components.Schemas.ErrorResponse.self, + from: responseBody, + transforming: { value in + .json(value) + } + ) + default: + preconditionFailure("bestContentType chose an invalid content type.") + } + return .internalServerError(.init(body: body)) + default: + return .undocumented( + statusCode: response.status.code, + .init( + headerFields: response.headerFields, + body: responseBody + ) + ) + } + } + ) + } + /// 프로필 위젯 삭제 + /// + /// 현재 사용자의 프로필 위젯을 삭제합니다. + /// + /// - Remark: HTTP `DELETE /users/profileWidgets/{type}`. + /// - Remark: Generated from `#/paths//users/profileWidgets/{type}/delete(deleteProfileWidget)`. + public func deleteProfileWidget(_ input: Operations.deleteProfileWidget.Input) async throws -> Operations.deleteProfileWidget.Output { + try await client.send( + input: input, + forOperation: Operations.deleteProfileWidget.id, + serializer: { input in + let path = try converter.renderedPath( + template: "/users/profileWidgets/{}", + parameters: [ + input.path._type + ] + ) + var request: HTTPTypes.HTTPRequest = .init( + soar_path: path, + method: .delete + ) + suppressMutabilityWarning(&request) + converter.setAcceptHeader( + in: &request.headerFields, + contentTypes: input.headers.accept + ) + return (request, nil) + }, + deserializer: { response, responseBody in + switch response.status.code { + case 204: + return .noContent(.init()) + case 400: + let contentType = converter.extractContentTypeIfPresent(in: response.headerFields) + let body: Components.Responses.BadRequest.Body + let chosenContentType = try converter.bestContentType( + received: contentType, + options: [ + "application/json" + ] + ) + switch chosenContentType { + case "application/json": + body = try await converter.getResponseBodyAsJSON( + Components.Schemas.ErrorResponse.self, + from: responseBody, + transforming: { value in + .json(value) + } + ) + default: + preconditionFailure("bestContentType chose an invalid content type.") + } + return .badRequest(.init(body: body)) + case 401: + let contentType = converter.extractContentTypeIfPresent(in: response.headerFields) + let body: Components.Responses.Unauthorized.Body + let chosenContentType = try converter.bestContentType( + received: contentType, + options: [ + "application/json" + ] + ) + switch chosenContentType { + case "application/json": + body = try await converter.getResponseBodyAsJSON( + Components.Schemas.ErrorResponse.self, + from: responseBody, + transforming: { value in + .json(value) + } + ) + default: + preconditionFailure("bestContentType chose an invalid content type.") + } + return .unauthorized(.init(body: body)) + case 500: + let contentType = converter.extractContentTypeIfPresent(in: response.headerFields) + let body: Components.Responses.InternalServerError.Body + let chosenContentType = try converter.bestContentType( + received: contentType, + options: [ + "application/json" + ] + ) + switch chosenContentType { + case "application/json": + body = try await converter.getResponseBodyAsJSON( + Components.Schemas.ErrorResponse.self, + from: responseBody, + transforming: { value in + .json(value) + } + ) + default: + preconditionFailure("bestContentType chose an invalid content type.") + } + return .internalServerError(.init(body: body)) + default: + return .undocumented( + statusCode: response.status.code, + .init( + headerFields: response.headerFields, + body: responseBody + ) ) } } @@ -783,7 +1182,10 @@ public struct Client: APIProtocol { default: return .undocumented( statusCode: response.status.code, - .init() + .init( + headerFields: response.headerFields, + body: responseBody + ) ) } } @@ -864,7 +1266,10 @@ public struct Client: APIProtocol { default: return .undocumented( statusCode: response.status.code, - .init() + .init( + headerFields: response.headerFields, + body: responseBody + ) ) } } @@ -969,7 +1374,10 @@ public struct Client: APIProtocol { default: return .undocumented( statusCode: response.status.code, - .init() + .init( + headerFields: response.headerFields, + body: responseBody + ) ) } } @@ -1071,7 +1479,10 @@ public struct Client: APIProtocol { default: return .undocumented( statusCode: response.status.code, - .init() + .init( + headerFields: response.headerFields, + body: responseBody + ) ) } } @@ -1176,7 +1587,10 @@ public struct Client: APIProtocol { default: return .undocumented( statusCode: response.status.code, - .init() + .init( + headerFields: response.headerFields, + body: responseBody + ) ) } } diff --git a/OpenApiGenerator/Sources/OpenapiGenerated/Types.swift b/OpenApiGenerator/Sources/OpenapiGenerated/Types.swift index 08622f3..5312f55 100644 --- a/OpenApiGenerator/Sources/OpenapiGenerated/Types.swift +++ b/OpenApiGenerator/Sources/OpenapiGenerated/Types.swift @@ -46,6 +46,27 @@ public protocol APIProtocol: Sendable { /// - Remark: HTTP `GET /users/my`. /// - Remark: Generated from `#/paths//users/my/get(getMyUserInfo)`. func getMyUserInfo(_ input: Operations.getMyUserInfo.Input) async throws -> Operations.getMyUserInfo.Output + /// 내 프로필 수정 + /// + /// 현재 로그인한 사용자의 프로필 정보를 수정합니다. (이름, 직군, 직장, 활동 지역) + /// + /// - Remark: HTTP `PATCH /users/my`. + /// - Remark: Generated from `#/paths//users/my/patch(updateMyUserInfo)`. + func updateMyUserInfo(_ input: Operations.updateMyUserInfo.Input) async throws -> Operations.updateMyUserInfo.Output + /// 프로필 위젯 추가 및 수정 + /// + /// 현재 사용자의 프로필 위젯을 추가 및 수정합니다. + /// + /// - Remark: HTTP `PUT /users/profileWidgets`. + /// - Remark: Generated from `#/paths//users/profileWidgets/put(putProfileWidget)`. + func putProfileWidget(_ input: Operations.putProfileWidget.Input) async throws -> Operations.putProfileWidget.Output + /// 프로필 위젯 삭제 + /// + /// 현재 사용자의 프로필 위젯을 삭제합니다. + /// + /// - Remark: HTTP `DELETE /users/profileWidgets/{type}`. + /// - Remark: Generated from `#/paths//users/profileWidgets/{type}/delete(deleteProfileWidget)`. + func deleteProfileWidget(_ input: Operations.deleteProfileWidget.Input) async throws -> Operations.deleteProfileWidget.Output /// 액세스 토큰 갱신 /// /// 리프레시 토큰을 사용하여 새로운 액세스 토큰을 발급받습니다. @@ -158,6 +179,51 @@ extension APIProtocol { public func getMyUserInfo(headers: Operations.getMyUserInfo.Input.Headers = .init()) async throws -> Operations.getMyUserInfo.Output { try await getMyUserInfo(Operations.getMyUserInfo.Input(headers: headers)) } + /// 내 프로필 수정 + /// + /// 현재 로그인한 사용자의 프로필 정보를 수정합니다. (이름, 직군, 직장, 활동 지역) + /// + /// - Remark: HTTP `PATCH /users/my`. + /// - Remark: Generated from `#/paths//users/my/patch(updateMyUserInfo)`. + public func updateMyUserInfo( + headers: Operations.updateMyUserInfo.Input.Headers = .init(), + body: Operations.updateMyUserInfo.Input.Body + ) async throws -> Operations.updateMyUserInfo.Output { + try await updateMyUserInfo(Operations.updateMyUserInfo.Input( + headers: headers, + body: body + )) + } + /// 프로필 위젯 추가 및 수정 + /// + /// 현재 사용자의 프로필 위젯을 추가 및 수정합니다. + /// + /// - Remark: HTTP `PUT /users/profileWidgets`. + /// - Remark: Generated from `#/paths//users/profileWidgets/put(putProfileWidget)`. + public func putProfileWidget( + headers: Operations.putProfileWidget.Input.Headers = .init(), + body: Operations.putProfileWidget.Input.Body + ) async throws -> Operations.putProfileWidget.Output { + try await putProfileWidget(Operations.putProfileWidget.Input( + headers: headers, + body: body + )) + } + /// 프로필 위젯 삭제 + /// + /// 현재 사용자의 프로필 위젯을 삭제합니다. + /// + /// - Remark: HTTP `DELETE /users/profileWidgets/{type}`. + /// - Remark: Generated from `#/paths//users/profileWidgets/{type}/delete(deleteProfileWidget)`. + public func deleteProfileWidget( + path: Operations.deleteProfileWidget.Input.Path, + headers: Operations.deleteProfileWidget.Input.Headers = .init() + ) async throws -> Operations.deleteProfileWidget.Output { + try await deleteProfileWidget(Operations.deleteProfileWidget.Input( + path: path, + headers: headers + )) + } /// 액세스 토큰 갱신 /// /// 리프레시 토큰을 사용하여 새로운 액세스 토큰을 발급받습니다. @@ -305,7 +371,7 @@ public enum Components { /// 사용자 상태 (신규 사용자 또는 기존 사용자) /// /// - Remark: Generated from `#/components/schemas/SendAuthCodeResponse/userStatus`. - @frozen public enum userStatusPayload: String, Codable, Hashable, Sendable { + @frozen public enum userStatusPayload: String, Codable, Hashable, Sendable, CaseIterable { case NEW = "NEW" case EXISTING = "EXISTING" } @@ -480,6 +546,8 @@ public enum Components { public var profile: Components.Schemas.UserProfile /// - Remark: Generated from `#/components/schemas/GetMyUserInfoResponse/desiredPartner`. public var desiredPartner: Components.Schemas.UserDesiredPartner + /// - Remark: Generated from `#/components/schemas/GetMyUserInfoResponse/profileWidgets`. + public var profileWidgets: [Components.Schemas.ProfileWidget] /// Creates a new `GetMyUserInfoResponse`. /// /// - Parameters: @@ -488,6 +556,100 @@ public enum Components { /// - phoneNumber: 사용자의 전화번호 (한국 휴대폰 번호 형식) /// - profile: /// - desiredPartner: + /// - profileWidgets: + public init( + id: Swift.String? = nil, + name: Swift.String, + phoneNumber: Swift.String, + profile: Components.Schemas.UserProfile, + desiredPartner: Components.Schemas.UserDesiredPartner, + profileWidgets: [Components.Schemas.ProfileWidget] + ) { + self.id = id + self.name = name + self.phoneNumber = phoneNumber + self.profile = profile + self.desiredPartner = desiredPartner + self.profileWidgets = profileWidgets + } + public enum CodingKeys: String, CodingKey { + case id + case name + case phoneNumber + case profile + case desiredPartner + case profileWidgets + } + } + /// 현재 사용자 프로필 수정 요청 (이름, 직군, 직장, 활동 지역) + /// + /// - Remark: Generated from `#/components/schemas/UpdateMyUserInfoRequest`. + public struct UpdateMyUserInfoRequest: Codable, Hashable, Sendable { + /// 사용자 이름 + /// + /// - Remark: Generated from `#/components/schemas/UpdateMyUserInfoRequest/name`. + public var name: Swift.String? + /// - Remark: Generated from `#/components/schemas/UpdateMyUserInfoRequest/jobOccupation`. + public var jobOccupation: Components.Schemas.JobOccupation? + /// 사용자의 회사 ID + /// + /// - Remark: Generated from `#/components/schemas/UpdateMyUserInfoRequest/companyId`. + public var companyId: Swift.String? + /// 사용자의 활동 지역 목록 ID 리스트 + /// + /// - Remark: Generated from `#/components/schemas/UpdateMyUserInfoRequest/locationIds`. + public var locationIds: [Swift.String]? + /// Creates a new `UpdateMyUserInfoRequest`. + /// + /// - Parameters: + /// - name: 사용자 이름 + /// - jobOccupation: + /// - companyId: 사용자의 회사 ID + /// - locationIds: 사용자의 활동 지역 목록 ID 리스트 + public init( + name: Swift.String? = nil, + jobOccupation: Components.Schemas.JobOccupation? = nil, + companyId: Swift.String? = nil, + locationIds: [Swift.String]? = nil + ) { + self.name = name + self.jobOccupation = jobOccupation + self.companyId = companyId + self.locationIds = locationIds + } + public enum CodingKeys: String, CodingKey { + case name + case jobOccupation + case companyId + case locationIds + } + } + /// - Remark: Generated from `#/components/schemas/UpdateMyUserInfoResponse`. + public struct UpdateMyUserInfoResponse: Codable, Hashable, Sendable { + /// 사용자 식별자 + /// + /// - Remark: Generated from `#/components/schemas/UpdateMyUserInfoResponse/id`. + public var id: Swift.String? + /// 사용자 이름 + /// + /// - Remark: Generated from `#/components/schemas/UpdateMyUserInfoResponse/name`. + public var name: Swift.String + /// 사용자의 전화번호 (한국 휴대폰 번호 형식) + /// + /// - Remark: Generated from `#/components/schemas/UpdateMyUserInfoResponse/phoneNumber`. + public var phoneNumber: Swift.String + /// - Remark: Generated from `#/components/schemas/UpdateMyUserInfoResponse/profile`. + public var profile: Components.Schemas.UserProfile + /// - Remark: Generated from `#/components/schemas/UpdateMyUserInfoResponse/desiredPartner`. + public var desiredPartner: Components.Schemas.UserDesiredPartner + /// Creates a new `UpdateMyUserInfoResponse`. + /// + /// - Parameters: + /// - id: 사용자 식별자 + /// - name: 사용자 이름 + /// - phoneNumber: 사용자의 전화번호 (한국 휴대폰 번호 형식) + /// - profile: + /// - desiredPartner: public init( id: Swift.String? = nil, name: Swift.String, @@ -783,21 +945,21 @@ public enum Components { /// 사용자의 운영 체제 유형 /// /// - Remark: Generated from `#/components/schemas/OSType`. - @frozen public enum OSType: String, Codable, Hashable, Sendable { + @frozen public enum OSType: String, Codable, Hashable, Sendable, CaseIterable { case IOS = "IOS" case AOS = "AOS" } /// 사용자의 성별 /// /// - Remark: Generated from `#/components/schemas/Gender`. - @frozen public enum Gender: String, Codable, Hashable, Sendable { + @frozen public enum Gender: String, Codable, Hashable, Sendable, CaseIterable { case MALE = "MALE" case FEMALE = "FEMALE" } /// 직업군 분류 /// /// - Remark: Generated from `#/components/schemas/JobOccupation`. - @frozen public enum JobOccupation: String, Codable, Hashable, Sendable { + @frozen public enum JobOccupation: String, Codable, Hashable, Sendable, CaseIterable { case BUSINESS_ADMIN = "BUSINESS_ADMIN" case SALES_MARKETING = "SALES_MARKETING" case RESEARCH_DEVELOPMENT = "RESEARCH_DEVELOPMENT" @@ -822,11 +984,59 @@ public enum Components { /// 선호하는 거리 (내 지역만, 주변 지역 포함, 어디든) /// /// - Remark: Generated from `#/components/schemas/PreferDistance`. - @frozen public enum PreferDistance: String, Codable, Hashable, Sendable { + @frozen public enum PreferDistance: String, Codable, Hashable, Sendable, CaseIterable { case ONLY_MY_AREA = "ONLY_MY_AREA" case INCLUDE_SURROUNDING_REGIONS = "INCLUDE_SURROUNDING_REGIONS" case ANYWHERE = "ANYWHERE" } + /// - Remark: Generated from `#/components/schemas/PutProfileWidgetRequest`. + public typealias PutProfileWidgetRequest = Components.Schemas.ProfileWidget + /// - Remark: Generated from `#/components/schemas/PutProfileWidgetResponse`. + public typealias PutProfileWidgetResponse = Components.Schemas.ProfileWidget + /// - Remark: Generated from `#/components/schemas/ProfileWidget`. + public struct ProfileWidget: Codable, Hashable, Sendable { + /// - Remark: Generated from `#/components/schemas/ProfileWidget/type`. + public var _type: Components.Schemas.ProfileWidgetType + /// 위젯 내용 + /// + /// - Remark: Generated from `#/components/schemas/ProfileWidget/content`. + public var content: Swift.String + /// Creates a new `ProfileWidget`. + /// + /// - Parameters: + /// - _type: + /// - content: 위젯 내용 + public init( + _type: Components.Schemas.ProfileWidgetType, + content: Swift.String + ) { + self._type = _type + self.content = content + } + public enum CodingKeys: String, CodingKey { + case _type = "type" + case content + } + } + /// 프로필 위젯 타입 + /// + /// - Remark: Generated from `#/components/schemas/ProfileWidgetType`. + @frozen public enum ProfileWidgetType: String, Codable, Hashable, Sendable, CaseIterable { + case HOBBY = "HOBBY" + case STYLE = "STYLE" + case MBTI = "MBTI" + case MUSIC = "MUSIC" + case BODY_TYPE = "BODY_TYPE" + case FOOD = "FOOD" + case MOVIE = "MOVIE" + case DRAMA = "DRAMA" + case BOOK = "BOOK" + case TRAVEL = "TRAVEL" + case DRINKING = "DRINKING" + case MARRIAGE = "MARRIAGE" + case RELIGION = "RELIGION" + case SMOKING = "SMOKING" + } } /// Types generated from the `#/components/parameters` section of the OpenAPI document. public enum Parameters { @@ -2021,6 +2231,567 @@ public enum Operations { } } } + /// 내 프로필 수정 + /// + /// 현재 로그인한 사용자의 프로필 정보를 수정합니다. (이름, 직군, 직장, 활동 지역) + /// + /// - Remark: HTTP `PATCH /users/my`. + /// - Remark: Generated from `#/paths//users/my/patch(updateMyUserInfo)`. + public enum updateMyUserInfo { + public static let id: Swift.String = "updateMyUserInfo" + public struct Input: Sendable, Hashable { + /// - Remark: Generated from `#/paths/users/my/PATCH/header`. + public struct Headers: Sendable, Hashable { + public var accept: [OpenAPIRuntime.AcceptHeaderContentType] + /// Creates a new `Headers`. + /// + /// - Parameters: + /// - accept: + public init(accept: [OpenAPIRuntime.AcceptHeaderContentType] = .defaultValues()) { + self.accept = accept + } + } + public var headers: Operations.updateMyUserInfo.Input.Headers + /// - Remark: Generated from `#/paths/users/my/PATCH/requestBody`. + @frozen public enum Body: Sendable, Hashable { + /// - Remark: Generated from `#/paths/users/my/PATCH/requestBody/content/application\/json`. + case json(Components.Schemas.UpdateMyUserInfoRequest) + } + public var body: Operations.updateMyUserInfo.Input.Body + /// Creates a new `Input`. + /// + /// - Parameters: + /// - headers: + /// - body: + public init( + headers: Operations.updateMyUserInfo.Input.Headers = .init(), + body: Operations.updateMyUserInfo.Input.Body + ) { + self.headers = headers + self.body = body + } + } + @frozen public enum Output: Sendable, Hashable { + public struct Ok: Sendable, Hashable { + /// - Remark: Generated from `#/paths/users/my/PATCH/responses/200/content`. + @frozen public enum Body: Sendable, Hashable { + /// - Remark: Generated from `#/paths/users/my/PATCH/responses/200/content/application\/json`. + case json(Components.Schemas.UpdateMyUserInfoResponse) + /// The associated value of the enum case if `self` is `.json`. + /// + /// - Throws: An error if `self` is not `.json`. + /// - SeeAlso: `.json`. + public var json: Components.Schemas.UpdateMyUserInfoResponse { + get throws { + switch self { + case let .json(body): + return body + } + } + } + } + /// Received HTTP response body + public var body: Operations.updateMyUserInfo.Output.Ok.Body + /// Creates a new `Ok`. + /// + /// - Parameters: + /// - body: Received HTTP response body + public init(body: Operations.updateMyUserInfo.Output.Ok.Body) { + self.body = body + } + } + /// 수정 성공 + /// + /// - Remark: Generated from `#/paths//users/my/patch(updateMyUserInfo)/responses/200`. + /// + /// HTTP response code: `200 ok`. + case ok(Operations.updateMyUserInfo.Output.Ok) + /// The associated value of the enum case if `self` is `.ok`. + /// + /// - Throws: An error if `self` is not `.ok`. + /// - SeeAlso: `.ok`. + public var ok: Operations.updateMyUserInfo.Output.Ok { + get throws { + switch self { + case let .ok(response): + return response + default: + try throwUnexpectedResponseStatus( + expectedStatus: "ok", + response: self + ) + } + } + } + /// 잘못된 요청 + /// + /// - Remark: Generated from `#/paths//users/my/patch(updateMyUserInfo)/responses/400`. + /// + /// HTTP response code: `400 badRequest`. + case badRequest(Components.Responses.BadRequest) + /// The associated value of the enum case if `self` is `.badRequest`. + /// + /// - Throws: An error if `self` is not `.badRequest`. + /// - SeeAlso: `.badRequest`. + public var badRequest: Components.Responses.BadRequest { + get throws { + switch self { + case let .badRequest(response): + return response + default: + try throwUnexpectedResponseStatus( + expectedStatus: "badRequest", + response: self + ) + } + } + } + /// 인증 실패 (토큰 만료 또는 유효하지 않은 토큰) + /// + /// - Remark: Generated from `#/paths//users/my/patch(updateMyUserInfo)/responses/401`. + /// + /// HTTP response code: `401 unauthorized`. + case unauthorized(Components.Responses.Unauthorized) + /// The associated value of the enum case if `self` is `.unauthorized`. + /// + /// - Throws: An error if `self` is not `.unauthorized`. + /// - SeeAlso: `.unauthorized`. + public var unauthorized: Components.Responses.Unauthorized { + get throws { + switch self { + case let .unauthorized(response): + return response + default: + try throwUnexpectedResponseStatus( + expectedStatus: "unauthorized", + response: self + ) + } + } + } + /// 서버 오류 + /// + /// - Remark: Generated from `#/paths//users/my/patch(updateMyUserInfo)/responses/500`. + /// + /// HTTP response code: `500 internalServerError`. + case internalServerError(Components.Responses.InternalServerError) + /// The associated value of the enum case if `self` is `.internalServerError`. + /// + /// - Throws: An error if `self` is not `.internalServerError`. + /// - SeeAlso: `.internalServerError`. + public var internalServerError: Components.Responses.InternalServerError { + get throws { + switch self { + case let .internalServerError(response): + return response + default: + try throwUnexpectedResponseStatus( + expectedStatus: "internalServerError", + response: self + ) + } + } + } + /// Undocumented response. + /// + /// A response with a code that is not documented in the OpenAPI document. + case undocumented(statusCode: Swift.Int, OpenAPIRuntime.UndocumentedPayload) + } + @frozen public enum AcceptableContentType: AcceptableProtocol { + case json + case other(Swift.String) + public init?(rawValue: Swift.String) { + switch rawValue.lowercased() { + case "application/json": + self = .json + default: + self = .other(rawValue) + } + } + public var rawValue: Swift.String { + switch self { + case let .other(string): + return string + case .json: + return "application/json" + } + } + public static var allCases: [Self] { + [ + .json + ] + } + } + } + /// 프로필 위젯 추가 및 수정 + /// + /// 현재 사용자의 프로필 위젯을 추가 및 수정합니다. + /// + /// - Remark: HTTP `PUT /users/profileWidgets`. + /// - Remark: Generated from `#/paths//users/profileWidgets/put(putProfileWidget)`. + public enum putProfileWidget { + public static let id: Swift.String = "putProfileWidget" + public struct Input: Sendable, Hashable { + /// - Remark: Generated from `#/paths/users/profileWidgets/PUT/header`. + public struct Headers: Sendable, Hashable { + public var accept: [OpenAPIRuntime.AcceptHeaderContentType] + /// Creates a new `Headers`. + /// + /// - Parameters: + /// - accept: + public init(accept: [OpenAPIRuntime.AcceptHeaderContentType] = .defaultValues()) { + self.accept = accept + } + } + public var headers: Operations.putProfileWidget.Input.Headers + /// - Remark: Generated from `#/paths/users/profileWidgets/PUT/requestBody`. + @frozen public enum Body: Sendable, Hashable { + /// - Remark: Generated from `#/paths/users/profileWidgets/PUT/requestBody/content/application\/json`. + case json(Components.Schemas.PutProfileWidgetRequest) + } + public var body: Operations.putProfileWidget.Input.Body + /// Creates a new `Input`. + /// + /// - Parameters: + /// - headers: + /// - body: + public init( + headers: Operations.putProfileWidget.Input.Headers = .init(), + body: Operations.putProfileWidget.Input.Body + ) { + self.headers = headers + self.body = body + } + } + @frozen public enum Output: Sendable, Hashable { + public struct Ok: Sendable, Hashable { + /// - Remark: Generated from `#/paths/users/profileWidgets/PUT/responses/200/content`. + @frozen public enum Body: Sendable, Hashable { + /// - Remark: Generated from `#/paths/users/profileWidgets/PUT/responses/200/content/application\/json`. + case json(Components.Schemas.PutProfileWidgetResponse) + /// The associated value of the enum case if `self` is `.json`. + /// + /// - Throws: An error if `self` is not `.json`. + /// - SeeAlso: `.json`. + public var json: Components.Schemas.PutProfileWidgetResponse { + get throws { + switch self { + case let .json(body): + return body + } + } + } + } + /// Received HTTP response body + public var body: Operations.putProfileWidget.Output.Ok.Body + /// Creates a new `Ok`. + /// + /// - Parameters: + /// - body: Received HTTP response body + public init(body: Operations.putProfileWidget.Output.Ok.Body) { + self.body = body + } + } + /// 프로필 위젯 추가/수정 성공 + /// + /// - Remark: Generated from `#/paths//users/profileWidgets/put(putProfileWidget)/responses/200`. + /// + /// HTTP response code: `200 ok`. + case ok(Operations.putProfileWidget.Output.Ok) + /// The associated value of the enum case if `self` is `.ok`. + /// + /// - Throws: An error if `self` is not `.ok`. + /// - SeeAlso: `.ok`. + public var ok: Operations.putProfileWidget.Output.Ok { + get throws { + switch self { + case let .ok(response): + return response + default: + try throwUnexpectedResponseStatus( + expectedStatus: "ok", + response: self + ) + } + } + } + /// 잘못된 요청 + /// + /// - Remark: Generated from `#/paths//users/profileWidgets/put(putProfileWidget)/responses/400`. + /// + /// HTTP response code: `400 badRequest`. + case badRequest(Components.Responses.BadRequest) + /// The associated value of the enum case if `self` is `.badRequest`. + /// + /// - Throws: An error if `self` is not `.badRequest`. + /// - SeeAlso: `.badRequest`. + public var badRequest: Components.Responses.BadRequest { + get throws { + switch self { + case let .badRequest(response): + return response + default: + try throwUnexpectedResponseStatus( + expectedStatus: "badRequest", + response: self + ) + } + } + } + /// 인증 실패 (토큰 만료 또는 유효하지 않은 토큰) + /// + /// - Remark: Generated from `#/paths//users/profileWidgets/put(putProfileWidget)/responses/401`. + /// + /// HTTP response code: `401 unauthorized`. + case unauthorized(Components.Responses.Unauthorized) + /// The associated value of the enum case if `self` is `.unauthorized`. + /// + /// - Throws: An error if `self` is not `.unauthorized`. + /// - SeeAlso: `.unauthorized`. + public var unauthorized: Components.Responses.Unauthorized { + get throws { + switch self { + case let .unauthorized(response): + return response + default: + try throwUnexpectedResponseStatus( + expectedStatus: "unauthorized", + response: self + ) + } + } + } + /// 서버 오류 + /// + /// - Remark: Generated from `#/paths//users/profileWidgets/put(putProfileWidget)/responses/500`. + /// + /// HTTP response code: `500 internalServerError`. + case internalServerError(Components.Responses.InternalServerError) + /// The associated value of the enum case if `self` is `.internalServerError`. + /// + /// - Throws: An error if `self` is not `.internalServerError`. + /// - SeeAlso: `.internalServerError`. + public var internalServerError: Components.Responses.InternalServerError { + get throws { + switch self { + case let .internalServerError(response): + return response + default: + try throwUnexpectedResponseStatus( + expectedStatus: "internalServerError", + response: self + ) + } + } + } + /// Undocumented response. + /// + /// A response with a code that is not documented in the OpenAPI document. + case undocumented(statusCode: Swift.Int, OpenAPIRuntime.UndocumentedPayload) + } + @frozen public enum AcceptableContentType: AcceptableProtocol { + case json + case other(Swift.String) + public init?(rawValue: Swift.String) { + switch rawValue.lowercased() { + case "application/json": + self = .json + default: + self = .other(rawValue) + } + } + public var rawValue: Swift.String { + switch self { + case let .other(string): + return string + case .json: + return "application/json" + } + } + public static var allCases: [Self] { + [ + .json + ] + } + } + } + /// 프로필 위젯 삭제 + /// + /// 현재 사용자의 프로필 위젯을 삭제합니다. + /// + /// - Remark: HTTP `DELETE /users/profileWidgets/{type}`. + /// - Remark: Generated from `#/paths//users/profileWidgets/{type}/delete(deleteProfileWidget)`. + public enum deleteProfileWidget { + public static let id: Swift.String = "deleteProfileWidget" + public struct Input: Sendable, Hashable { + /// - Remark: Generated from `#/paths/users/profileWidgets/{type}/DELETE/path`. + public struct Path: Sendable, Hashable { + /// 삭제할 프로필 위젯 타입 + /// + /// - Remark: Generated from `#/paths/users/profileWidgets/{type}/DELETE/path/type`. + public var _type: Components.Schemas.ProfileWidgetType + /// Creates a new `Path`. + /// + /// - Parameters: + /// - _type: 삭제할 프로필 위젯 타입 + public init(_type: Components.Schemas.ProfileWidgetType) { + self._type = _type + } + } + public var path: Operations.deleteProfileWidget.Input.Path + /// - Remark: Generated from `#/paths/users/profileWidgets/{type}/DELETE/header`. + public struct Headers: Sendable, Hashable { + public var accept: [OpenAPIRuntime.AcceptHeaderContentType] + /// Creates a new `Headers`. + /// + /// - Parameters: + /// - accept: + public init(accept: [OpenAPIRuntime.AcceptHeaderContentType] = .defaultValues()) { + self.accept = accept + } + } + public var headers: Operations.deleteProfileWidget.Input.Headers + /// Creates a new `Input`. + /// + /// - Parameters: + /// - path: + /// - headers: + public init( + path: Operations.deleteProfileWidget.Input.Path, + headers: Operations.deleteProfileWidget.Input.Headers = .init() + ) { + self.path = path + self.headers = headers + } + } + @frozen public enum Output: Sendable, Hashable { + public struct NoContent: Sendable, Hashable { + /// Creates a new `NoContent`. + public init() {} + } + /// 프로필 위젯 삭제 성공 + /// + /// - Remark: Generated from `#/paths//users/profileWidgets/{type}/delete(deleteProfileWidget)/responses/204`. + /// + /// HTTP response code: `204 noContent`. + case noContent(Operations.deleteProfileWidget.Output.NoContent) + /// The associated value of the enum case if `self` is `.noContent`. + /// + /// - Throws: An error if `self` is not `.noContent`. + /// - SeeAlso: `.noContent`. + public var noContent: Operations.deleteProfileWidget.Output.NoContent { + get throws { + switch self { + case let .noContent(response): + return response + default: + try throwUnexpectedResponseStatus( + expectedStatus: "noContent", + response: self + ) + } + } + } + /// 잘못된 요청 + /// + /// - Remark: Generated from `#/paths//users/profileWidgets/{type}/delete(deleteProfileWidget)/responses/400`. + /// + /// HTTP response code: `400 badRequest`. + case badRequest(Components.Responses.BadRequest) + /// The associated value of the enum case if `self` is `.badRequest`. + /// + /// - Throws: An error if `self` is not `.badRequest`. + /// - SeeAlso: `.badRequest`. + public var badRequest: Components.Responses.BadRequest { + get throws { + switch self { + case let .badRequest(response): + return response + default: + try throwUnexpectedResponseStatus( + expectedStatus: "badRequest", + response: self + ) + } + } + } + /// 인증 실패 (토큰 만료 또는 유효하지 않은 토큰) + /// + /// - Remark: Generated from `#/paths//users/profileWidgets/{type}/delete(deleteProfileWidget)/responses/401`. + /// + /// HTTP response code: `401 unauthorized`. + case unauthorized(Components.Responses.Unauthorized) + /// The associated value of the enum case if `self` is `.unauthorized`. + /// + /// - Throws: An error if `self` is not `.unauthorized`. + /// - SeeAlso: `.unauthorized`. + public var unauthorized: Components.Responses.Unauthorized { + get throws { + switch self { + case let .unauthorized(response): + return response + default: + try throwUnexpectedResponseStatus( + expectedStatus: "unauthorized", + response: self + ) + } + } + } + /// 서버 오류 + /// + /// - Remark: Generated from `#/paths//users/profileWidgets/{type}/delete(deleteProfileWidget)/responses/500`. + /// + /// HTTP response code: `500 internalServerError`. + case internalServerError(Components.Responses.InternalServerError) + /// The associated value of the enum case if `self` is `.internalServerError`. + /// + /// - Throws: An error if `self` is not `.internalServerError`. + /// - SeeAlso: `.internalServerError`. + public var internalServerError: Components.Responses.InternalServerError { + get throws { + switch self { + case let .internalServerError(response): + return response + default: + try throwUnexpectedResponseStatus( + expectedStatus: "internalServerError", + response: self + ) + } + } + } + /// Undocumented response. + /// + /// A response with a code that is not documented in the OpenAPI document. + case undocumented(statusCode: Swift.Int, OpenAPIRuntime.UndocumentedPayload) + } + @frozen public enum AcceptableContentType: AcceptableProtocol { + case json + case other(Swift.String) + public init?(rawValue: Swift.String) { + switch rawValue.lowercased() { + case "application/json": + self = .json + default: + self = .other(rawValue) + } + } + public var rawValue: Swift.String { + switch self { + case let .other(string): + return string + case .json: + return "application/json" + } + } + public static var allCases: [Self] { + [ + .json + ] + } + } + } /// 액세스 토큰 갱신 /// /// 리프레시 토큰을 사용하여 새로운 액세스 토큰을 발급받습니다. diff --git a/OpenApiGenerator/Sources/openapi-generator-cli/3days-oas b/OpenApiGenerator/Sources/openapi-generator-cli/3days-oas index f81da79..aba707b 160000 --- a/OpenApiGenerator/Sources/openapi-generator-cli/3days-oas +++ b/OpenApiGenerator/Sources/openapi-generator-cli/3days-oas @@ -1 +1 @@ -Subproject commit f81da79dc8537bde60d3ec02da999d4369f74a1e +Subproject commit aba707baf59df237f4492bdb9ebbcf58af41b351 diff --git a/Projects/Core/CommonKit/Sources/AppCoordinator.swift b/Projects/Core/CommonKit/Sources/AppCoordinator.swift index bf18f85..e192070 100644 --- a/Projects/Core/CommonKit/Sources/AppCoordinator.swift +++ b/Projects/Core/CommonKit/Sources/AppCoordinator.swift @@ -100,4 +100,12 @@ public final class AppCoordinator: ObservableObject { } } } + + public func refreshMyUserInfo() async throws { + let userInfo = try await authService.requestMyUserInfo() + await MainActor.run { + self.userInfo = userInfo + AuthState.change(.login) + } + } } diff --git a/Projects/Core/NetworkKit/Sources/AuthService/AuthServiceMock.swift b/Projects/Core/NetworkKit/Sources/AuthService/AuthServiceMock.swift index 9d51fdf..337b489 100644 --- a/Projects/Core/NetworkKit/Sources/AuthService/AuthServiceMock.swift +++ b/Projects/Core/NetworkKit/Sources/AuthService/AuthServiceMock.swift @@ -63,7 +63,8 @@ public class AuthServiceMock: AuthServiceProtocol { jobOccupations: [], distanceType: .anywhere, allowSameCompany: nil - ) + ), + profileWidgets: [] ) } } diff --git a/Projects/Core/NetworkKit/Sources/ProfileService/ProfileService.swift b/Projects/Core/NetworkKit/Sources/ProfileService/ProfileService.swift new file mode 100644 index 0000000..3c9b5c4 --- /dev/null +++ b/Projects/Core/NetworkKit/Sources/ProfileService/ProfileService.swift @@ -0,0 +1,41 @@ +// +// ProfileService.swift +// CommonKit +// +// Created by 김지수 on 11/14/24. +// Copyright © 2024 com.weave. All rights reserved. +// + +import Foundation +import CoreKit +import OpenapiGenerated +import Model + +public protocol ProfileServiceProtocol { + func requestPutProfileWidget( + widgetType: Components.Schemas.ProfileWidgetType, + content: String + ) async throws +} + +public final class ProfileService { + public static let shared = ProfileService() + private init() {} +} + +extension ProfileService: ProfileServiceProtocol { + public func requestPutProfileWidget( + widgetType: Components.Schemas.ProfileWidgetType, + content: String + ) async throws { + let result = try await client.putProfileWidget( + body: .json( + .init( + _type: widgetType, + content: content + ) + ) + ) + let _ = try result.ok + } +} diff --git a/Projects/Features/Home/Sources/Profile/ProfileView.swift b/Projects/Features/Home/Sources/Profile/ProfileView.swift index b49abdc..e1a80a4 100644 --- a/Projects/Features/Home/Sources/Profile/ProfileView.swift +++ b/Projects/Features/Home/Sources/Profile/ProfileView.swift @@ -24,6 +24,11 @@ public struct ProfileView: View { (Device.width - 36 - 12) / 2 } + private let columns = [ + GridItem(.flexible(), spacing: 16), + GridItem(.flexible(), spacing: 16) + ] + public init(userInfo: UserInfo) { let model = ProfileModel() let intent = ProfileIntent( @@ -56,50 +61,105 @@ public struct ProfileView: View { LeftAlignText("Introductions") .typography(.en_medium_16) .padding(.bottom, 14) + .padding(.bottom, 16) + .foregroundStyle(Color(hex: 0x5E5E5E)) - ZStack { - Capsule() - .inset(by: 1) - .stroke(DesignCore.Colors.blue300, lineWidth: 1) - .fill(Color(hex: 0xF2F9FF)) - LeftAlignText("프로필 위젯을 추가해 나를 더 소개해보세요!🙌") - .padding(.leading, 26) - .typography(.semibold_14) - .foregroundStyle(DesignCore.Colors.blue300) - } - .frame(height: 57) - .shadow(.default) - .padding(.bottom, 14) - - ZStack { - RoundedRectangle(cornerRadius: 24) - .fill(.white) + // 비어있을 때의 뷰 + if userInfo.profileWidgets.isEmpty { + ZStack { + Capsule() + .inset(by: 1) + .stroke(DesignCore.Colors.blue300, lineWidth: 1) + .fill(Color(hex: 0xF2F9FF)) + LeftAlignText("프로필 위젯을 추가해 나를 더 소개해보세요!🙌") + .padding(.leading, 26) + .typography(.semibold_14) + .foregroundStyle(DesignCore.Colors.blue300) + } + .frame(height: 57) + .shadow(.default) + .padding(.bottom, 14) - RoundedRectangle(cornerRadius: 10) - .fill(DesignCore.Colors.grey50) - .strokeBorder( - style: StrokeStyle( - lineWidth: 3, - dash: [8, 8] + ZStack { + RoundedRectangle(cornerRadius: 24) + .fill(.white) + + RoundedRectangle(cornerRadius: 10) + .fill(DesignCore.Colors.grey50) + .strokeBorder( + style: StrokeStyle( + lineWidth: 3, + dash: [8, 8] + ) ) - ) - .foregroundStyle(Color(hex: 0xE0DEDD)) - .padding(.all, 8) - - VStack { - Image(systemName: "plus") - .resizable() - .frame(width: 24, height: 24) - Text("프로필 위젯 추가하기") - .typography(.semibold_14) + .foregroundStyle(Color(hex: 0xE0DEDD)) + .padding(.all, 8) + + VStack { + Image(systemName: "plus") + .resizable() + .frame(width: 24, height: 24) + Text("프로필 위젯 추가하기") + .typography(.semibold_14) + } + .foregroundStyle(DesignCore.Colors.grey200) + } + .frame(height: widgetSize) + .shadow(.default) + .padding(.bottom, 36) + .onTapGesture { + isPresentWidgetSelectionView = true + } + } else { + LazyVGrid(columns: columns, spacing: 16) { + ForEach(userInfo.profileWidgets, id: \.self) { widget in + ZStack { + RoundedRectangle(cornerRadius: 24) + .fill(.white) + ProfileWidgetView( + title: widget.widgetType.title, + bodyText: widget.content, + titleColor: widget.widgetType.titleColor, + bodyColor: widget.widgetType.bodyColor, + gradientColors: widget.widgetType.gradationColors + ) + .padding(.all, 4) + } + .shadow(.default) + .onTapGesture { + + } + } + + ZStack { + RoundedRectangle(cornerRadius: 24) + .fill(.white) + + RoundedRectangle(cornerRadius: 10) + .fill(DesignCore.Colors.grey50) + .strokeBorder( + style: StrokeStyle( + lineWidth: 3, + dash: [8, 8] + ) + ) + .foregroundStyle(Color(hex: 0xE0DEDD)) + .padding(.all, 8) + + VStack { + Image(systemName: "plus") + .resizable() + .frame(width: 24, height: 24) + Text("프로필 위젯 추가하기") + .typography(.semibold_14) + } + .foregroundStyle(DesignCore.Colors.grey200) + } + .shadow(.default) + .onTapGesture { + isPresentWidgetSelectionView = true + } } - .foregroundStyle(DesignCore.Colors.grey200) - } - .frame(height: widgetSize) - .shadow(.default) - .padding(.bottom, 36) - .onTapGesture { - isPresentWidgetSelectionView = true } } .padding(.horizontal, 18) diff --git a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift index 63f7adc..90cc3a1 100644 --- a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift +++ b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift @@ -10,19 +10,23 @@ import Foundation import CommonKit import CoreKit import Model +import NetworkKit //MARK: - Intent class WidgetWritingIntent { private weak var model: WidgetWritingModelActionable? private let input: DataModel + private let service: ProfileServiceProtocol // MARK: Life cycle init( model: WidgetWritingModelActionable, - input: DataModel + input: DataModel, + service: ProfileServiceProtocol = ProfileService.shared ) { self.input = input self.model = model + self.service = service model.setWidgetType(input.widgetType) } } @@ -61,9 +65,26 @@ extension WidgetWritingIntent: WidgetWritingIntent.Intentable { func onTapNextButton(state: any WidgetWritingModel.Stateful) { guard let selectedWidget = state.selectedWidgetType else { return } // 창닫기 - model?.modalDismiss() - + Task { + model?.setLoading(status: true) + do { + try await requestPutProfileWidget( + widget: selectedWidget, + content: state.widgetBodyText + ) + try await AppCoordinator.shared.refreshMyUserInfo() + model?.setLoading(status: false) + model?.modalDismiss() + } catch { + print(error) + } + } } - + func requestPutProfileWidget(widget: WidgetType, content: String) async throws { + try await service.requestPutProfileWidget( + widgetType: widget.toDto, + content: content + ) + } } diff --git a/Projects/Model/Model/Sources/Auth/Domain/UserInfo.swift b/Projects/Model/Model/Sources/Auth/Domain/UserInfo.swift index f173fae..ee60a59 100644 --- a/Projects/Model/Model/Sources/Auth/Domain/UserInfo.swift +++ b/Projects/Model/Model/Sources/Auth/Domain/UserInfo.swift @@ -15,19 +15,22 @@ public struct UserInfo { public let phone: String public let profile: UserInfoProfile public let dreamPartner: DreamPartnerInfo + public let profileWidgets: [ProfileWidget] public init( id: String, name: String, phone: String, profile: UserInfoProfile, - dreamPartner: DreamPartnerInfo + dreamPartner: DreamPartnerInfo, + profileWidgets: [ProfileWidget] ) { self.id = id self.name = name self.phone = phone self.profile = profile self.dreamPartner = dreamPartner + self.profileWidgets = profileWidgets } public init(from dto: Components.Schemas.GetMyUserInfoResponse) { @@ -36,6 +39,7 @@ public struct UserInfo { self.phone = dto.phoneNumber self.profile = .init(from: dto.profile) self.dreamPartner = .init(from: dto.desiredPartner) + self.profileWidgets = dto.profileWidgets.map { .init(from: $0) } } public static var mock: UserInfo { @@ -44,7 +48,8 @@ public struct UserInfo { name: "김지수", phone: "01012341234", profile: .mock, - dreamPartner: .mock + dreamPartner: .mock, + profileWidgets: ProfileWidget.mock ) } } @@ -136,3 +141,28 @@ public struct DreamPartnerInfo { ) } } + +public struct ProfileWidget: Hashable { + public let widgetType: WidgetType + public let content: String + + static var mock: [ProfileWidget] { + [ + .init(widgetType: .body, content: "GOOD BODY"), + .init(widgetType: .smoking, content: "Heavy Smoker !!") + ] + } + + public init( + widgetType: WidgetType, + content: String + ) { + self.widgetType = widgetType + self.content = content + } + + public init(from dto: Components.Schemas.ProfileWidget) { + self.widgetType = WidgetType(from: dto._type) + self.content = dto.content + } +} diff --git a/Projects/Model/Model/Sources/Profile/Widget/WidgetType.swift b/Projects/Model/Model/Sources/Profile/Widget/WidgetType.swift index e570ee4..486ff6d 100644 --- a/Projects/Model/Model/Sources/Profile/Widget/WidgetType.swift +++ b/Projects/Model/Model/Sources/Profile/Widget/WidgetType.swift @@ -8,6 +8,7 @@ import SwiftUI import CoreKit +import OpenapiGenerated public enum WidgetType: CaseIterable { case hobby @@ -187,3 +188,43 @@ extension WidgetType { colorSet.gradientColors } } + +extension WidgetType { + public var toDto: Components.Schemas.ProfileWidgetType { + switch self { + case .hobby: .HOBBY + case .style: .STYLE + case .mbti: .MBTI + case .music: .MUSIC + case .body: .BODY_TYPE + case .food: .FOOD + case .movie: .MOVIE + case .drama: .DRAMA + case .book: .BOOK + case .travel: .TRAVEL + case .alcohol: .DRINKING + case .marriage: .MARRIAGE + case .religion: .RELIGION + case .smoking: .SMOKING + } + } + + init(from dto: Components.Schemas.ProfileWidgetType) { + switch dto { + case .HOBBY: self = .hobby + case .STYLE: self = .style + case .MBTI: self = .mbti + case .MUSIC: self = .music + case .BODY_TYPE: self = .body + case .FOOD: self = .food + case .MOVIE: self = .movie + case .DRAMA: self = .drama + case .BOOK: self = .book + case .TRAVEL: self = .travel + case .DRINKING: self = .alcohol + case .MARRIAGE: self = .marriage + case .RELIGION: self = .religion + case .SMOKING: self = .smoking + } + } +} From 4fba3a7140d6aaf2a7fd95933bc5d7484f8edc55 Mon Sep 17 00:00:00 2001 From: Jisu Kim Date: Fri, 15 Nov 2024 00:22:13 +0900 Subject: [PATCH 06/11] =?UTF-8?q?[WEAV-122]=20Widget=20Service=20Add,=20Pr?= =?UTF-8?q?ofile=20Widget=20=EC=83=9D=EC=84=B1=20API=20=EC=97=B0=EA=B2=B0,?= =?UTF-8?q?=20=EB=B7=B0=20=EC=A0=95=EB=A6=AC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../CommonKit/Sources/AppCoordinator.swift | 11 +++++++++ .../pencil 1.imageset/Contents.json | 23 ++++++++++++++++++ .../pencil 1.imageset/pencil.png | Bin 0 -> 296 bytes .../pencil 1.imageset/pencil@2x.png | Bin 0 -> 503 bytes .../pencil 1.imageset/pencil@3x.png | Bin 0 -> 700 bytes .../plus_circle_filled.imageset/Contents.json | 23 ++++++++++++++++++ .../plus_circle_filled.png | Bin 0 -> 392 bytes .../plus_circle_filled@2x.png | Bin 0 -> 673 bytes .../plus_circle_filled@3x.png | Bin 0 -> 980 bytes .../ProfileWidget/ProfileWidgetView.swift | 23 ++++++++++++++++-- .../Home/Sources/HomeMain/HomeMainView.swift | 4 +-- .../Home/Sources/Profile/ProfileIntent.swift | 7 +++++- .../Home/Sources/Profile/ProfileView.swift | 20 ++++++++++----- .../WidgetSelection/WidgetSelectionView.swift | 13 +++++++--- .../WidgetWriting/WidgetWritingIntent.swift | 3 ++- 15 files changed, 112 insertions(+), 15 deletions(-) create mode 100644 Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/pencil 1.imageset/Contents.json create mode 100644 Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/pencil 1.imageset/pencil.png create mode 100644 Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/pencil 1.imageset/pencil@2x.png create mode 100644 Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/pencil 1.imageset/pencil@3x.png create mode 100644 Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/plus_circle_filled.imageset/Contents.json create mode 100644 Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/plus_circle_filled.imageset/plus_circle_filled.png create mode 100644 Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/plus_circle_filled.imageset/plus_circle_filled@2x.png create mode 100644 Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/plus_circle_filled.imageset/plus_circle_filled@3x.png diff --git a/Projects/Core/CommonKit/Sources/AppCoordinator.swift b/Projects/Core/CommonKit/Sources/AppCoordinator.swift index e192070..986c7b8 100644 --- a/Projects/Core/CommonKit/Sources/AppCoordinator.swift +++ b/Projects/Core/CommonKit/Sources/AppCoordinator.swift @@ -43,6 +43,7 @@ public final class AppCoordinator: ObservableObject { } } } + startRefreshMyUserInfo() } @MainActor @@ -108,4 +109,14 @@ public final class AppCoordinator: ObservableObject { AuthState.change(.login) } } + + // 20초마다 한번씩 refreshMyUserInfo() 를 호출 + private func startRefreshMyUserInfo() { + Task { + while true { + try await Task.sleep(for: .seconds(20)) + try await refreshMyUserInfo() + } + } + } } diff --git a/Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/pencil 1.imageset/Contents.json b/Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/pencil 1.imageset/Contents.json new file mode 100644 index 0000000..4c3712f --- /dev/null +++ b/Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/pencil 1.imageset/Contents.json @@ -0,0 +1,23 @@ +{ + "images" : [ + { + "filename" : "pencil.png", + "idiom" : "universal", + "scale" : "1x" + }, + { + "filename" : "pencil@2x.png", + "idiom" : "universal", + "scale" : "2x" + }, + { + "filename" : "pencil@3x.png", + "idiom" : "universal", + "scale" : "3x" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/pencil 1.imageset/pencil.png b/Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/pencil 1.imageset/pencil.png new file mode 100644 index 0000000000000000000000000000000000000000..63927463fe8fe6695506326b8bf5271c8392bcf0 GIT binary patch literal 296 zcmeAS@N?(olHy`uVBq!ia0vp^Vj#@H1|*Mc$*~4foCO|{#S9GG!XV7ZFl&wkP>``W z$lZxy-8q?;Kn_c~qpu?a!^VE@KZ&eB{$5WP$B+ufw^P>g9WoGc%m2xFl7&OSSb`KSwt8knK7zm$e_}Z~=)f5V6T+3TJ($ l9n;U|{fB1Xxah#=@3KJUE1UMT9-v1UJYD@<);T3K0RX%oZ?FIW literal 0 HcmV?d00001 diff --git a/Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/pencil 1.imageset/pencil@2x.png b/Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/pencil 1.imageset/pencil@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..8e0f95838dcf022b58efa2576f3f29751aafcbfc GIT binary patch literal 503 zcmV6- z1Thdr%NOb_bm1ttRnWii^L615I3kKDqM+coa5FMkmvrMGNu`p`qZc`FH;1fQeFtW-5NqBU!KoXvko+7e9Isr5d|8Q=|@zEm(lJLIeHwezVd~{Ji te5XJXekZdnAoogtpG1oaHX4ma7{6aFTa0yV#)|*|002ovPDHLkV1nNh&+GsI literal 0 HcmV?d00001 diff --git a/Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/pencil 1.imageset/pencil@3x.png b/Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/pencil 1.imageset/pencil@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..fe0651a1d18cc4686bc08b79803c4cb3ae83c8ae GIT binary patch literal 700 zcmV;t0z>_YP)Fx7!|C!a8N7`2;Z(4dP(pz%$-o>c;?u+ygb7W+ z{Ju=QNCu`3I{-U@kBo&2XR1@54L)Fr}2LWx8M7soys>vZ(j ziz+K3v;;D(GMuPNL<|Wfl<^CTn=sR;;BqR_U5v+z)gd-UV z#70EOYDh`~v8{*@BqxE`hzQX!l9WKCA~-F4*N0>!5NUFv=aA;W1KD#Vj8sGyZt&pc z62-$BNfWvvnE24dBZ`wJuhXY0k&0NG@qZ>>COt$(iBBTEF~i6G_Dy`7ct(*lkKnye ip%q0@6h%>#hVTca^Nz10b7&9%0000 z_c*A`)U+^Frn%`f?zg73sWvrn{f|&ET$(Q8?;&1H&qKzfy4G*Hv5@*nS^5(PQ{c|d zOz);oHYVAhJ59T;dq66af=zSdOYN6*oB1*sv$&EU{h?!A$(MdiFLMU8FjgDzN4xRH^#b6pT1|AK~ mM@h}}_hZWJGp6;XVd5LY+dZTzc?xg<0000&|>| z(A^B=+sT>Bo(ZrW9T7)ybVc;yd!6|6P{cySQp8NeG>#C41Tdp7;#3o?%DafS_-`eu z3J*k#5cQ@F*!$0javWiUUc1mOBtg-BBwYt@>gCMwp8SFI9 z*RT4`R-#ciZs-bh`7?zkg)d>Y5Yz=-#`+CinA1uR?!kN>s0&@#rBp~VQvWDnK7p;} zwhrjV{mWW-2=|d0Kw&2K0PZ7qh7_)ALt`7c_oi?k{vd^2oXWE3svP!(9yb`*NZ(UP zl6!-?DcF02?BO>2q0@qAPvpL6?bsBPq=KtV;v%5s8{{RzbVzVjy8=@2ag@-a#kG)vZ;$Yl^9k_yV!76=TXL_*6 z($)}WB`$R;$~>*SP?-uV^rPE@_?qd{*5tksz11vW;pp}kOrd9=R=Q-f00000NkvXX Hu0mjf{tY%8 literal 0 HcmV?d00001 diff --git a/Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/plus_circle_filled.imageset/plus_circle_filled@3x.png b/Projects/DesignSystem/DesignCore/Resources/Images/Images.xcassets/plus_circle_filled.imageset/plus_circle_filled@3x.png new file mode 100644 index 0000000000000000000000000000000000000000..569f26a619bc1db2772ca66f80de013b225eb95e GIT binary patch literal 980 zcmV;_11tQAP))!55)KnaJx_XiEqRaEMrhO>%^m1ZgS+@A_@T9JiKZ_k~?Tx1o2u zfAt*xMJRuLqcm)+~0Z*{H9%zN-a`{~M^TA;740hmpg$YzE#!kr>IZ?7A z!j@p3J1qKTkq~OV;2O6OSCbWVLa23vt7vD#Iw2^6OCgbmBCOo#ybx*`?K%vR%+1aV z;c`gi$&58EgfHHp;8+u{NDJYUHz<-DpB7>S|460?N(k0gmn%gPtRUQE&IsWv+-XmQ z@G&eDQJ$a>oP+xq7K-RW6NT7AU_>Df5ZDhM5fg=m&-pH4{v(z;Vce-Z5 zBVvX-2t))SBtZxXg=p=tMUR>7ikF4SH`;pcu!ZLuWh{b6#6;zDjzA=Y^`m<|U`ipT z2+UQEFiMz?5glk%^N6($U^&7uUz8Bv;4jfhUL^$cgpc9Du*e8OKEOYY)kRu}FYu3v zx=0Jb>fuw;b#$Mk~-HUuVXB&4N*Aw&xos* zmFF-vg-FAsN-=n_btr8Y`hTRcknc2>!6`QIeD(|z7NfmM_x!p50000 Date: Fri, 15 Nov 2024 01:41:35 +0900 Subject: [PATCH 07/11] =?UTF-8?q?[WEAV-122]=20Profile=20Widget=20Delete=20?= =?UTF-8?q?API=20=EB=A1=9C=EC=A7=81=20=EC=97=B0=EA=B2=B0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../CommonKit/Sources/AppCoordinator.swift | 4 +- .../ProfileService/ProfileService.swift | 15 ++ .../Home/Sources/Profile/ProfileIntent.swift | 41 ++++- .../Home/Sources/Profile/ProfileModel.swift | 20 ++ .../Home/Sources/Profile/ProfileView.swift | 172 ++++++++++++++---- .../WidgetWriting/WidgetWritingIntent.swift | 4 + .../WidgetWriting/WidgetWritingModel.swift | 4 + .../WidgetWriting/WidgetWritingView.swift | 31 +++- 8 files changed, 252 insertions(+), 39 deletions(-) diff --git a/Projects/Core/CommonKit/Sources/AppCoordinator.swift b/Projects/Core/CommonKit/Sources/AppCoordinator.swift index 986c7b8..8143960 100644 --- a/Projects/Core/CommonKit/Sources/AppCoordinator.swift +++ b/Projects/Core/CommonKit/Sources/AppCoordinator.swift @@ -114,8 +114,8 @@ public final class AppCoordinator: ObservableObject { private func startRefreshMyUserInfo() { Task { while true { - try await Task.sleep(for: .seconds(20)) - try await refreshMyUserInfo() + try? await Task.sleep(for: .seconds(20)) + try? await refreshMyUserInfo() } } } diff --git a/Projects/Core/NetworkKit/Sources/ProfileService/ProfileService.swift b/Projects/Core/NetworkKit/Sources/ProfileService/ProfileService.swift index 3c9b5c4..7d0b0d6 100644 --- a/Projects/Core/NetworkKit/Sources/ProfileService/ProfileService.swift +++ b/Projects/Core/NetworkKit/Sources/ProfileService/ProfileService.swift @@ -16,6 +16,10 @@ public protocol ProfileServiceProtocol { widgetType: Components.Schemas.ProfileWidgetType, content: String ) async throws + + func requestDeleteProfileWidget( + widgetType: Components.Schemas.ProfileWidgetType + ) async throws } public final class ProfileService { @@ -38,4 +42,15 @@ extension ProfileService: ProfileServiceProtocol { ) let _ = try result.ok } + + public func requestDeleteProfileWidget( + widgetType: Components.Schemas.ProfileWidgetType + ) async throws { + let response = try await client.deleteProfileWidget( + .init( + path: .init(_type: widgetType) + ) + ) + _ = try response.noContent + } } diff --git a/Projects/Features/Home/Sources/Profile/ProfileIntent.swift b/Projects/Features/Home/Sources/Profile/ProfileIntent.swift index 6222f81..9adcd08 100644 --- a/Projects/Features/Home/Sources/Profile/ProfileIntent.swift +++ b/Projects/Features/Home/Sources/Profile/ProfileIntent.swift @@ -10,19 +10,23 @@ import Foundation import CommonKit import CoreKit import Model +import NetworkKit //MARK: - Intent class ProfileIntent { private weak var model: ProfileModelActionable? private let input: DataModel + private let profileService: ProfileServiceProtocol // MARK: Life cycle init( model: ProfileModelActionable, - input: DataModel + input: DataModel, + service: ProfileServiceProtocol = ProfileService.shared ) { self.input = input self.model = model + self.profileService = service } } @@ -30,6 +34,11 @@ class ProfileIntent { extension ProfileIntent { protocol Intentable { // content + func onTapModifyWidget(_ widget: ProfileWidget) + func onTapDeleteWidget(_ widget: ProfileWidget) + func onTapAddWidget() + func deleteWidget(_ widget: ProfileWidget) async + func onTapNextButton() func fetchUserInfo(_ userInfo: UserInfo) @@ -46,6 +55,32 @@ extension ProfileIntent { //MARK: - Intentable extension ProfileIntent: ProfileIntent.Intentable { // default + func onTapAddWidget() { + + } + + func onTapDeleteWidget(_ widget: ProfileWidget) { + model?.setSelectedWidget(widget) + model?.setDeleteConfirmSheetPresented(true) + } + + func onTapModifyWidget(_ widget: ProfileWidget) { + model?.setSelectedWidget(widget) + model?.setModifyWidgetViewPresented(true) + } + + func deleteWidget(_ widget: ProfileWidget) async { + do { + model?.setLoading(status: true) + try await requestDeleteWidget(widget) + try await AppCoordinator.shared.refreshMyUserInfo() + model?.setLoading(status: false) + } catch { + print(error) + model?.setLoading(status: false) + } + } + func onAppear() { fetchUserInfo(input.userInfo) } @@ -58,4 +93,8 @@ extension ProfileIntent: ProfileIntent.Intentable { // content func onTapNextButton() {} + + func requestDeleteWidget(_ widget: ProfileWidget) async throws { + try await profileService.requestDeleteProfileWidget(widgetType: widget.widgetType.toDto) + } } diff --git a/Projects/Features/Home/Sources/Profile/ProfileModel.swift b/Projects/Features/Home/Sources/Profile/ProfileModel.swift index 51a88cf..2544d58 100644 --- a/Projects/Features/Home/Sources/Profile/ProfileModel.swift +++ b/Projects/Features/Home/Sources/Profile/ProfileModel.swift @@ -16,6 +16,10 @@ final class ProfileModel: ObservableObject { //MARK: Stateful protocol Stateful { // content + var isPresentedModifyWidgetView: Bool { get set } + var isPresentedDeleteConfirmSheet: Bool { get set } + var selectedWidgetType: ProfileWidget? { get } + var userInfoModel: UserInfo? { get } var isValidated: Bool { get } @@ -30,6 +34,10 @@ final class ProfileModel: ObservableObject { //MARK: State Properties // content @Published var userInfoModel: UserInfo? + @Published var isPresentedModifyWidgetView: Bool = false + @Published var isPresentedDeleteConfirmSheet: Bool = false + var selectedWidgetType: ProfileWidget? + @Published var isValidated: Bool = false // default @@ -45,8 +53,11 @@ extension ProfileModel: ProfileModel.Stateful {} //MARK: - Actionable protocol ProfileModelActionable: AnyObject { // content + func setModifyWidgetViewPresented(_ isPresented: Bool) + func setDeleteConfirmSheetPresented(_ isPresented: Bool) func setUserInfo(_ userInfo: UserInfo) func setValidation(value: Bool) + func setSelectedWidget(_ widget: ProfileWidget) // default func setLoading(status: Bool) @@ -59,12 +70,21 @@ protocol ProfileModelActionable: AnyObject { extension ProfileModel: ProfileModelActionable { // content + func setModifyWidgetViewPresented(_ isPresented: Bool) { + isPresentedModifyWidgetView = isPresented + } + func setDeleteConfirmSheetPresented(_ isPresented: Bool) { + isPresentedDeleteConfirmSheet = isPresented + } func setUserInfo(_ userInfo: UserInfo) { userInfoModel = userInfo } func setValidation(value: Bool) { isValidated = value } + func setSelectedWidget(_ widget: ProfileWidget) { + selectedWidgetType = widget + } // default func setLoading(status: Bool) { diff --git a/Projects/Features/Home/Sources/Profile/ProfileView.swift b/Projects/Features/Home/Sources/Profile/ProfileView.swift index 9e14e63..f7e1e70 100644 --- a/Projects/Features/Home/Sources/Profile/ProfileView.swift +++ b/Projects/Features/Home/Sources/Profile/ProfileView.swift @@ -126,41 +126,55 @@ public struct ProfileView: View { ) .padding(.all, 4) } + .frame(minHeight: widgetSize) .shadow(.default) - .onTapGesture { - + .contextMenu { + Button(action: { + intent.onTapModifyWidget(widget) + }) { + Text("수정하기") + } + + Button( + role: .destructive, + action: { + intent.onTapDeleteWidget(widget) + }) { + Text("삭제하기") + } } - .frame(minHeight: widgetSize) } - - ZStack { - RoundedRectangle(cornerRadius: 24) - .fill(.white) - - RoundedRectangle(cornerRadius: 10) - .fill(DesignCore.Colors.grey50) - .strokeBorder( - style: StrokeStyle( - lineWidth: 3, - dash: [8, 8] + let isEveryWidgetAdded = WidgetType.allCases.count == userInfo.profileWidgets.count + if !isEveryWidgetAdded { + ZStack { + RoundedRectangle(cornerRadius: 24) + .fill(.white) + + RoundedRectangle(cornerRadius: 10) + .fill(DesignCore.Colors.grey50) + .strokeBorder( + style: StrokeStyle( + lineWidth: 3, + dash: [8, 8] + ) ) - ) - .foregroundStyle(Color(hex: 0xE0DEDD)) - .padding(.all, 8) - - VStack { - Image(systemName: "plus") - .resizable() - .frame(width: 24, height: 24) - Text("프로필 위젯\n추가하기") - .typography(.semibold_14) + .foregroundStyle(Color(hex: 0xE0DEDD)) + .padding(.all, 8) + + VStack { + Image(systemName: "plus") + .resizable() + .frame(width: 24, height: 24) + Text("프로필 위젯\n추가하기") + .typography(.semibold_14) + } + .foregroundStyle(DesignCore.Colors.grey200) + .frame(minHeight: widgetSize) + } + .shadow(.default) + .onTapGesture { + isPresentWidgetSelectionView = true } - .foregroundStyle(DesignCore.Colors.grey200) - .frame(minHeight: widgetSize) - } - .shadow(.default) - .onTapGesture { - isPresentWidgetSelectionView = true } } } @@ -180,6 +194,20 @@ public struct ProfileView: View { } } } + .onChange(of: state.isPresentedModifyWidgetView) { + if !state.isPresentedModifyWidgetView { + if let userInfo = AppCoordinator.shared.userInfo { + intent.fetchUserInfo(userInfo) + } + } + } + .onChange(of: state.isPresentedDeleteConfirmSheet) { + if !state.isPresentedDeleteConfirmSheet { + if let userInfo = AppCoordinator.shared.userInfo { + intent.fetchUserInfo(userInfo) + } + } + } .sheet( isPresented: $isPresentWidgetSelectionView, content: { @@ -189,6 +217,41 @@ public struct ProfileView: View { ) } }) + .sheet( + isPresented: $container.model.isPresentedModifyWidgetView, + content: { + if let widget = state.selectedWidgetType { + NavigationStack { + WidgetWritingView( + widgetType: widget.widgetType, + isModalPresented: $container.model.isPresentedModifyWidgetView, + isPushed: .constant(false), + isEditing: true, + contentString: widget.content + ) + } + } + } + ) + .sheet( + isPresented: $container.model.isPresentedDeleteConfirmSheet, + content: { + if let widget = state.selectedWidgetType { + DeleteWidgetConfirmView { + Task { + await intent.deleteWidget(widget) + await MainActor.run { + container.model.isPresentedDeleteConfirmSheet = false + } + } + } cancelHandler: { + container.model.isPresentedDeleteConfirmSheet = false + } + .presentationDetents([.height(280)]) + .presentationCornerRadius(20) + } + } + ) .task { await intent.task() } @@ -201,8 +264,53 @@ public struct ProfileView: View { } } +fileprivate struct DeleteWidgetConfirmView: View { + let confirmHandler: () -> Void + let cancelHandler: () -> Void + + var body: some View { + VStack(spacing: 12) { + VStack(spacing: 0) { + LeftAlignText("위젯을 삭제하시겠어요?") + .typography(.semibold_20) + .foregroundStyle(Color(hex: 0x454545)) + LeftAlignText("삭제된 위젯은 복구할 수 없어요.") + .typography(.regular_14) + .foregroundStyle(DesignCore.Colors.grey200) + } + + Spacer() + + VStack(spacing: 8) { + CTAButton( + title: "네, 삭제할게요", + titleColor: .white, + backgroundStyle: DesignCore.Colors.red300 + ) { + confirmHandler() + } + CTAButton( + title: "아니요", + titleColor: DesignCore.Colors.grey400, + backgroundStyle: Color(hex: 0xF2F1F1) + ) { + cancelHandler() + } + } + } + .padding(.horizontal, 28) + .padding(.vertical, 30) + } +} + #Preview { - NavigationView { - HomeMainView(userInfo: .mock) + DeleteWidgetConfirmView { + + } cancelHandler: { + } + +// NavigationView { +// HomeMainView(userInfo: .mock) +// } } diff --git a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift index 4466f24..88de46e 100644 --- a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift +++ b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift @@ -28,6 +28,9 @@ class WidgetWritingIntent { self.model = model self.service = service model.setWidgetType(input.widgetType) + if let contentString = input.content { + model.setContentString(contentString) + } } } @@ -45,6 +48,7 @@ extension WidgetWritingIntent { struct DataModel { let widgetType: WidgetType + let content: String? } } diff --git a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift index d5cd728..6e8c295 100644 --- a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift +++ b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift @@ -64,6 +64,7 @@ protocol WidgetWritingModelActionable: AnyObject { func setBodyText(_ text: String) func setValidation(value: Bool) func setWidgetType(_ widget: WidgetType) + func setContentString(_ content: String) func modalDismiss() // default @@ -92,6 +93,9 @@ extension WidgetWritingModel: WidgetWritingModelActionable { func modalDismiss() { isModalPresented = false } + func setContentString(_ content: String) { + widgetBodyText = content + } // default func setLoading(status: Bool) { diff --git a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift index 7109ffd..64f01fd 100644 --- a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift +++ b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift @@ -17,6 +17,8 @@ public struct WidgetWritingView: View { @Binding var isPushed: Bool @Binding var isModalPresented: Bool @FocusState var isFocused: Bool + let isEditingMode: Bool + let contentString: String? @StateObject var container: MVIContainer @@ -27,15 +29,24 @@ public struct WidgetWritingView: View { Device.height * 0.25 } + private var navigationTitle: String { + if isEditingMode { + return "프로필 위젯 수정" + } + return state.selectedWidgetType?.title ?? "" + } + public init( widgetType: WidgetType, isModalPresented: Binding, - isPushed: Binding + isPushed: Binding, + isEditing: Bool = false, + contentString: String? = nil ) { let model = WidgetWritingModel() let intent = WidgetWritingIntent( model: model, - input: .init(widgetType: widgetType) + input: .init(widgetType: widgetType, content: contentString) ) let container = MVIContainer( intent: intent as WidgetWritingIntent.Intentable, @@ -45,6 +56,8 @@ public struct WidgetWritingView: View { self._container = StateObject(wrappedValue: container) self._isModalPresented = isModalPresented self._isPushed = isPushed + self.isEditingMode = isEditing + self.contentString = contentString } public var body: some View { @@ -101,13 +114,23 @@ public struct WidgetWritingView: View { intent.onAppear() } .ignoresSafeArea(.keyboard) - .navigationTitle(state.selectedWidgetType?.title ?? "") + .navigationTitle(navigationTitle) .navigationBarTitleDisplayMode(.inline) .setNavigation( - showLeftBackButton: true, + showLeftBackButton: isEditingMode ? false : true, handler: { isPushed = false }) + .toolbar { + if isEditingMode { + ToolbarItem { + Button("닫기") { + isModalPresented = false + } + .typography(.medium_16) + } + } + } .setLoading(state.isLoading) } } From a12c6997fd0debc7e9750392f9ab379cdad5e54e Mon Sep 17 00:00:00 2001 From: Jisu Kim Date: Fri, 15 Nov 2024 22:47:33 +0900 Subject: [PATCH 08/11] =?UTF-8?q?[WEAV-122]=20Widget=20=EB=B7=B0=20MVI=20?= =?UTF-8?q?=EB=A1=9C=EC=A7=81=EC=97=90=20=EB=A7=9E=EC=B6=B0=20=EB=B6=84?= =?UTF-8?q?=EB=A6=AC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../WritableProfileWidgetView.swift | 2 +- .../WidgetSelectionIntent.swift | 6 ++++ .../WidgetSelectionModel.swift | 19 +++++++++++ .../WidgetSelection/WidgetSelectionView.swift | 24 ++++++++------ .../WidgetWriting/WidgetWritingIntent.swift | 8 +++++ .../WidgetWriting/WidgetWritingModel.swift | 6 ++++ .../WidgetWriting/WidgetWritingView.swift | 8 +++-- .../Home/UnitTest/WidgetUnitTest.swift | 33 +++++++++++++++++++ Projects/Features/Home/UnitTest/temp.swift | 9 +++++ Projects/Features/Project.swift | 6 ++++ 10 files changed, 108 insertions(+), 13 deletions(-) create mode 100644 Projects/Features/Home/UnitTest/WidgetUnitTest.swift create mode 100644 Projects/Features/Home/UnitTest/temp.swift diff --git a/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/WritableProfileWidgetView.swift b/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/WritableProfileWidgetView.swift index 99c1926..fee45bc 100644 --- a/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/WritableProfileWidgetView.swift +++ b/Projects/DesignSystem/DesignCore/Sources/ProfileWidget/WritableProfileWidgetView.swift @@ -70,7 +70,7 @@ public struct WritableProfileWidgetView: View { PlainTextEditorStyle() ) .focused($isfocused) - .flatTextFieldOption() + .flatTextFieldOption(keyboardType: .namePhonePad) .typography(.regular_14) .foregroundStyle(bodyColor) } diff --git a/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionIntent.swift b/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionIntent.swift index 2b442bc..9308815 100644 --- a/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionIntent.swift +++ b/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionIntent.swift @@ -9,6 +9,7 @@ import Foundation import CommonKit import CoreKit +import Model //MARK: - Intent class WidgetSelectionIntent { @@ -29,6 +30,7 @@ class WidgetSelectionIntent { extension WidgetSelectionIntent { protocol Intentable { // content + func onTapWidget(_ widget: WidgetType) func onTapNextButton() // default @@ -42,6 +44,10 @@ extension WidgetSelectionIntent { //MARK: - Intentable extension WidgetSelectionIntent: WidgetSelectionIntent.Intentable { // default + func onTapWidget(_ widget: WidgetType) { + model?.setSelectedWidget(widget: widget) + model?.setPushWriteContentView(status: true) + } func onAppear() {} func task() async {} diff --git a/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionModel.swift b/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionModel.swift index 377846e..4742cf2 100644 --- a/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionModel.swift +++ b/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionModel.swift @@ -9,13 +9,17 @@ import Foundation import CommonKit import CoreKit +import Model final class WidgetSelectionModel: ObservableObject { //MARK: Stateful protocol Stateful { // content + var isModalPresented: Bool { get set } + var isPushWriteContentView: Bool { get set } var isValidated: Bool { get } + var selectedWidget: WidgetType? { get } // default var isLoading: Bool { get } @@ -27,7 +31,10 @@ final class WidgetSelectionModel: ObservableObject { //MARK: State Properties // content + @Published var isModalPresented: Bool = false + @Published var isPushWriteContentView: Bool = false @Published var isValidated: Bool = false + @Published var selectedWidget: WidgetType? // default @Published var isLoading: Bool = false @@ -42,7 +49,10 @@ extension WidgetSelectionModel: WidgetSelectionModel.Stateful {} //MARK: - Actionable protocol WidgetSelectionModelActionable: AnyObject { // content + func setModalPresented(status: Bool) + func setPushWriteContentView(status: Bool) func setValidation(value: Bool) + func setSelectedWidget(widget: WidgetType) // default func setLoading(status: Bool) @@ -55,6 +65,15 @@ protocol WidgetSelectionModelActionable: AnyObject { extension WidgetSelectionModel: WidgetSelectionModelActionable { // content + func setModalPresented(status: Bool) { + isModalPresented = status + } + func setPushWriteContentView(status: Bool) { + isPushWriteContentView = status + } + func setSelectedWidget(widget: WidgetType) { + selectedWidget = widget + } func setValidation(value: Bool) { isValidated = value } diff --git a/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionView.swift b/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionView.swift index 877c549..59ed089 100644 --- a/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionView.swift +++ b/Projects/Features/Home/Sources/Widget/WidgetSelection/WidgetSelectionView.swift @@ -14,8 +14,7 @@ import Model public struct WidgetSelectionView: View { - @State private var isPushedWriteView: Bool = false - @State private var selectedWidget: WidgetType? +// @State private var isPushedWriteView: Bool = false @Binding var isPresentedSelectionView: Bool @StateObject var container: MVIContainer @@ -41,6 +40,7 @@ public struct WidgetSelectionView: View { ) self._container = StateObject(wrappedValue: container) self._isPresentedSelectionView = isPresented + model.setModalPresented(status: isPresented.wrappedValue) } public var body: some View { @@ -66,8 +66,7 @@ public struct WidgetSelectionView: View { iconType: .add ) .onTapGesture { - selectedWidget = widget - isPushedWriteView = true + intent.onTapWidget(widget) } } } @@ -75,15 +74,20 @@ public struct WidgetSelectionView: View { } } .navigationDestination( - isPresented: $isPushedWriteView, + isPresented: $container.model.isPushWriteContentView, destination: { - WidgetWritingView( - widgetType: selectedWidget ?? .book, - isModalPresented: $isPresentedSelectionView, - isPushed: $isPushedWriteView - ) + if let widget = state.selectedWidget { + WidgetWritingView( + widgetType: widget, + isModalPresented: $container.model.isModalPresented, + isPushed: $container.model.isPushWriteContentView + ) + } } ) + .onChange(of: state.isModalPresented) { + isPresentedSelectionView = state.isModalPresented + } .task { await intent.task() } diff --git a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift index 88de46e..d5d17e9 100644 --- a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift +++ b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingIntent.swift @@ -40,6 +40,8 @@ extension WidgetWritingIntent { // content func onChangedBodyText(_ text: String, maxCount: Int) func onTapNextButton(state: WidgetWritingModel.Stateful) + func onTapBackButton() + func onTapDismissButton() // default func onAppear() @@ -59,6 +61,12 @@ extension WidgetWritingIntent: WidgetWritingIntent.Intentable { let formattedText = text.clipMaxCount(maxCount) model?.setBodyText(formattedText) } + func onTapBackButton() { + model?.navigationPop() + } + func onTapDismissButton() { + model?.modalDismiss() + } func onAppear() { model?.setFocusState(true) } diff --git a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift index 6e8c295..063a4f9 100644 --- a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift +++ b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingModel.swift @@ -21,6 +21,7 @@ final class WidgetWritingModel: ObservableObject { var isValidated: Bool { get } var textMaxCount: Int { get } + var isPushedWriteContentView: Bool { get set } var isModalPresented: Bool { get } var isFocused: Bool { get } var isCTAButtonEnabled: Bool { get } @@ -37,6 +38,7 @@ final class WidgetWritingModel: ObservableObject { // content @Published var selectedWidgetType: WidgetType? @Published var widgetBodyText = String() + @Published var isPushedWriteContentView: Bool = true @Published var isModalPresented: Bool = true @Published var isFocused: Bool = false var textMaxCount: Int = 40 @@ -65,6 +67,7 @@ protocol WidgetWritingModelActionable: AnyObject { func setValidation(value: Bool) func setWidgetType(_ widget: WidgetType) func setContentString(_ content: String) + func navigationPop() func modalDismiss() // default @@ -90,6 +93,9 @@ extension WidgetWritingModel: WidgetWritingModelActionable { func setWidgetType(_ widget: WidgetType) { selectedWidgetType = widget } + func navigationPop() { + isPushedWriteContentView = false + } func modalDismiss() { isModalPresented = false } diff --git a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift index 64f01fd..f271219 100644 --- a/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift +++ b/Projects/Features/Home/Sources/Widget/WidgetWriting/WidgetWritingView.swift @@ -16,6 +16,7 @@ public struct WidgetWritingView: View { @Binding var isPushed: Bool @Binding var isModalPresented: Bool + @FocusState var isFocused: Bool let isEditingMode: Bool let contentString: String? @@ -107,6 +108,9 @@ public struct WidgetWritingView: View { .onChange(of: state.isModalPresented) { isModalPresented = state.isModalPresented } + .onChange(of: state.isPushedWriteContentView) { + isPushed = state.isPushedWriteContentView + } .task { await intent.task() } @@ -119,13 +123,13 @@ public struct WidgetWritingView: View { .setNavigation( showLeftBackButton: isEditingMode ? false : true, handler: { - isPushed = false + intent.onTapBackButton() }) .toolbar { if isEditingMode { ToolbarItem { Button("닫기") { - isModalPresented = false + intent.onTapDismissButton() } .typography(.medium_16) } diff --git a/Projects/Features/Home/UnitTest/WidgetUnitTest.swift b/Projects/Features/Home/UnitTest/WidgetUnitTest.swift new file mode 100644 index 0000000..9feaa1b --- /dev/null +++ b/Projects/Features/Home/UnitTest/WidgetUnitTest.swift @@ -0,0 +1,33 @@ +// +// WidgetUnitTest.swift +// Home-UnitTest +// +// Created by 김지수 on 11/15/24. +// Copyright © 2024 com.weave. All rights reserved. +// + +import Testing +@testable import Home + +struct WidgetUnitTest { + + let widgetSelectionState: WidgetSelectionModel + let widgetSelectionIntent: WidgetSelectionIntent + + init () { + self.widgetSelectionState = WidgetSelectionModel() + self.widgetSelectionIntent = WidgetSelectionIntent( + model: widgetSelectionState, + input: .init() + ) + } + + @Test func someFunctions() async throws { + // Write your test here and use APIs like `#expect(...)` to check expected conditions. + #expect(widgetSelectionState.isValidated == false) + let a = 0 + let b = 1 + #expect(a + b == 1) + + } +} diff --git a/Projects/Features/Home/UnitTest/temp.swift b/Projects/Features/Home/UnitTest/temp.swift new file mode 100644 index 0000000..082e492 --- /dev/null +++ b/Projects/Features/Home/UnitTest/temp.swift @@ -0,0 +1,9 @@ +// +// temp.swift +// DesignPreview +// +// Created by 김지수 on 11/15/24. +// Copyright © 2024 com.weave. All rights reserved. +// + +import Foundation diff --git a/Projects/Features/Project.swift b/Projects/Features/Project.swift index b029799..d587e5f 100644 --- a/Projects/Features/Project.swift +++ b/Projects/Features/Project.swift @@ -30,6 +30,12 @@ let project: Project = .make( .project(target: .commonKit), .project(target: .designCore) ] + ), + .makeUnitTest( + target: .home, + dependencies: [ + .project(target: .home) + ] ) ] ) From 0d1bf3c9694266ba31aad989be4516a50d70f871 Mon Sep 17 00:00:00 2001 From: Jisu Kim Date: Fri, 15 Nov 2024 23:36:29 +0900 Subject: [PATCH 09/11] =?UTF-8?q?[WEAV-122]=20swift=20test=20=EC=BD=94?= =?UTF-8?q?=EB=93=9C=20=EC=9E=91=EC=84=B1?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .github/workflows/deploy.yml | 2 +- .../CommonKit/Sources/AppCoordinator.swift | 3 + .../ProfileService/ProfileServiceMock.swift | 30 ++++++++ .../Home/Sources/Profile/ProfileIntent.swift | 2 +- .../Home/Sources/Profile/ProfileModel.swift | 6 ++ .../Home/Sources/Profile/ProfileView.swift | 14 ++-- .../Home/UnitTest/ProfileUnitTest.swift | 56 +++++++++++++++ .../Home/UnitTest/WidgetUnitTest.swift | 69 +++++++++++++++---- Projects/Features/Home/UnitTest/temp.swift | 9 --- Workspace.swift | 3 +- 10 files changed, 163 insertions(+), 31 deletions(-) create mode 100644 Projects/Core/NetworkKit/Sources/ProfileService/ProfileServiceMock.swift create mode 100644 Projects/Features/Home/UnitTest/ProfileUnitTest.swift delete mode 100644 Projects/Features/Home/UnitTest/temp.swift diff --git a/.github/workflows/deploy.yml b/.github/workflows/deploy.yml index f59f8fb..1962c0f 100644 --- a/.github/workflows/deploy.yml +++ b/.github/workflows/deploy.yml @@ -10,7 +10,7 @@ on: jobs: build: - runs-on: macos-latest + runs-on: macos-15 env: # app archive 및 export 에 쓰일 환경 변수 설정 diff --git a/Projects/Core/CommonKit/Sources/AppCoordinator.swift b/Projects/Core/CommonKit/Sources/AppCoordinator.swift index 8143960..d349216 100644 --- a/Projects/Core/CommonKit/Sources/AppCoordinator.swift +++ b/Projects/Core/CommonKit/Sources/AppCoordinator.swift @@ -103,6 +103,9 @@ public final class AppCoordinator: ObservableObject { } public func refreshMyUserInfo() async throws { + if TokenManager.accessToken == nil || TokenManager.accessToken == "" { + return + } let userInfo = try await authService.requestMyUserInfo() await MainActor.run { self.userInfo = userInfo diff --git a/Projects/Core/NetworkKit/Sources/ProfileService/ProfileServiceMock.swift b/Projects/Core/NetworkKit/Sources/ProfileService/ProfileServiceMock.swift new file mode 100644 index 0000000..1d493ee --- /dev/null +++ b/Projects/Core/NetworkKit/Sources/ProfileService/ProfileServiceMock.swift @@ -0,0 +1,30 @@ +// +// ProfileServiceMock.swift +// NetworkKit +// +// Created by 김지수 on 11/15/24. +// Copyright © 2024 com.weave. All rights reserved. +// + +import Foundation +import OpenapiGenerated +import CoreKit + +public final class ProfileServiceMock: ProfileServiceProtocol { + public init() {} + + public func requestPutProfileWidget( + widgetType: Components.Schemas.ProfileWidgetType, + content: String + ) async throws { + print("✅ [ProfileServiceMock] requestPutProfileWidget 성공!") + return + } + + public func requestDeleteProfileWidget( + widgetType: Components.Schemas.ProfileWidgetType + ) async throws { + print("✅ [ProfileServiceMock] requestDeleteProfileWidget 성공!") + return + } +} diff --git a/Projects/Features/Home/Sources/Profile/ProfileIntent.swift b/Projects/Features/Home/Sources/Profile/ProfileIntent.swift index 9adcd08..6d4cb3a 100644 --- a/Projects/Features/Home/Sources/Profile/ProfileIntent.swift +++ b/Projects/Features/Home/Sources/Profile/ProfileIntent.swift @@ -56,7 +56,7 @@ extension ProfileIntent { extension ProfileIntent: ProfileIntent.Intentable { // default func onTapAddWidget() { - + model?.setAddWidgetModalPresented(true) } func onTapDeleteWidget(_ widget: ProfileWidget) { diff --git a/Projects/Features/Home/Sources/Profile/ProfileModel.swift b/Projects/Features/Home/Sources/Profile/ProfileModel.swift index 2544d58..c6c8133 100644 --- a/Projects/Features/Home/Sources/Profile/ProfileModel.swift +++ b/Projects/Features/Home/Sources/Profile/ProfileModel.swift @@ -16,6 +16,7 @@ final class ProfileModel: ObservableObject { //MARK: Stateful protocol Stateful { // content + var isPresentedAddWidgetModal: Bool { get set } var isPresentedModifyWidgetView: Bool { get set } var isPresentedDeleteConfirmSheet: Bool { get set } var selectedWidgetType: ProfileWidget? { get } @@ -34,6 +35,7 @@ final class ProfileModel: ObservableObject { //MARK: State Properties // content @Published var userInfoModel: UserInfo? + @Published var isPresentedAddWidgetModal: Bool = false @Published var isPresentedModifyWidgetView: Bool = false @Published var isPresentedDeleteConfirmSheet: Bool = false var selectedWidgetType: ProfileWidget? @@ -53,6 +55,7 @@ extension ProfileModel: ProfileModel.Stateful {} //MARK: - Actionable protocol ProfileModelActionable: AnyObject { // content + func setAddWidgetModalPresented(_ isPresented: Bool) func setModifyWidgetViewPresented(_ isPresented: Bool) func setDeleteConfirmSheetPresented(_ isPresented: Bool) func setUserInfo(_ userInfo: UserInfo) @@ -70,6 +73,9 @@ protocol ProfileModelActionable: AnyObject { extension ProfileModel: ProfileModelActionable { // content + func setAddWidgetModalPresented(_ isPresented: Bool) { + isPresentedAddWidgetModal = isPresented + } func setModifyWidgetViewPresented(_ isPresented: Bool) { isPresentedModifyWidgetView = isPresented } diff --git a/Projects/Features/Home/Sources/Profile/ProfileView.swift b/Projects/Features/Home/Sources/Profile/ProfileView.swift index f7e1e70..7ca4ac1 100644 --- a/Projects/Features/Home/Sources/Profile/ProfileView.swift +++ b/Projects/Features/Home/Sources/Profile/ProfileView.swift @@ -15,7 +15,7 @@ import Model public struct ProfileView: View { @StateObject var container: MVIContainer - @State var isPresentWidgetSelectionView = false +// @State var isPresentWidgetSelectionView = false private var intent: ProfileIntent.Intentable { container.intent } private var state: ProfileModel.Stateful { container.model } @@ -108,7 +108,7 @@ public struct ProfileView: View { .shadow(.default) .padding(.bottom, 36) .onTapGesture { - isPresentWidgetSelectionView = true + intent.onTapAddWidget() } } else { LazyVGrid(columns: columns, spacing: 16) { @@ -173,7 +173,7 @@ public struct ProfileView: View { } .shadow(.default) .onTapGesture { - isPresentWidgetSelectionView = true + intent.onTapAddWidget() } } } @@ -187,8 +187,8 @@ public struct ProfileView: View { ProgressView() } } - .onChange(of: isPresentWidgetSelectionView) { - if !isPresentWidgetSelectionView { + .onChange(of: state.isPresentedAddWidgetModal) { + if !state.isPresentedAddWidgetModal { if let userInfo = AppCoordinator.shared.userInfo { intent.fetchUserInfo(userInfo) } @@ -209,11 +209,11 @@ public struct ProfileView: View { } } .sheet( - isPresented: $isPresentWidgetSelectionView, + isPresented: $container.model.isPresentedAddWidgetModal, content: { NavigationStack { WidgetSelectionView( - isPresented: $isPresentWidgetSelectionView + isPresented: $container.model.isPresentedAddWidgetModal ) } }) diff --git a/Projects/Features/Home/UnitTest/ProfileUnitTest.swift b/Projects/Features/Home/UnitTest/ProfileUnitTest.swift new file mode 100644 index 0000000..5c3d10e --- /dev/null +++ b/Projects/Features/Home/UnitTest/ProfileUnitTest.swift @@ -0,0 +1,56 @@ +// +// ProfileUnitTest.swift +// Home-UnitTest +// +// Created by 김지수 on 11/15/24. +// Copyright © 2024 com.weave. All rights reserved. +// + +import Testing +import CommonKit +import NetworkKit +import Model +@testable import Home + +struct ProfileUnitTest { + + let appCoordinator = AppCoordinator.shared + let state: ProfileModel + let intent: ProfileIntent + + init() { + self.state = ProfileModel() + self.intent = ProfileIntent( + model: state, + input: .init(userInfo: .mock), + service: ProfileServiceMock() + ) + appCoordinator.userInfo = .mock + intent.onAppear() + } + + @Test func onTapAddWidget() async throws { + intent.onTapAddWidget() + #expect(state.isPresentedAddWidgetModal == true) + } + + @Test func modifyWidget() async throws { + let widget = ProfileWidget( + widgetType: .body, + content: "이것은 콘텐츠으" + ) + intent.onTapModifyWidget(widget) + #expect(state.selectedWidgetType == widget) + #expect(state.isPresentedModifyWidgetView == true) + } + + @Test func onTapDelete() async throws { + let widget = ProfileWidget( + widgetType: .body, + content: "이것은 콘텐츠으" + ) + intent.onTapDeleteWidget(widget) + #expect(state.selectedWidgetType == widget) + #expect(state.isPresentedDeleteConfirmSheet == true) + } +} diff --git a/Projects/Features/Home/UnitTest/WidgetUnitTest.swift b/Projects/Features/Home/UnitTest/WidgetUnitTest.swift index 9feaa1b..0f2e4fc 100644 --- a/Projects/Features/Home/UnitTest/WidgetUnitTest.swift +++ b/Projects/Features/Home/UnitTest/WidgetUnitTest.swift @@ -8,26 +8,71 @@ import Testing @testable import Home +import NetworkKit struct WidgetUnitTest { - let widgetSelectionState: WidgetSelectionModel - let widgetSelectionIntent: WidgetSelectionIntent + let selectionState: WidgetSelectionModel + let selectionIntent: WidgetSelectionIntent + + let writeState: WidgetWritingModel + let writeIntent: WidgetWritingIntent init () { - self.widgetSelectionState = WidgetSelectionModel() - self.widgetSelectionIntent = WidgetSelectionIntent( - model: widgetSelectionState, + self.selectionState = WidgetSelectionModel() + self.selectionIntent = WidgetSelectionIntent( + model: selectionState, input: .init() ) + + self.writeState = WidgetWritingModel() + self.writeIntent = WidgetWritingIntent( + model: writeState, + input: .init( + widgetType: .body, + content: nil + ), + service: ProfileServiceMock() + ) + + selectionIntent.onAppear() + writeIntent.onAppear() } - @Test func someFunctions() async throws { - // Write your test here and use APIs like `#expect(...)` to check expected conditions. - #expect(widgetSelectionState.isValidated == false) - let a = 0 - let b = 1 - #expect(a + b == 1) - + @Test func selectWidget() async throws { + selectionIntent.onTapWidget(.body) + #expect(selectionState.selectedWidget == .body) + #expect(selectionState.isPushWriteContentView == true) + } + + @Test func writeWidgetContent() async throws { + let content: String = "Hello, World!" + writeIntent.onChangedBodyText(content, maxCount: 15) + #expect(writeState.widgetBodyText == content) + let longContent: String = "Hello World!! This is Hipster Text! Oh YEAH!!" + writeIntent.onChangedBodyText(longContent, maxCount: 15) + #expect(writeState.widgetBodyText == "Hello World!! T") + #expect(writeState.widgetBodyText.count == 15) + } + + @Test func onTapBackButton() async throws { + writeIntent.onTapBackButton() + writeState.isPushedWriteContentView = false + } + + @Test func onTapDismissButton() async throws { + writeIntent.onTapDismissButton() + writeState.isPushedWriteContentView = false + } + + @Test func keyboardFocusState() async throws { + #expect(writeState.isFocused == true) + } + + @Test func onTapNextButton() async throws { + writeIntent.onTapNextButton(state: writeState) + // 3초 후 실행 + try await Task.sleep(for: .seconds(1)) + #expect(writeState.isModalPresented == false) } } diff --git a/Projects/Features/Home/UnitTest/temp.swift b/Projects/Features/Home/UnitTest/temp.swift deleted file mode 100644 index 082e492..0000000 --- a/Projects/Features/Home/UnitTest/temp.swift +++ /dev/null @@ -1,9 +0,0 @@ -// -// temp.swift -// DesignPreview -// -// Created by 김지수 on 11/15/24. -// Copyright © 2024 com.weave. All rights reserved. -// - -import Foundation diff --git a/Workspace.swift b/Workspace.swift index 353a5c6..bd3660b 100644 --- a/Workspace.swift +++ b/Workspace.swift @@ -46,7 +46,8 @@ let workspace = Workspace( [ .target(.coreKit), .target(.designCore), - .target(.signUp) + .target(.signUp), + .target(.home) ] ) ), From b16d5fbf12aaac04ae87eed80e8191b92fa867d1 Mon Sep 17 00:00:00 2001 From: Jisu Kim Date: Fri, 15 Nov 2024 23:39:43 +0900 Subject: [PATCH 10/11] [Infra] Runner on macos-15 --- .github/workflows/unitTest.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unitTest.yml b/.github/workflows/unitTest.yml index 53bb63d..4257d96 100644 --- a/.github/workflows/unitTest.yml +++ b/.github/workflows/unitTest.yml @@ -6,7 +6,7 @@ on: jobs: run-unitTest: - runs-on: macos-latest + runs-on: macos-15 steps: - uses: actions/checkout@v4 From 59e6b7fa52e9928be955a6e9bdf1319a3fc05911 Mon Sep 17 00:00:00 2001 From: Jisu Kim Date: Fri, 15 Nov 2024 23:45:35 +0900 Subject: [PATCH 11/11] [Infra] Unit Test Device Edited --- .github/workflows/unitTest.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unitTest.yml b/.github/workflows/unitTest.yml index 4257d96..d25a2cf 100644 --- a/.github/workflows/unitTest.yml +++ b/.github/workflows/unitTest.yml @@ -27,7 +27,7 @@ jobs: run: | set -o pipefail && xcodebuild test \ -scheme three-days-UnitTest \ - -destination 'platform=iOS Simulator,name=iPhone 15,OS=latest' \ + -destination 'platform=iOS Simulator,name=iPhone 16,OS=latest' \ | tee result.log if grep -q "** TEST FAILED **" result.log; then exit 1