iOS 기본 (SwiftUI)

[iOS - SwiftUI] Path 사용 방법, Line Chart 그리는 방법 (주식 차트)

jake-kim 2022. 10. 19. 02:02

목차) SwiftUI의 기본 - 목차 링크

Path 사용 방법

https://developer.apple.com/documentation/swiftui/path

  • 2D 선과 면을 그릴때 사용
  • move(to:) 메소드를 통해 시작점을 정하고, addLine(to:)를 통해 선을 그림
struct ContentView: View {
  var body: some View {
    Path { path in
      path.move(to: CGPoint(x: 100, y: 100))
      path.addLine(to: CGPoint(x: 100, y: 200))
      path.addLine(to: CGPoint(x: 200, y: 200))
    }
  }
}

path로 그린 삼각혐

  • 사각형 그리기
struct ContentView: View {
  var body: some View {
    Path { path in
      path.move(to: CGPoint(x: 100, y: 100))
      path.addLine(to: CGPoint(x: 200, y: 100))
      path.addLine(to: CGPoint(x: 200, y: 200))
      path.addLine(to: CGPoint(x: 100, y: 200))
    }
  }
}

path로 그린 사각형

  • addArc로 반원 그리기
struct ContentView: View {
  var body: some View {
    Path { path in
      path.addArc(
        center: CGPoint(x: 100, y: 100),
        radius: 50,
        startAngle: Angle(degrees: 90),
        endAngle: Angle(degrees: -90),
        clockwise: true
      )
    }
  }
}

addArc로 그린 반원

  • addArc로 원 그리기
struct ContentView: View {
  var body: some View {
    Path { path in
      path.addArc(
        center: CGPoint(x: 100, y: 100),
        radius: 50,
        startAngle: Angle(degrees: 180),
        endAngle: Angle(degrees: -180),
        clockwise: true
      )
    }
  }
}

addArc로 그린 원

  • 이 밖에도 다양한 메소드가 존재
    • addArc(center:radius:startAngle:endAngle:clocwise:transform:) - 원호
    • addRect(:transform:) - 직사각형
    • addCurve(to:control1:control2:) - 3차 베지어 곡선
    • addQuadCurve(to:control:) - 2차 베지어 곡선
    • addEllipse(in:transform:) - 타원

Line Chart 그리는 방법 (주식 차트)

  • 데이터 준비
let datas = [205.87664248616824, 212.36591529463246, 258.8049099609528, 271.15419904110513, 273.86096203639727, 200.5959198527124, 217.96883356777028, 200.65573463890772, 261.3261775414218, 296.3419276530915, 242.74453036813378, 261.1769798545026, 221.55057031565565, 249.33001867026115, 235.89561987798092, 231.4278357622111, 257.1897839965552, 291.26324127070046, 234.27449372375384, 225.20342071447678, 256.0687554460256, 251.9884291086963, 271.4570906908704, 285.46843524640906, 221.93316123890222, 289.2200239699109, 270.0400095996903, 288.05903959368084, 265.5227107292555, 249.45236572757034, 228.50355096112676, 266.0659074095188, 219.7226062948684, 252.55327498248647, 289.5215302974416, 257.2631777132392, 241.1341168669185, 251.2296256239748, 247.38185750757185, 278.1093403927073, 216.7238654210636]
  • iOS의 CGPoint()의 (0,0)은 좌측 상단이기 때문에, y가 증가할수록 아래로 가고 작아질수록 위로 이동
    • 데이터를 그릴때도 y의 값이 클 수록 아래쪽으로 길어지게 만든 후, rotation을 통해 돌린 후 대칭이동을 통해 사용

ex) 사용하는쪽

  • Path 인스턴스를 얻고, rotation과 대칭이동을 사용
  var body: some View {
    getLinePath(values: datas) // 이 부분 아래에서 구현
      .stroke(.green ,style: StrokeStyle(lineWidth: 2, lineJoin: .round))
      .rotationEffect(.degrees(180), anchor: .center) // 시계방향으로 180도 회전
      .rotation3DEffect(.degrees(180), axis: (x: 0, y: 1, z: 0)) // y축 대칭
  }
  • getLinePath(values:) 구현
private func getLinePath(values: [Double]) -> Path {
  
}
  • 2개 이상일때만 Path 그리기
private func getLinePath(values: [Double]) -> Path {
  guard values.count > 1 else { return Path() }
}
  • offset 구하기
    • 중앙에 배치하기 위해서, max값과 min값을 이용
private func getLinePath(values: [Double]) -> Path {
  guard values.count > 1 else { return Path() }
  let offset = (values.max()! - values.min()!) / 2.0
  
}
  • 데이터들의 좌표를 알기 위해서, x좌표 y좌표에 사용할 step 프로퍼티 구하기
var totalWidth = UIScreen.main.bounds.width
var totalHeight = 200.0

private func getLinePath(values: [Double]) -> Path {
  guard values.count > 1 else { return Path() }
  let offset = (values.max()! - values.min()!) / 2.0
  
  let step = CGPoint(
    x: totalWidth / CGFloat(values.count - 1),
    y: totalHeight / CGFloat(values.max()! - values.min()!)
  )
}
  • path의 시작점을 구하고, 나머지 값들에 관한 값을 addLine(to:)을 통해 그리기
private func getLinePath(values: [Double]) -> Path {
  guard values.count > 1 else { return Path() }
  let offset = (values.max()! - values.min()!) / 2.0
  
  let step = CGPoint(
    x: totalWidth / CGFloat(values.count - 1),
    y: totalHeight / CGFloat(values.max()! - values.min()!)
  )
  var path = Path()
  let p1 = CGPoint(x: 0, y: step.y * (values[0] - offset))
  path.move(to: p1)
  
  for i in 1..<values.count {
    let p2 = CGPoint(x: step.x * CGFloat(i), y: step.y * (values[i] - offset))
    path.addLine(to: p2)
  }
  
  return path
}

완료)

* 전체 코드: https://github.com/JK0369/ExPath-SwiftUI

* 참고

https://medium.com/better-programming/create-a-line-chart-in-swiftui-using-paths-183d0ddd4578

https://developer.apple.com/documentation/swiftui/path