Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Feat] Button, Navigation 컴포넌트 구현 #22

Open
wants to merge 4 commits into
base: develop
Choose a base branch
from

Conversation

syss220211
Copy link
Member

@syss220211 syss220211 commented Jan 15, 2025

📌 What is the PR?

  • 할당 컴포넌트 중 버튼과 네비게이션 컴포넌트를 작성했습니다.
  • 네비게이션은 현재 나와있는 케이스만 우선 정의하였습니다.

🪄 Changes

  • 할당 컴포넌트 중 버튼(이미지 옵셔널 + 텍스트), 아이콘 버튼, 바텀 버튼 컴포넌트를 작성했습니다.

🌐 Common Changes

  • Tuist 의 xcconfig path를 Tuist/Config 하위로 수정하였습니다.

🔥 PR Point

  1. Buttom 컴포넌트를 작성했습니다.
  • 디자인 시스템 상 크게 enable, disable 으로 나뉘며, 각각 4가지, 2가지 색상 타입을 가지고 있습니다.
  • 버튼은 xlarge-xsmall까지 5가지의 사이즈를 가지고 있으며, small과 xsmall은 텍스트만 존재합니다.
  • 버튼 사이즈별로 또 다른 enum을 생성시 너무 과한 쪼개기처럼 보여서 Image를 옵셔널 처리하였습니다.
  • 버튼의 속성과 관련된 코드는 TButtonInfo.swift 에 정의해 놓았습니다.
  • action과 관련한 코드 작성의 편의성을 위해서 action을 옵셔널로 선언하였습니다.
  • isEnable을 통해 버튼의 tap 여부가 조절 가능합니다.
    사용법은 아래와 같습니다.
 TButton(state: .disable(.grayDisable), config: .large, title: "title")
    .tap {
        print("gray tapped")
    }

TIcnButton(image: .icnHeartFilled, imageSize: 24, config: .large)
    .tap {
        print("icn tapped")
    }
    .isEnable(text.count > 3) // 조건 설정
  1. Navigation 컴포넌트를 작성했습니다.
  • action과 관련한 코드 작성의 편의성을 위해서 action을 옵셔널로 선언하였습니다.
    사용법은 아래와 같습니다.
TNavigation(type: .LButtonRTextWithTitle(.icnArrowLeft, "title", "건너뛰기"))
    .leftTap {
        print("왼쪽")
    }
    .rightTap {
        print("오른쪽")
    }

📸 Screenshot

기능 스크린샷
GIF

🙆🏻 To Reviewers

  • 버튼과 관련해서 더 좋은 방법이 있으면 피드백 부탁드립니다.
image

💭 Related Issues

Copy link
Contributor

@FpRaArNkK FpRaArNkK left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🎉대서연님의 첫 PR🎉
고생 많으셨습니다! 몇 가지 코멘트들을 달아놓았는데 확인 부탁드려요!

Comment on lines 14 to 17
public enum ButtonState {
case `default`(DefaultStyle)
case disable(DisableStyle)
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ButtonState 대신 Style을 isEnabled: Bool과 같은 친구를 enum 연관값으로 받아서 사용하는 건 어떨까요?

     case primary(isEnabled: Bool)
     case gray(isEnabled: Bool)
     case outline(isEnabled: Bool)
     case red(isEnabled: Bool)

과 같이 선언하고, 계산 프로퍼티에서 삼항 연산자로 값 줘도 좋을 것 같아요!

 var backgound: Color {
         switch self {
         case .primary(let isEnabled):
             return isEnabled ? .neutral900 : .neutral200
...

이런 느낌으로요..! 디자인 시스템이랑 1:1 매칭도 되고 좋지 않을까요?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

해당 부분 무조건 1:1 매칭되는 컴포넌트인지 한번 확인하고 해당되면 고려해보겠습니다.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

해당 내용은 모두 TButton의 UI 스타일을 규정하는 내용인 것 같은데, TButton의 Extension으로 넣으면 어떨까요?

Comment on lines 74 to 79
.padding(.horizontal, 20)
.background(backgroundColor)
.clipShape(RoundedRectangle(cornerRadius: config.radius))
.overlay {
RoundedRectangle(cornerRadius: config.radius)
.stroke(borderColor, lineWidth: 1.5)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

padding이나 stroke 같은 내용도 버튼 스타일 정의에 포함되면 좋을 것 같아요!

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

네 앞선 리뷰와 함께 UI 스타일 관련 부분 반영해보겠습니다.

Comment on lines 34 to 51
/// TButton의 초기화 메서드
/// - Parameters:
/// - state: 버튼의 상태
/// - config: 버튼 구성 (크기, 글꼴 등)
/// - title: 버튼 제목
/// - image: 버튼 이미지 (옵셔널)
/// - imageSize: 이미지 크기 (옵셔널)
/// - isEnable: 활성화 여부 (기본값: `false`)
/// - action: 버튼 탭 시 동작 (옵셔널)
public init(
state: ButtonState,
config: ButtonConfiguration,
title: String,
image: ImageResource? = nil,
imageSize: CGFloat? = nil,
isEnable: Bool = false,
action: (() -> Void)? = nil
) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

init 때 한 번에 입력받는 양이 많은 것 같은데,
init때는 필수적인 값들만 챙기고, 최대한 기본 Button 선언과 비슷하게 가면서 -
밑에 정의하신 것 처럼 Modifier에서 추가 값들을 입력받는건 어떨까요?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

기존 button 처럼 최대한 title만 입력 받는 형태로 수정해보겠습니다. View 타입 자체에 추가 되는 Modifier말고 TButton 한정으로 적용하기 위해서 option action, 초기값이 있는 isEnable로 설정했어요! View 타입 자체보다는 TButton 타입에 한정해서 정의하는게 더 좋지 않을까요?

Comment on lines 62 to 72
HStack(spacing: 4) {
if let image = image, let imageSize = imageSize {
Image(image)
.resizable()
.aspectRatio(contentMode: .fit)
.frame(width: imageSize, height: imageSize)
}

Text(title)
.typographyStyle(config.font, with: textColor)
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

저희 디자인 시스템 상에서 이미지가 우측에도 있는 케이스가 있어서, 반영해주시면 좋을 것 같아요!

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

좌우로 있는지 확인을 못했네요. 바로 추가해놓겠습니다!

Comment on lines 40 to 50
Text(title)
.typographyStyle(.heading4, with: .neutral900)
.frame(maxWidth: .infinity, alignment: .center)
.overlay(alignment: .topLeading) {
Image(imageResource)
.resizable()
.frame(width: 32, height: 32)
}
.onTapGesture {
leftAction?()
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

이렇게 되면 제목 탭해도 onTapGesture가 실행되지 않을까요?!

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

엇 Image에 달아야하는데 실수했네요... 수정해놓겠습니다. 감사합니다 !

Comment on lines 35 to 107
public var body: some View {
switch type {
/// 왼쪽 이미지 버튼, 센터 타이틀
case .LButtonWithTitle(let imageResource, let title):
HStack {
Text(title)
.typographyStyle(.heading4, with: .neutral900)
.frame(maxWidth: .infinity, alignment: .center)
.overlay(alignment: .topLeading) {
Image(imageResource)
.resizable()
.frame(width: 32, height: 32)
}
.onTapGesture {
leftAction?()
}
}
.padding(.init(top: 22.5, leading: 20, bottom: 10.5, trailing: 20))

/// 왼쪽 이미지 버튼, 센터 타이틀, 오른쪽 이미지 버튼
case .LRButtonWithTitle(let LImage, let title, let RImage):
HStack {
Image(LImage)
.resizable()
.frame(width: 32, height: 32)
.onTapGesture {
leftAction?()
}

Text(title)
.typographyStyle(.heading4, with: .neutral900)
.frame(maxWidth: .infinity, alignment: .center)

Image(RImage)
.resizable()
.frame(width: 32, height: 32)
.onTapGesture {
rightAction?()
}
}
.padding(.init(top: 22.5, leading: 20, bottom: 10.5, trailing: 20))

/// 왼쪽 이미지 버튼, 센터 타이틀, 오른쪽 텍스트 버튼
case .LButtonRTextWithTitle(let LImage, let title, let RText):
HStack {
Image(LImage)
.resizable()
.frame(width: 32, height: 32)
.onTapGesture {
leftAction?()
}

Text(title)
.typographyStyle(.heading4, with: .neutral900)
.frame(maxWidth: .infinity, alignment: .center)

Text(RText)
.typographyStyle(.body2Medium, with: .neutral400)
.onTapGesture {
rightAction?()
}
}
.padding(.init(top: 22.5, leading: 20, bottom: 10.5, trailing: 20))

/// 타이틀
case .Title(let title):
Text(title)
.typographyStyle(.heading4, with: .neutral900)
.frame(maxWidth: .infinity, alignment: .center)
.padding(.init(top: 22.5, leading: 20, bottom: 10.5, trailing: 20))
}
}
}
Copy link
Contributor

@FpRaArNkK FpRaArNkK Jan 15, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

해당 내용에서 겹치는 내용들이 있는 것 같은데,
특정 뷰만 동적으로 생성하도록 작성하면 더 명확하고 유지보수에 좋을 것 같아요..!

@FpRaArNkK FpRaArNkK added the ✨Feat 새로운 기능 구현 (새로운 로직 추가, UI 구현 등) label Jan 15, 2025
@stealmh
Copy link
Member

stealmh commented Jan 16, 2025

안녕하세요 🙌 잘부탁드립니다~~
천천히 살펴보고 궁금한것들 남겨놓겠습니다

Comment on lines 34 to 46
public var body: some View {
Text(title)
.typographyStyle(.heading4, with: .neutral50)
.padding(.vertical, 20)
.frame(maxWidth: .infinity)
.background(isEnable ? Color.neutral900 : Color.neutral300)
.ignoresSafeArea()
.onTapGesture {
if isEnable {
action?()
}
}
}
Copy link
Member

@stealmh stealmh Jan 16, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

버튼들 공통질문인데 왜 onTapGesture로 액션처리하게 구성하셨나요?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

단순하게 앱 내에서 사용되는 버튼이 클릭 처리라고 생각해서 onTapGeture로 충분하다고 판단했던 것 같아요. 기존에도 Text에 onTapGesture를 자주 달아 써서 Button을 활용할 생각은 안 해봤네요.. 🤔 해당 부분을 Button을 사용해서 구현하는 쪽으로 수정해보겠습니다.

Comment on lines 53 to 57
func tap(action: @escaping (() -> Void)) -> Self {
var copy: Self = self
copy.action = action
return copy
}
Copy link
Member

@stealmh stealmh Jan 16, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

�여기도 tap은 action이랑 동일한데 선언형으로 코드 작성하고 싶어서 함수를 만드신건지 궁금합니다.
스유도 버튼 액션을 클로저로 받고있지 선언형으로 받고있지는 않잖아요 이부분 궁금합니다

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

해당 부분은 선언형 코드로 작성하고 싶어서 함수를 만든게 맞습니다. 예를 들어, 네비바 같이 버튼이 한번에 2가지 있는 컴포넌트를 만들 때 클로저를 2번 사용하게 되면 가독성이 안 좋아진다고 생각해서 해당 부분을 해결하기 위해서 이 방법을 선택했어요. 근데 action이 한개인 경우는 그닥... 이렇게까지 안 해도 될것 같네요. 기존 Button을 활용한다면 더더욱 그럴것 같구요. 하나의 액션만 있는 컴포넌트는 기존 스유 방식을 반영해볼게요!

Comment on lines 53 to 66
func tap(action: @escaping (() -> Void)) -> Self {
var copy: Self = self
copy.action = action
return copy
}

/// 버튼 활성화 상태를 설정하는 메서드
/// - Parameter isEnable: 활성화 여부
/// - Returns: 업데이트된 TBottomButton
func isEnable(_ isEnable: Bool) -> Self {
var copy: Self = self
copy.isEnable = isEnable
return copy
}
Copy link
Member

@stealmh stealmh Jan 16, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

disabled modifier가 이미 해당 함수를 대체한다고 생각하는데 새로 만드신 이유가 있을까요??

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

텍스트라고 생각하다보니 따로 만들었는데 그럼에도 불구하고 굳이 필요하진 않겠군요... ? 수정해도 될것 같습니다. 감사합니다!

Comment on lines 22 to 26
/// 버튼에 표시될 이미지 (옵셔널)
public let image: ImageResource?

/// 이미지의 크기 (옵셔널)
public let imageSize: CGFloat?
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

서로 관련있는것들끼리 묶어서 추상화하고 이 결과를 optional로 받는 형태가 가장 이상적일 것 같아요.
하단에 image, imagesize가 둘다 있어야 이미지를 설정하는데 이렇게 생성자를 넣다보면 서로 연관되어있는지 헷갈릴 때가 많아요.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

imge, imageSize 를 한번 더 묶는게 사용성이 더 좋을 것 같다는 말씀이 맞을까요??

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

넵 맞습니다!

Comment on lines 31 to 32
/// 버튼 탭 시 수행할 동작 (옵셔널)
public var action: (() -> Void)?
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

구조체 이름이 Button인데 action이 옵셔널인 경우는 대체 어떤 케이스에서 사용되나요??

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

선언형처럼 구현하기 위해서 클로저를 안 받을려고 옵셔널로 구현하였습니다. 이 부분은 기존 Button를 활용하는 방향으로 수정하면 없어질것 같아요!

public var body: some View {
switch type {
/// 왼쪽 이미지 버튼, 센터 타이틀
case .LButtonWithTitle(let imageResource, let title):
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

case let을 통해 많아지는 연관값을 정리해보시져!

Comment on lines 56 to 75
HStack {
Image(LImage)
.resizable()
.frame(width: 32, height: 32)
.onTapGesture {
leftAction?()
}

Text(title)
.typographyStyle(.heading4, with: .neutral900)
.frame(maxWidth: .infinity, alignment: .center)

Image(RImage)
.resizable()
.frame(width: 32, height: 32)
.onTapGesture {
rightAction?()
}
}
.padding(.init(top: 22.5, leading: 20, bottom: 10.5, trailing: 20))
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

case 내부에서 뷰를 그리지 마시고 함수로 빼서 외부에서 그리는건 어떠신가요??

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

민서님 리뷰와 종합해서 수정 후 반영해보겠습니다-!

Comment on lines 130 to 139
public enum TNavigationCase {
/// 왼쪽 이미지 버튼, 센터 타이틀
case LButtonWithTitle(ImageResource, String)
/// 왼쪽 이미지 버튼, 센터 타이틀, 오른쪽 이미지 버튼
case LRButtonWithTitle(ImageResource, String, ImageResource)
/// 왼쪽 이미지 버튼, 센터 타이틀, 오른쪽 텍스트 버튼
case LButtonRTextWithTitle(ImageResource, String, String)
/// 타이틀
case Title(String)
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

연관값 이름을 정해주는 것 어떠신가요??

스크린샷 2025-01-16 오후 1 26 06 스크린샷 2025-01-16 오후 1 30 58

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

네 추가하는게 좋을 것 같네요. 감사합니다!

Copy link
Member

@stealmh stealmh left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

궁금한것들 남겨뒀습니다~! 고생하셨어요

Copy link
Member Author

@syss220211 syss220211 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

말씀해주신 부분들 모두 참고해서 코드 개선해볼게요!
피드백 감사합니다 - !! 😊

+) 민호님 인사가 늦었네요.
안녕하세요! 앞으로 잘 부탁드립니다. 🙌🏻

Comment on lines 34 to 46
public var body: some View {
Text(title)
.typographyStyle(.heading4, with: .neutral50)
.padding(.vertical, 20)
.frame(maxWidth: .infinity)
.background(isEnable ? Color.neutral900 : Color.neutral300)
.ignoresSafeArea()
.onTapGesture {
if isEnable {
action?()
}
}
}
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

단순하게 앱 내에서 사용되는 버튼이 클릭 처리라고 생각해서 onTapGeture로 충분하다고 판단했던 것 같아요. 기존에도 Text에 onTapGesture를 자주 달아 써서 Button을 활용할 생각은 안 해봤네요.. 🤔 해당 부분을 Button을 사용해서 구현하는 쪽으로 수정해보겠습니다.

Comment on lines 53 to 57
func tap(action: @escaping (() -> Void)) -> Self {
var copy: Self = self
copy.action = action
return copy
}
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

해당 부분은 선언형 코드로 작성하고 싶어서 함수를 만든게 맞습니다. 예를 들어, 네비바 같이 버튼이 한번에 2가지 있는 컴포넌트를 만들 때 클로저를 2번 사용하게 되면 가독성이 안 좋아진다고 생각해서 해당 부분을 해결하기 위해서 이 방법을 선택했어요. 근데 action이 한개인 경우는 그닥... 이렇게까지 안 해도 될것 같네요. 기존 Button을 활용한다면 더더욱 그럴것 같구요. 하나의 액션만 있는 컴포넌트는 기존 스유 방식을 반영해볼게요!

Comment on lines 53 to 66
func tap(action: @escaping (() -> Void)) -> Self {
var copy: Self = self
copy.action = action
return copy
}

/// 버튼 활성화 상태를 설정하는 메서드
/// - Parameter isEnable: 활성화 여부
/// - Returns: 업데이트된 TBottomButton
func isEnable(_ isEnable: Bool) -> Self {
var copy: Self = self
copy.isEnable = isEnable
return copy
}
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

텍스트라고 생각하다보니 따로 만들었는데 그럼에도 불구하고 굳이 필요하진 않겠군요... ? 수정해도 될것 같습니다. 감사합니다!

Comment on lines 22 to 26
/// 버튼에 표시될 이미지 (옵셔널)
public let image: ImageResource?

/// 이미지의 크기 (옵셔널)
public let imageSize: CGFloat?
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

imge, imageSize 를 한번 더 묶는게 사용성이 더 좋을 것 같다는 말씀이 맞을까요??

Comment on lines 31 to 32
/// 버튼 탭 시 수행할 동작 (옵셔널)
public var action: (() -> Void)?
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

선언형처럼 구현하기 위해서 클로저를 안 받을려고 옵셔널로 구현하였습니다. 이 부분은 기존 Button를 활용하는 방향으로 수정하면 없어질것 같아요!

Comment on lines 56 to 75
HStack {
Image(LImage)
.resizable()
.frame(width: 32, height: 32)
.onTapGesture {
leftAction?()
}

Text(title)
.typographyStyle(.heading4, with: .neutral900)
.frame(maxWidth: .infinity, alignment: .center)

Image(RImage)
.resizable()
.frame(width: 32, height: 32)
.onTapGesture {
rightAction?()
}
}
.padding(.init(top: 22.5, leading: 20, bottom: 10.5, trailing: 20))
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

민서님 리뷰와 종합해서 수정 후 반영해보겠습니다-!

Comment on lines 130 to 139
public enum TNavigationCase {
/// 왼쪽 이미지 버튼, 센터 타이틀
case LButtonWithTitle(ImageResource, String)
/// 왼쪽 이미지 버튼, 센터 타이틀, 오른쪽 이미지 버튼
case LRButtonWithTitle(ImageResource, String, ImageResource)
/// 왼쪽 이미지 버튼, 센터 타이틀, 오른쪽 텍스트 버튼
case LButtonRTextWithTitle(ImageResource, String, String)
/// 타이틀
case Title(String)
}
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

네 추가하는게 좋을 것 같네요. 감사합니다!

Copy link
Contributor

@FpRaArNkK FpRaArNkK left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

고생 많으셨습니다 ! 달려봅시다 🔥

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
✨Feat 새로운 기능 구현 (새로운 로직 추가, UI 구현 등)
Projects
None yet
Development

Successfully merging this pull request may close these issues.

[TNT-112] 디자인 시스템 및 컴포넌트 작업
3 participants