설정 페이지 추가

This commit is contained in:
Yu Sung 2023-08-10 22:05:14 +09:00
parent d06f4d6a36
commit dbeb15ba17
24 changed files with 1368 additions and 4 deletions

View File

@ -144,6 +144,15 @@
"version" : "1.1.1"
}
},
{
"identity" : "richtext",
"kind" : "remoteSourceControl",
"location" : "https://github.com/NuPlay/RichText.git",
"state" : {
"revision" : "ff468d18b066ea5838a2d3f9cb572d55b8ebdb11",
"version" : "2.3.0"
}
},
{
"identity" : "rxswift",
"kind" : "remoteSourceControl",

View File

@ -23,4 +23,18 @@ enum AppStep {
case writeVoiceMessage(userId: Int?, nickname: String?, onRefresh: () -> Void)
case settings
case notices
case noticeDetail(notice: NoticeItem)
case events
case eventDetail(event: EventItem)
case terms
case privacy
case notificationSettings
}

View File

@ -36,8 +36,28 @@ struct ContentView: View {
VoiceMessageWriteView(replySenderId: userId, replySenderNickname: nickname, onRefresh: onRefresh)
case .settings:
EmptyView()
.frame(width: 0, height: 0, alignment: .topLeading)
SettingsView()
case .notices:
NoticeListView()
case .noticeDetail(let notice):
NoticeDetailView(notice: notice)
case .events:
EventListView()
case .eventDetail(let event):
EventDetailView(event: event)
case .terms:
TermsView(isPrivacyPolicy: false)
case .privacy:
TermsView(isPrivacyPolicy: true)
case .notificationSettings:
NotificationSettingsView()
default:
EmptyView()

View File

@ -0,0 +1,69 @@
//
// EventDetailView.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import SwiftUI
import Kingfisher
struct EventDetailView: View {
let event: EventItem
var body: some View {
BaseView {
GeometryReader { proxy in
VStack(spacing: 0) {
DetailNavigationBar(title: "이벤트 상세")
ScrollView(.vertical, showsIndicators: false) {
KFImage(URL(string: event.detailImageUrl!))
.resizable()
.scaledToFit()
}
Spacer()
if let link = event.link, link.count > 0, let url = URL(string: link), UIApplication.shared.canOpenURL(url) {
Text("이벤트 참여하기")
.font(.custom(Font.bold.rawValue, size: 18.3))
.foregroundColor(.white)
.padding(.vertical, 16)
.frame(width: screenSize().width - 26.7)
.background(Color(hex: "3e737c"))
.cornerRadius(10)
.padding(13.3)
.background(Color(hex: "222222"))
.cornerRadius(16.7, corners: [.topLeft, .topRight])
.onTapGesture {
UIApplication.shared.open(url)
}
}
if proxy.safeAreaInsets.bottom > 0 {
Rectangle()
.foregroundColor(Color(hex: "222222"))
.frame(width: proxy.size.width, height: 15.3)
}
}
.edgesIgnoringSafeArea(.bottom)
}
}
}
}
struct EventDetailView_Previews: PreviewProvider {
static var previews: some View {
EventDetailView(
event: EventItem(
id: 1,
thumbnailImageUrl: "",
detailImageUrl: "",
popupImageUrl: "",
link: "http://m.naver.com"
)
)
}
}

View File

@ -0,0 +1,73 @@
//
// EventListView.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import SwiftUI
import Kingfisher
struct EventListView: View {
@ObservedObject var viewModel = EventListViewModel()
var body: some View {
BaseView(isLoading: $viewModel.isLoading) {
VStack(spacing: 0) {
DetailNavigationBar(title: "이벤트")
ScrollView(.vertical, showsIndicators: false) {
VStack(spacing: 0) {
ForEach(viewModel.events, id: \.self) { event in
KFImage(URL(string: event.thumbnailImageUrl))
.resizable()
.scaledToFill()
.frame(
width: screenSize().width - 26.7,
height: (screenSize().width - 26.7) * 300 / 1000
)
.contentShape(Rectangle())
.onTapGesture {
if let _ = event.detailImageUrl {
AppState.shared.setAppStep(step: .eventDetail(event: event))
} else if let link = event.link, link.trimmingCharacters(in: .whitespaces).count > 0, let url = URL(string: link), UIApplication.shared.canOpenURL(url) {
UIApplication.shared.open(url)
}
}
}
}
}
.padding(.vertical, 13.3)
}
}
.popup(isPresented: $viewModel.isShowPopup, type: .toast, position: .top, autohideIn: 2) {
GeometryReader { geo in
HStack {
Spacer()
Text(viewModel.errorMessage)
.padding(.vertical, 13.3)
.padding(.horizontal, 6.7)
.frame(width: geo.size.width - 66.7, alignment: .center)
.font(.custom(Font.medium.rawValue, size: 12))
.background(Color(hex: "9970ff"))
.foregroundColor(Color.white)
.multilineTextAlignment(.leading)
.fixedSize(horizontal: false, vertical: true)
.cornerRadius(20)
.padding(.top, 66.7)
Spacer()
}
}
}
.onAppear {
viewModel.getEvents()
}
}
}
struct EventListView_Previews: PreviewProvider {
static var previews: some View {
EventListView()
}
}

View File

@ -0,0 +1,60 @@
//
// EventListViewModel.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import Foundation
import Combine
final class EventListViewModel: ObservableObject {
private let eventRepository = EventRepository()
private var subscription = Set<AnyCancellable>()
@Published private(set) var events = [EventItem]()
@Published var errorMessage = ""
@Published var isShowPopup = false
@Published var isLoading = false
private var page = 1
private let size = 10
func getEvents() {
isLoading = true
eventRepository.getEvents()
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { [unowned self] response in
self.isLoading = false
let responseData = response.data
do {
let jsonDecoder = JSONDecoder()
let decoded = try jsonDecoder.decode(ApiResponse<GetEventResponse>.self, from: responseData)
if let data = decoded.data, decoded.success {
self.events.append(contentsOf: data.eventList)
} else {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
}
self.isShowPopup = true
}
} catch {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.isShowPopup = true
}
}
.store(in: &subscription)
}
}

View File

@ -0,0 +1,19 @@
//
// GetNoticeResponse.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import Foundation
struct GetNoticeResponse: Decodable {
let totalCount: Int
let noticeList: [NoticeItem]
}
struct NoticeItem: Decodable, Hashable {
let title: String
let content: String
let date: String
}

View File

@ -0,0 +1,45 @@
//
// NoticeApi.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import Foundation
import Moya
enum NoticeApi {
case getNotices
}
extension NoticeApi: TargetType {
var baseURL: URL {
return URL(string: BASE_URL)!
}
var path: String {
switch self {
case .getNotices:
return "/notice"
}
}
var method: Moya.Method {
return .get
}
var task: Task {
switch self {
case .getNotices:
let parameters = ["timezone": TimeZone.current.identifier] as [String: Any]
return .requestParameters(parameters: parameters, encoding: URLEncoding.queryString)
}
}
var headers: [String : String]? {
switch self {
default:
return ["Authorization": "Bearer \(UserDefaults.string(forKey: UserDefaultsKey.token))"]
}
}
}

View File

@ -0,0 +1,57 @@
//
// NoticeDetailView.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import SwiftUI
import RichText
struct NoticeDetailView: View {
let notice: NoticeItem
var body: some View {
BaseView {
VStack(spacing: 0) {
DetailNavigationBar(title: "공지사항 상세")
VStack(alignment: .leading, spacing: 6.7) {
Text(notice.title)
.font(.custom(Font.medium.rawValue, size: 14.7))
.foregroundColor(Color(hex: "eeeeee"))
.padding(.horizontal, 13.3)
Text(notice.date)
.font(.custom(Font.medium.rawValue, size: 12))
.foregroundColor(Color(hex: "525252"))
.padding(.horizontal, 13.3)
}
.padding(.horizontal, 13.3)
.padding(.vertical, 21.7)
.frame(width: screenSize().width - 26.7, alignment: .leading)
.background(Color(hex: "222222"))
.cornerRadius(6.7)
ScrollView(.vertical, showsIndicators: false) {
RichText(html: notice.content)
.padding(.horizontal, 33.3)
.padding(.vertical, 26.7)
}
}
}
}
}
struct NoticeDetailView_Previews: PreviewProvider {
static var previews: some View {
NoticeDetailView(
notice: NoticeItem(
title: "제목",
content: "<h1>콘텐츠</h1>",
date: "2022.03.03"
)
)
}
}

View File

@ -0,0 +1,82 @@
//
// NoticeListView.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import SwiftUI
import RichText
struct NoticeListView: View {
@ObservedObject var viewModel = NoticeListViewModel()
var body: some View {
BaseView(isLoading: $viewModel.isLoading) {
VStack(spacing: 0) {
DetailNavigationBar(title: "공지사항")
ScrollView(.vertical, showsIndicators: false) {
VStack(spacing: 0) {
ForEach(viewModel.notices, id: \.self) { notice in
VStack(alignment: .leading, spacing: 6.7) {
Spacer()
Text(notice.title)
.font(.custom(Font.medium.rawValue, size: 14.7))
.foregroundColor(Color(hex: "eeeeee"))
.padding(.horizontal, 13.3)
Text(notice.date)
.font(.custom(Font.medium.rawValue, size: 12))
.foregroundColor(Color(hex: "525252"))
.padding(.horizontal, 13.3)
Spacer()
Rectangle()
.frame(height: 1)
.foregroundColor(Color(hex: "909090").opacity(0.5))
}
.padding(.horizontal, 13.3)
.frame(width: screenSize().width, height: 80)
.contentShape(Rectangle())
.onTapGesture {
AppState.shared.setAppStep(step: .noticeDetail(notice: notice))
}
}
}
}
.padding(.vertical, 13.3)
}
}
.popup(isPresented: $viewModel.isShowPopup, type: .toast, position: .top, autohideIn: 2) {
GeometryReader { geo in
HStack {
Spacer()
Text(viewModel.errorMessage)
.padding(.vertical, 13.3)
.padding(.horizontal, 6.7)
.frame(width: geo.size.width - 66.7, alignment: .center)
.font(.custom(Font.medium.rawValue, size: 12))
.background(Color(hex: "9970ff"))
.foregroundColor(Color.white)
.multilineTextAlignment(.leading)
.fixedSize(horizontal: false, vertical: true)
.cornerRadius(20)
.padding(.top, 66.7)
Spacer()
}
}
}
.onAppear {
viewModel.getNotices()
}
}
}
struct NoticeListView_Previews: PreviewProvider {
static var previews: some View {
NoticeListView()
}
}

View File

@ -0,0 +1,59 @@
//
// NoticeListViewModel.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import Foundation
import Combine
final class NoticeListViewModel: ObservableObject {
private let repository = NoticeRepository()
private var subscription = Set<AnyCancellable>()
@Published var errorMessage = ""
@Published var isShowPopup = false
@Published var isLoading = false
@Published var notices = [NoticeItem]()
func getNotices() {
isLoading = true
repository.getNotices()
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { [unowned self] response in
self.isLoading = false
let responseData = response.data
do {
let jsonDecoder = JSONDecoder()
let decoded = try jsonDecoder.decode(ApiResponse<GetNoticeResponse>.self, from: responseData)
if let data = decoded.data, decoded.success {
self.notices.append(contentsOf: data.noticeList)
} else {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
}
self.isShowPopup = true
}
} catch {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.isShowPopup = true
}
}
.store(in: &subscription)
}
}

View File

@ -0,0 +1,21 @@
//
// NoticeRepository.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import Foundation
import CombineMoya
import Combine
import Moya
final class NoticeRepository {
private let api = MoyaProvider<NoticeApi>()
func getNotices() -> AnyPublisher<Response, MoyaError> {
return api.requestPublisher(.getNotices)
}
}

View File

@ -0,0 +1,115 @@
//
// NotificationSettingsView.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import SwiftUI
struct NotificationSettingsView: View {
@StateObject var viewModel = NotificationSettingsViewModel()
var body: some View {
BaseView(isLoading: $viewModel.isLoading) {
VStack(spacing: 0) {
DetailNavigationBar(title: "알림 설정")
ScrollView(.vertical, showsIndicators: false) {
VStack(spacing: 0) {
HStack(spacing: 0) {
Text("라이브 알림")
.font(.custom(Font.bold.rawValue, size: 15))
.foregroundColor(Color(hex: "eeeeee"))
Spacer()
Image(viewModel.followingChannelLive ? "btn_toggle_on_big" : "btn_toggle_off_big")
.resizable()
.frame(width: 44, height: 27)
.onTapGesture {
viewModel.followingChannelLive.toggle()
}
}
.frame(height: 50)
Rectangle()
.frame(height: 1)
.foregroundColor(Color(hex: "909090").opacity(0.3))
HStack(spacing: 0) {
Text("콘텐츠 업로드 알림")
.font(.custom(Font.bold.rawValue, size: 15))
.foregroundColor(Color(hex: "eeeeee"))
Spacer()
Image(viewModel.followingChannelUploadContent ? "btn_toggle_on_big" : "btn_toggle_off_big")
.resizable()
.frame(width: 44, height: 27)
.onTapGesture {
viewModel.followingChannelUploadContent.toggle()
}
}
.frame(height: 50)
Rectangle()
.frame(height: 1)
.foregroundColor(Color(hex: "909090").opacity(0.3))
HStack(spacing: 0) {
Text("메시지 알림")
.font(.custom(Font.bold.rawValue, size: 15))
.foregroundColor(Color(hex: "eeeeee"))
Spacer()
Image(viewModel.message ? "btn_toggle_on_big" : "btn_toggle_off_big")
.resizable()
.frame(width: 44, height: 27)
.onTapGesture {
viewModel.message.toggle()
}
}
.frame(height: 50)
}
.padding(.vertical, 6.7)
.padding(.horizontal, 13.3)
.background(Color(hex: "222222"))
.cornerRadius(10)
.padding(.top, 26.7)
.padding(.horizontal, 13.3)
}
}
.onAppear {
viewModel.getMemberInfo()
}
}
.popup(isPresented: $viewModel.isShowPopup, type: .toast, position: .top, autohideIn: 2) {
GeometryReader { geo in
HStack {
Spacer()
Text(viewModel.errorMessage)
.padding(.vertical, 13.3)
.padding(.horizontal, 6.7)
.frame(width: geo.size.width - 66.7, alignment: .center)
.font(.custom(Font.medium.rawValue, size: 12))
.background(Color(hex: "9970ff"))
.foregroundColor(Color.white)
.multilineTextAlignment(.leading)
.fixedSize(horizontal: false, vertical: true)
.cornerRadius(20)
.padding(.top, 66.7)
Spacer()
}
}
}
}
}
struct NotificationSettingsView_Previews: PreviewProvider {
static var previews: some View {
NotificationSettingsView()
}
}

View File

@ -0,0 +1,93 @@
//
// NotificationSettingsViewModel.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import Foundation
import Combine
final class NotificationSettingsViewModel: ObservableObject {
private let userRepository = UserRepository()
private var subscription = Set<AnyCancellable>()
@Published var followingChannelLive = false {
didSet {
submit(live: followingChannelLive)
}
}
@Published var followingChannelUploadContent = false {
didSet {
submit(uploadContent: followingChannelUploadContent)
}
}
@Published var message = false {
didSet {
submit(message: message)
}
}
@Published var errorMessage = ""
@Published var isShowPopup = false
@Published var isLoading = false
func getMemberInfo() {
isLoading = true
userRepository.getMemberInfo()
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { [unowned self] response in
let responseData = response.data
do {
let jsonDecoder = JSONDecoder()
let decoded = try jsonDecoder.decode(ApiResponse<GetMemberInfoResponse>.self, from: responseData)
if let data = decoded.data, decoded.success {
self.followingChannelLive = data.followingChannelLiveNotice ?? false
self.followingChannelUploadContent = data.followingChannelUploadContentNotice ?? false
self.message = data.messageNotice ?? false
} else {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
}
self.isShowPopup = true
}
} catch {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.isShowPopup = true
}
self.isLoading = false
}
.store(in: &subscription)
}
func submit(live: Bool? = nil, uploadContent: Bool? = nil, message: Bool? = nil) {
if !isLoading && (live != nil || uploadContent != nil || message != nil) {
userRepository
.updateNotificationSettings(live: live, uploadContent: uploadContent, message: message)
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { _ in
}
.store(in: &subscription)
}
}
}

View File

@ -0,0 +1,14 @@
//
// UpdateNotificationSettingRequest.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import Foundation
struct UpdateNotificationSettingRequest: Encodable {
var live: Bool? = nil
var uploadContent: Bool? = nil
var message: Bool? = nil
}

View File

@ -0,0 +1,252 @@
//
// SettingsView.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import SwiftUI
struct SettingsView: View {
@State private var isShowLogoutDialog = false
@State private var isShowLogoutAllDeviceDialog = false
@StateObject var viewModel = SettingsViewModel()
var body: some View {
let cardWidth = screenSize().width - 26.7
BaseView(isLoading: $viewModel.isLoading) {
GeometryReader { geo in
VStack(spacing: 0) {
DetailNavigationBar(title: "설정")
VStack(spacing: 0) {
HStack(spacing: 0) {
Text("공지사항")
.font(.custom(Font.bold.rawValue, size: 14.7))
.foregroundColor(Color(hex: "eeeeee"))
Spacer()
Image("ic_forward")
.resizable()
.frame(width: 20, height: 20)
}
.padding(.horizontal, 3.3)
.frame(width: cardWidth - 26.7, height: 50)
.contentShape(Rectangle())
.onTapGesture {
AppState.shared.setAppStep(step: .notices)
}
Rectangle()
.frame(width: cardWidth - 26.7, height: 0.3)
.foregroundColor(Color(hex: "909090"))
HStack(spacing: 0) {
Text("이벤트")
.font(.custom(Font.bold.rawValue, size: 14.7))
.foregroundColor(Color(hex: "eeeeee"))
Spacer()
Image("ic_forward")
.resizable()
.frame(width: 20, height: 20)
}
.padding(.horizontal, 3.3)
.frame(width: cardWidth - 26.7, height: 50)
.contentShape(Rectangle())
.onTapGesture {
AppState.shared.setAppStep(step: .events)
}
Rectangle()
.frame(width: cardWidth - 26.7, height: 0.3)
.foregroundColor(Color(hex: "909090"))
HStack(spacing: 0) {
Text("알림 설정")
.font(.custom(Font.bold.rawValue, size: 14.7))
.foregroundColor(Color(hex: "eeeeee"))
Spacer()
Image("ic_forward")
.resizable()
.frame(width: 20, height: 20)
}
.padding(.horizontal, 3.3)
.frame(width: cardWidth - 26.7, height: 50)
.contentShape(Rectangle())
.onTapGesture {
AppState.shared.setAppStep(step: .notificationSettings)
}
}
.padding(.horizontal, 13.3)
.frame(width: cardWidth)
.background(Color(hex: "222222"))
.cornerRadius(6.7)
.padding(.top, 26.7)
VStack(spacing: 0) {
HStack(spacing: 0) {
Text("이용약관")
.font(.custom(Font.bold.rawValue, size: 14.7))
.foregroundColor(Color(hex: "eeeeee"))
Spacer()
Image("ic_forward")
.resizable()
.frame(width: 20, height: 20)
}
.padding(.horizontal, 3.3)
.frame(width: cardWidth - 26.7, height: 50)
.contentShape(Rectangle())
.onTapGesture {
AppState.shared.setAppStep(step: .terms)
}
Rectangle()
.frame(width: cardWidth - 26.7, height: 0.3)
.foregroundColor(Color(hex: "909090"))
HStack(spacing: 0) {
Text("개인정보처리방침")
.font(.custom(Font.bold.rawValue, size: 14.7))
.foregroundColor(Color(hex: "eeeeee"))
Spacer()
Image("ic_forward")
.resizable()
.frame(width: 20, height: 20)
}
.padding(.horizontal, 3.3)
.frame(width: cardWidth - 26.7, height: 50)
.contentShape(Rectangle())
.onTapGesture {
AppState.shared.setAppStep(step: .privacy)
}
}
.padding(.horizontal, 13.3)
.frame(width: cardWidth)
.background(Color(hex: "222222"))
.cornerRadius(6.7)
.padding(.top, 13.3)
HStack(spacing: 0) {
Text("앱 버전 정보")
.font(.custom(Font.bold.rawValue, size: 14.7))
.foregroundColor(Color(hex: "eeeeee"))
Spacer()
let version = Bundle.main.infoDictionary?["CFBundleShortVersionString"] as? String
Text("Ver \(version!)")
.font(.custom(Font.medium.rawValue, size: 13.3))
.foregroundColor(Color(hex: "eeeeee"))
}
.padding(.horizontal, 16.7)
.frame(width: cardWidth, height: 50)
.background(Color(hex: "222222"))
.cornerRadius(6.7)
.padding(.top, 13.3)
Spacer()
Text("로그아웃")
.font(.custom(Font.bold.rawValue, size: 14.7))
.foregroundColor(Color(hex: "eeeeee"))
.frame(width: cardWidth, height: 50)
.background(Color(hex: "222222"))
.cornerRadius(6.7)
.onTapGesture {
isShowLogoutDialog = true
}
Text("모든 기기에서 로그아웃")
.font(.custom(Font.medium.rawValue, size: 14.7))
.foregroundColor(Color(hex: "777777"))
.padding(.top, 13.3)
.onTapGesture {
isShowLogoutAllDeviceDialog = true
}
Text("회원탈퇴")
.font(.custom(Font.medium.rawValue, size: 14.7))
.foregroundColor(Color(hex: "777777"))
.underline()
.padding(.vertical, 26.7)
.onTapGesture {
}
}
}
if isShowLogoutDialog {
SodaDialog(
title: "알림",
desc: "로그아웃 하시겠어요?",
confirmButtonTitle: "확인",
confirmButtonAction: {
viewModel.logout {
self.isShowLogoutDialog = false
AppState.shared.setAppStep(step: .main)
UserDefaults.reset()
}
},
cancelButtonTitle: "취소",
cancelButtonAction: {
self.isShowLogoutDialog = false
}
)
}
if isShowLogoutAllDeviceDialog {
SodaDialog(
title: "알림",
desc: "모든 기기에서 로그아웃 하시겠어요?",
confirmButtonTitle: "확인",
confirmButtonAction: {
viewModel.logoutAllDevice {
self.isShowLogoutDialog = false
AppState.shared.setAppStep(step: .main)
UserDefaults.reset()
}
},
cancelButtonTitle: "취소",
cancelButtonAction: {
self.isShowLogoutDialog = false
}
)
}
}
.popup(isPresented: $viewModel.isShowPopup, type: .toast, position: .top, autohideIn: 2) {
GeometryReader { geo in
HStack {
Spacer()
Text(viewModel.errorMessage)
.padding(.vertical, 13.3)
.padding(.horizontal, 6.7)
.frame(width: geo.size.width - 66.7, alignment: .center)
.font(.custom(Font.medium.rawValue, size: 12))
.background(Color(hex: "9970ff"))
.foregroundColor(Color.white)
.multilineTextAlignment(.leading)
.fixedSize(horizontal: false, vertical: true)
.cornerRadius(20)
.padding(.top, 66.7)
Spacer()
}
}
}
}
}
struct SettingsView_Previews: PreviewProvider {
static var previews: some View {
SettingsView()
}
}

View File

@ -0,0 +1,97 @@
//
// SettingsViewModel.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import Foundation
import Combine
final class SettingsViewModel: ObservableObject {
private let userRepository = UserRepository()
private var subscription = Set<AnyCancellable>()
@Published var errorMessage = ""
@Published var isShowPopup = false
@Published var isLoading = false
func logout(onSuccess: @escaping () -> Void) {
isLoading = true
userRepository.logout()
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { [unowned self] response in
let responseData = response.data
do {
let jsonDecoder = JSONDecoder()
let decoded = try jsonDecoder.decode(ApiResponseWithoutData.self, from: responseData)
if decoded.success {
onSuccess()
} else {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
}
self.isShowPopup = true
}
} catch {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.isShowPopup = true
}
self.isLoading = false
}
.store(in: &subscription)
}
func logoutAllDevice(onSuccess: @escaping () -> Void) {
isLoading = true
userRepository.logoutAllDevice()
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { [unowned self] response in
let responseData = response.data
do {
let jsonDecoder = JSONDecoder()
let decoded = try jsonDecoder.decode(ApiResponseWithoutData.self, from: responseData)
if decoded.success {
onSuccess()
} else {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
}
self.isShowPopup = true
}
} catch {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.isShowPopup = true
}
self.isLoading = false
}
.store(in: &subscription)
}
}

View File

@ -0,0 +1,13 @@
//
// Terms.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import Foundation
struct Terms: Decodable {
let title: String
let description: String
}

View File

@ -0,0 +1,42 @@
//
// TermsApi.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import Foundation
import Moya
enum TermsApi {
case terms
case privacy
}
extension TermsApi: TargetType {
var baseURL: URL {
return URL(string: BASE_URL)!
}
var path: String {
switch self {
case .terms:
return "/stplat/terms_of_service"
case .privacy:
return "/stplat/privacy_policy"
}
}
var method: Moya.Method {
return .get
}
var task: Task {
return .requestPlain
}
var headers: [String : String]? {
return nil
}
}

View File

@ -0,0 +1,23 @@
//
// TermsRepository.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import Foundation
import CombineMoya
import Combine
import Moya
final class TermsRepository {
private let api = MoyaProvider<TermsApi>()
func getTermsOfService() -> AnyPublisher<Response, MoyaError> {
return api.requestPublisher(.terms)
}
func getPrivacyPolicy() -> AnyPublisher<Response, MoyaError> {
return api.requestPublisher(.privacy)
}
}

View File

@ -0,0 +1,42 @@
//
// TermsView.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import SwiftUI
import RichText
struct TermsView: View {
let isPrivacyPolicy: Bool
@ObservedObject var viewModel = TermsViewModel()
var body: some View {
BaseView(isLoading: $viewModel.isLoading) {
VStack(spacing: 0) {
DetailNavigationBar(title: viewModel.title)
ScrollView(.vertical, showsIndicators: false) {
RichText(html: viewModel.description)
.frame(width: screenSize().width - 26.7)
}
}
}
.onAppear {
if isPrivacyPolicy {
viewModel.getPrivacyPolicy()
} else {
viewModel.getTermsOfService()
}
}
}
}
struct TermsView_Previews: PreviewProvider {
static var previews: some View {
TermsView(isPrivacyPolicy: false)
}
}

View File

@ -0,0 +1,99 @@
//
// TermsViewModel.swift
// SodaLive
//
// Created by klaus on 2023/08/10.
//
import Foundation
import Combine
final class TermsViewModel: ObservableObject {
private let repository = TermsRepository()
private var subscription = Set<AnyCancellable>()
@Published var title: String = ""
@Published var description: String = ""
@Published var errorMessage = ""
@Published var isShowPopup = false
@Published var isLoading = false
func getTermsOfService() {
isLoading = true
repository.getTermsOfService()
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { [unowned self] response in
self.isLoading = false
let responseData = response.data
do {
let jsonDecoder = JSONDecoder()
let decoded = try jsonDecoder.decode(ApiResponse<Terms>.self, from: responseData)
if let data = decoded.data, decoded.success {
self.title = data.title
self.description = data.description
} else {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
}
self.isShowPopup = true
}
} catch {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.isShowPopup = true
}
}
.store(in: &subscription)
}
func getPrivacyPolicy() {
isLoading = true
repository.getPrivacyPolicy()
.sink { result in
switch result {
case .finished:
DEBUG_LOG("finish")
case .failure(let error):
ERROR_LOG(error.localizedDescription)
}
} receiveValue: { [unowned self] response in
self.isLoading = false
let responseData = response.data
do {
let jsonDecoder = JSONDecoder()
let decoded = try jsonDecoder.decode(ApiResponse<Terms>.self, from: responseData)
if let data = decoded.data, decoded.success {
self.title = data.title
self.description = data.description
} else {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
}
self.isShowPopup = true
}
} catch {
self.errorMessage = "다시 시도해 주세요.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.isShowPopup = true
}
}
.store(in: &subscription)
}
}

View File

@ -14,6 +14,10 @@ enum UserApi {
case findPassword(request: ForgotPasswordRequest)
case searchUser(nickname: String)
case getMypage
case getMemberInfo
case notification(request: UpdateNotificationSettingRequest)
case logout
case logoutAllDevice
}
extension UserApi: TargetType {
@ -37,15 +41,27 @@ extension UserApi: TargetType {
case .getMypage:
return "/member/mypage"
case .getMemberInfo:
return "/member/info"
case .notification:
return "/member/notification"
case .logout:
return "/member/logout"
case .logoutAllDevice:
return "/member/logout/all"
}
}
var method: Moya.Method {
switch self {
case .login, .signUp, .findPassword:
case .login, .signUp, .findPassword, .notification, .logout, .logoutAllDevice:
return .post
case .searchUser, .getMypage:
case .searchUser, .getMypage, .getMemberInfo:
return .get
}
}
@ -66,6 +82,12 @@ extension UserApi: TargetType {
case .getMypage:
return .requestParameters(parameters: ["container" : "ios"], encoding: URLEncoding.queryString)
case .getMemberInfo, .logout, .logoutAllDevice:
return .requestPlain
case .notification(let request):
return .requestJSONEncodable(request)
}
}

View File

@ -32,4 +32,28 @@ final class UserRepository {
func getMypage() -> AnyPublisher<Response, MoyaError> {
return api.requestPublisher(.getMypage)
}
func getMemberInfo() -> AnyPublisher<Response, MoyaError> {
return api.requestPublisher(.getMemberInfo)
}
func updateNotificationSettings(live: Bool? = nil, uploadContent: Bool? = nil, message: Bool? = nil) -> AnyPublisher<Response, MoyaError> {
return api.requestPublisher(
.notification(
request: UpdateNotificationSettingRequest(
live: live,
uploadContent: uploadContent,
message: message
)
)
)
}
func logout() -> AnyPublisher<Response, MoyaError> {
return api.requestPublisher(.logout)
}
func logoutAllDevice() -> AnyPublisher<Response, MoyaError> {
return api.requestPublisher(.logoutAllDevice)
}
}