프로필과 시리즈 문구를 I18n으로 정리

프로필/시리즈 화면의 문자열을 I18n 및 Localizable로 통일한다.
채널 공유·예약·유료 라이브 관련 안내 문구를 다국어로 제공한다.
This commit is contained in:
Yu Sung
2025-12-18 17:52:02 +09:00
parent 982a17bb41
commit 8bcbd3aca0
7 changed files with 182 additions and 90 deletions

View File

@@ -23,7 +23,7 @@ struct UserProfileFanTalkAllView: View {
GeometryReader { proxy in
BaseView(isLoading: $viewModel.isLoading) {
VStack(spacing: 0) {
DetailNavigationBar(title: "팬 Talk 전체보기")
DetailNavigationBar(title: String(localized: "팬 Talk 전체보기"))
VStack(alignment: .leading, spacing: 0) {
HStack(spacing: 6.7) {

View File

@@ -65,7 +65,7 @@ struct UserProfileView: View {
}
if creatorProfile.creator.creatorId == UserDefaults.int(forKey: .userId) {
Text("팔로워 리스트")
Text(I18n.MemberChannel.followersList)
.font(.custom(Font.preBold.rawValue, size: 16))
.foregroundColor(Color.black)
.padding(.vertical, 8)
@@ -77,7 +77,7 @@ struct UserProfileView: View {
}
} else {
VStack(alignment: .leading, spacing: 9.3) {
Text("팔로워 \(creatorProfile.creator.notificationRecipientCount)")
Text(I18n.MemberChannel.followerCount(creatorProfile.creator.notificationRecipientCount))
.font(.custom(Font.preMedium.rawValue, size: 16))
.foregroundColor(Color.white)
}
@@ -98,7 +98,7 @@ struct UserProfileView: View {
.cornerRadius(12)
VStack(alignment: .leading, spacing: 8) {
Text("최신 콘텐츠")
Text(I18n.Common.latestContent)
.font(.custom(Font.preMedium.rawValue, size: 12))
.foregroundColor(.button)
.padding(.horizontal, 7)
@@ -113,7 +113,7 @@ struct UserProfileView: View {
HStack(spacing: 8) {
if item.isScheduledToOpen {
Text("오픈예정")
Text(I18n.Common.openScheduled)
.font(.custom(Font.preMedium.rawValue, size: 12))
.foregroundColor(Color(hex: "3bb9f1"))
.padding(2.6)
@@ -136,7 +136,7 @@ struct UserProfileView: View {
.cornerRadius(2.6)
if item.isPointAvailable {
Text("포인트")
Text(I18n.Common.points)
.font(.custom(Font.preMedium.rawValue, size: 12))
.foregroundColor(.white)
.padding(2.6)
@@ -189,7 +189,7 @@ struct UserProfileView: View {
if creatorProfile.creator.creatorId == UserDefaults.int(forKey: .userId) || creatorProfile.liveRoomList.count > 0 {
VStack(alignment: .leading, spacing: 14) {
HStack(spacing: 0) {
Text("라이브")
Text(I18n.MemberChannel.liveHeader)
.font(.custom(Font.preBold.rawValue, size: 26))
.foregroundColor(Color.white)
@@ -198,7 +198,7 @@ struct UserProfileView: View {
if creatorProfile.creator.creatorId == UserDefaults.int(forKey: .userId) {
HStack(spacing: 8) {
Text("룰렛 설정")
Text(I18n.MemberChannel.rouletteSettings)
.font(.custom(Font.preBold.rawValue, size: 16))
.foregroundColor(Color.grayee)
.padding(.vertical, 12)
@@ -207,7 +207,7 @@ struct UserProfileView: View {
.cornerRadius(12)
.onTapGesture { isShowRouletteSettings = true }
Text("메뉴 설정")
Text(I18n.MemberChannel.menuSettings)
.font(.custom(Font.preBold.rawValue, size: 16))
.foregroundColor(Color.grayee)
.padding(.vertical, 12)
@@ -224,7 +224,7 @@ struct UserProfileView: View {
liveRoomList: creatorProfile.liveRoomList,
onClickParticipant: { liveRoom in
if creatorProfile.creator.creatorId == UserDefaults.int(forKey: .userId) {
viewModel.errorMessage = "현재 라이브 중입니다."
viewModel.errorMessage = I18n.MemberChannel.liveOnNow
viewModel.isShowPopup = true
} else {
AppState.shared.isShowPlayer = false
@@ -235,7 +235,7 @@ struct UserProfileView: View {
},
onClickReservation: { liveRoom in
if creatorProfile.creator.creatorId == UserDefaults.int(forKey: .userId) {
viewModel.errorMessage = "내가 만든 라이브는 예약할 수 없습니다."
viewModel.errorMessage = I18n.MemberChannel.cannotReserveOwnLive
viewModel.isShowPopup = true
} else {
viewModel.reservationLiveRoom(roomId: liveRoom.roomId)
@@ -259,7 +259,7 @@ struct UserProfileView: View {
VStack(alignment: .leading, spacing: 14) {
HStack(spacing: 0) {
Text("커뮤니티")
Text(I18n.MemberChannel.communityHeader)
.font(.custom(Font.preBold.rawValue, size: 26))
.foregroundColor(Color.white)
.padding(.horizontal, 24)
@@ -430,14 +430,14 @@ struct UserProfileView: View {
if viewModel.isShowCheersDeleteView {
SodaDialog(
title: "응원글 삭제",
desc: "삭제하시겠습니까?",
confirmButtonTitle: "삭제",
title: I18n.MemberChannel.cheersDeleteTitle,
desc: I18n.Common.confirmDeleteQuestion,
confirmButtonTitle: I18n.Common.delete,
confirmButtonAction: {
viewModel.deleteCheers(creatorId: userId)
viewModel.isShowCheersDeleteView = false
},
cancelButtonTitle: "취소",
cancelButtonTitle: I18n.Common.cancel,
cancelButtonAction: { viewModel.isShowCheersDeleteView = false }
)
}

View File

@@ -52,7 +52,7 @@ final class UserProfileViewModel: ObservableObject {
@Published var liveStartDate: String? = nil
@Published var nowDate: String? = nil
let paymentDialogCancelTitle = "취소"
let paymentDialogCancelTitle = I18n.Common.cancel
func getCreatorProfile(userId: Int) {
creatorProfile = nil
@@ -75,7 +75,7 @@ final class UserProfileViewModel: ObservableObject {
if let data = decoded.data, decoded.success {
self.creatorProfile = data
self.navigationTitle = "\(data.creator.nickname)님의 채널"
self.navigationTitle = I18n.MemberChannel.channelTitle(data.creator.nickname)
self.communityPostList.removeAll()
self.communityPostList.append(contentsOf: data.communityPostList)
@@ -113,7 +113,7 @@ final class UserProfileViewModel: ObservableObject {
func reservationLiveRoom(roomId: Int) {
getRoomDetail(roomId: roomId) { [unowned self] in
if ($0.manager.id == UserDefaults.int(forKey: .userId)) {
self.errorMessage = "내가 만든 라이브는 예약할 수 없습니다."
self.errorMessage = I18n.MemberChannel.cannotReserveOwnLive
self.isShowPopup = true
} else {
if $0.isPrivateRoom {
@@ -125,9 +125,9 @@ final class UserProfileViewModel: ObservableObject {
if ($0.price == 0 || $0.isPaid) {
self.reservation(roomId: roomId)
} else {
self.paymentDialogTitle = "\($0.price)캔으로 예약"
self.paymentDialogDesc = "'\($0.title)' 라이브에 참여하기 위해 결제합니다."
self.paymentDialogConfirmTitle = "결제 후 예약하기"
self.paymentDialogTitle = I18n.MemberChannel.reserveWithCansTitle($0.price)
self.paymentDialogDesc = I18n.MemberChannel.reservePaymentDesc($0.title)
self.paymentDialogConfirmTitle = I18n.MemberChannel.reservePaymentConfirmTitle
self.paymentDialogConfirmAction = { [unowned self] in
hidePaymentPopup()
reservation(roomId: roomId)
@@ -216,12 +216,12 @@ final class UserProfileViewModel: ObservableObject {
if hours >= 1 {
self.liveStartDate = beginDate.convertDateFormat(dateFormat: "yyyy-MM-dd, HH:mm")
self.nowDate = now.convertDateFormat(dateFormat: "yyyy-MM-dd, HH:mm")
self.paymentDialogDesc2 = "라이브를 시작한 지 \(hours)시간 \(minutes)분이 지났습니다. 라이브에 입장 후 30분 이내에 라이브가 종료될 수도 있습니다."
self.paymentDialogDesc2 = I18n.MemberChannel.elapsedLiveWarning(hours: hours, minutes: minutes)
}
self.paymentDialogTitle = "유료 라이브 입장"
self.paymentDialogDesc = "\($0.price)캔을 차감하고\n라이브에 입장 하시겠습니까?"
self.paymentDialogConfirmTitle = "결제 후 참여하기"
self.paymentDialogTitle = I18n.MemberChannel.paidLiveEnterTitle
self.paymentDialogDesc = I18n.MemberChannel.paidLiveEnterDesc($0.price)
self.paymentDialogConfirmTitle = I18n.MemberChannel.paidLiveConfirmTitle
self.paymentDialogConfirmAction = { [unowned self] in
hidePaymentPopup()
self.enterRoom(roomId: roomId)
@@ -267,13 +267,13 @@ final class UserProfileViewModel: ObservableObject {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = "라이브에 입장하지 못했습니다.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.errorMessage = I18n.MemberChannel.enterLiveFailed
}
self.isShowPopup = true
}
} catch {
self.errorMessage = "라이브에 입장하지 못했습니다.\n계속 같은 문제가 발생할 경우 고객센터로 문의 주시기 바랍니다."
self.errorMessage = I18n.MemberChannel.enterLiveFailed
self.isShowPopup = true
}
}
@@ -383,13 +383,13 @@ final class UserProfileViewModel: ObservableObject {
if let message = decoded.message {
self.errorMessage = message
} else {
self.errorMessage = "라이브 정보를 가져오지 못했습니다.\n다시 시도해 주세요."
self.errorMessage = I18n.MemberChannel.fetchLiveInfoFailed
}
self.isShowPopup = true
}
} catch {
self.errorMessage = "라이브 정보를 가져오지 못했습니다.\n다시 시도해 주세요."
self.errorMessage = I18n.MemberChannel.fetchLiveInfoFailed
self.isShowPopup = true
}
@@ -408,10 +408,10 @@ final class UserProfileViewModel: ObservableObject {
]
if let shareUrl = createOneLinkUrlWithURLComponents(params: params) {
self.shareMessage = "보이스온 \(nickname)님의 채널입니다.\n\(shareUrl)"
self.shareMessage = I18n.MemberChannel.shareChannelMessage(nickname) + "\n\(shareUrl)"
self.isShowShareView = true
} else {
self.errorMessage = "공유링크를 생성하지 못했습니다.\n다시 시도해 주세요."
self.errorMessage = I18n.MemberChannel.shareLinkCreateFailed
self.isShowPopup = true
}
@@ -438,7 +438,7 @@ final class UserProfileViewModel: ObservableObject {
if decoded.success {
getCreatorProfile(userId: userId)
self.errorMessage = "차단하였습니다."
self.errorMessage = I18n.MemberChannel.userBlocked
} else {
if let message = decoded.message {
self.errorMessage = message