iOS 접근성 (SwiftUI)

[iOS - SwiftUI] 선언형 프로그래밍 관점으로 코딩하는 방법 (명령형 vs 선언형, 선언적 코딩)

jake-kim 2024. 12. 19. 01:45

명령형 vs 선언형 코드 작성의 흐름

  • 명령형 프로그래밍
    • "어떻게" UI를 업데이트할지 명시적으로 작성
let label = UILabel()
label.text = "Hello, World!"
label.textColor = .blue
label.frame = CGRect(x: 0, y: 0, width: 200, height: 50)
view.addSubview(label)

 

  • 선언형 프로그래밍
    • 좀 더 코드가 뷰의 형태와 닮은 형태로 작성
struct ContentView: View {
    var body: some View {
        Text("Hello, World!")
            .foregroundColor(.blue)
            .frame(width: 200, height: 50)
    }
}

 

 

ex) 카운터 뷰 만들기

  • 선언형 프로그래밍 SwiftUI로 만들면, 뷰의 배치 위치와 동일하게 Text, Button, Button이 나열되어 있고 마치 코드가 뷰의 형태와 닮은 형태처럼 되어서 더욱 직관적인 프로그래밍이 가능
struct ContentView: View {
    @State private var count = 0

    var body: some View {
        VStack(spacing: 20) {
            Text("\(count)")
                .font(.largeTitle)
                .foregroundColor(count < 0 ? .red : .green)
                .padding()

            Button(action: {
                count += 1
            }) {
                Text("Increment")
                    .padding()
                    .background(Color.blue)
                    .foregroundColor(.white)
                    .cornerRadius(10)
            }

            Button(action: {
                count -= 1
            }) {
                Text("Decrement")
                    .padding()
                    .background(Color.gray)
                    .foregroundColor(.white)
                    .cornerRadius(10)
            }
        }
        .padding()
    }
}
  • 반면에 명령형 프로그래밍 UIKit으로 작성하면 각 뷰의 속성 (font, addTarget)작성하는 곳과 layout을 작성하는 곳 (오토레이아웃)코드가 분리되어 있어서 직관적이지 않은 단점이 존재
import UIKit

class CounterViewController: UIViewController {
    private var count = 0
    private let countLabel = UILabel()
    private let incrementButton = UIButton(type: .system)
    private let decrementButton = UIButton(type: .system)

    override func viewDidLoad() {
        super.viewDidLoad()
        view.backgroundColor = .white

        countLabel.text = "\(count)"
        countLabel.textAlignment = .center
        countLabel.font = UIFont.systemFont(ofSize: 32)
        countLabel.translatesAutoresizingMaskIntoConstraints = false
        view.addSubview(countLabel)

        incrementButton.setTitle("Increment", for: .normal)
        incrementButton.addTarget(self, action: #selector(incrementCount), for: .touchUpInside)
        incrementButton.translatesAutoresizingMaskIntoConstraints = false
        view.addSubview(incrementButton)

        decrementButton.setTitle("Decrement", for: .normal)
        decrementButton.addTarget(self, action: #selector(decrementCount), for: .touchUpInside)
        decrementButton.translatesAutoresizingMaskIntoConstraints = false
        view.addSubview(decrementButton)

        NSLayoutConstraint.activate([
            countLabel.centerXAnchor.constraint(equalTo: view.centerXAnchor),
            countLabel.centerYAnchor.constraint(equalTo: view.centerYAnchor, constant: -50),
            incrementButton.centerXAnchor.constraint(equalTo: view.centerXAnchor),
            incrementButton.topAnchor.constraint(equalTo: countLabel.bottomAnchor, constant: 20),
            decrementButton.centerXAnchor.constraint(equalTo: view.centerXAnchor),
            decrementButton.topAnchor.constraint(equalTo: incrementButton.bottomAnchor, constant: 20)
        ])
    }

    @objc private func incrementCount() {
        count += 1
        updateUI()
    }

    @objc private func decrementCount() {
        count -= 1
        updateUI()
    }

    private func updateUI() {
        countLabel.text = "\(count)"
        countLabel.textColor = count < 0 ? .red : .green
    }
}

선언적으로 코딩하는 방법

  • 위에서 알아본것처럼 선언형 프로그래밍은 코드가 마치 뷰처럼 보여야 하는것이 핵심
  • ex) padding을 하나 작성 할 때도 이런 관점이 매우 중요
    • 아래 뷰에서 padding을 30만큼 주고 싶은 경우?
struct ContentView: View {
    var body: some View {
        VStack(spacing: 0) {
            Text("첫 번째 text")
                .font(.largeTitle)
            
            Text("두 번째 text")
                .font(.largeTitle)
        }
    }
}

  • 아래처럼 두 번째 text에다 padding top을 줄 수 있지만, 코드를 읽을 때는 위에서 아래로 읽기 때문에, 가능하면 윗쪽에 bottom padding으로 작성할 것
// bad
struct ContentView: View {
    var body: some View {
        VStack(spacing: 0) {
            Text("첫 번째 text")
                .font(.largeTitle)
            
            Text("두 번째 text")
                .font(.largeTitle)
                .padding(.top, 30) // <-
        }
    }
}

// good
struct ContentView: View {
    var body: some View {
        VStack(spacing: 0) {
            Text("첫 번째 text")
                .font(.largeTitle) 
                .padding(.bottom, 30) // <-
            
            Text("두 번째 text")
                .font(.largeTitle)
        }
    }
}