UIViewRepresentable不会更新我的iOS图表数据集

时间:2020-03-24 08:26:01

标签: ios swift swiftui ios-charts uiviewrepresentable

我正在尝试将来自API的数据转换为折线图,但似乎无法正常工作。我将数据存储在一个可观察的对象中,因此需要花费几秒钟的时间,因此它不会显示在我的图形上,但是当我对数据进行硬编码时,可以肯定我正在获取数据,但根本不会出现。谢谢

struct HomeView: View {
    @State var tabIndex:Int = 0

    @ObservedObject var homeViewModel = HomeViewModel()
    init() {
        homeViewModel.getTimelineBy("US")
    }

    var body: some View {

        VStack(alignment: .center) {

            TimelineChartView(timelineDataSet: self.$homeViewModel.countryTimeline)

        }.frame(height: 500.0)


    }
}

struct TimelineChartView: UIViewRepresentable {


    @Binding var timelineDataSet: [ChartDataEntry]

    func updateUIView(_ uiView: LineChartView, context: UIViewRepresentableContext<TimelineChartView>) {

    }

    var lineChart = LineChartView()

    func makeUIView(context: UIViewRepresentableContext<TimelineChartView>) -> LineChartView {
        setUpChart()
        return lineChart
    }

    func setUpChart() {
        lineChart.noDataText = "No Data Available"
        lineChart.rightAxis.enabled = false
        lineChart.backgroundColor = .white
        let dataSets = [getLineChartDataSet()]
        let yAxis = lineChart.leftAxis
        yAxis.labelFont = .boldSystemFont(ofSize: 13)
        yAxis.setLabelCount(5, force: false)
        yAxis.labelTextColor = .black
        yAxis.axisLineColor = .black
        yAxis.labelPosition = .outsideChart
        lineChart.xAxis.labelPosition = .bottom
        lineChart.xAxis.labelFont = .boldSystemFont(ofSize: 13)
        lineChart.xAxis.labelTextColor = .black
        lineChart.xAxis.axisLineColor = .systemBlue
        lineChart.animate(xAxisDuration: 2.5)
        lineChart.notifyDataSetChanged()
        let data = LineChartData(dataSets: dataSets)
        data.setValueFont(.systemFont(ofSize: 7, weight: .black))

        lineChart.data = data
    }

    func getChartDataPoints(selectedTimelineData: [ChartDataEntry]) -> [ChartDataEntry] {
        var dataPoints: [ChartDataEntry] = []
        for eachTimeline in selectedTimelineData {
            let entry = ChartDataEntry(x: eachTimeline.x, y: eachTimeline.y)
            dataPoints.append(entry)
        }
        return dataPoints

    }



    func getLineChartDataSet() -> LineChartDataSet {
        let test = getChartDataPoints(selectedTimelineData: timelineDataSet)

        let set = LineChartDataSet(entries: test, label: "DataSet")
        set.lineWidth = 4
        set.drawCirclesEnabled = false
        set.mode = .cubicBezier
        set.fillAlpha = 0.9
        set.drawFilledEnabled = true
        set.highlightColor = .systemRed
        return set
    }

}

struct ContentView_Previews: PreviewProvider {
    static var previews: some View {
        HomeView()
    }
}

1 个答案:

答案 0 :(得分:1)

让我分享一些代码。更改绑定对象后,图表将显示更改的数据。

struct HomeView: View {
  @State var barData: [String: Double] = [String: Double]()
  var body: some View {
     NavigationView {
       List {            
         CustomeView([BarChartVM(arg: self.$barData, title: "BarChart")])
       }
     }
   }
}

struct CustomeView<Page:View>: View {
  var viewControllers: [UIHostingController<Page>]
  init(_ views: [Page]) {
    self.viewControllers = views.map { UIHostingController(rootView: $0) }
  }
  var body: some View {
      CustomeViewController(controllers: viewControllers)
  }
}
struct CustomeViewController: UIViewControllerRepresentable {
  var controllers: [UIViewController]
  func makeUIViewController(context: Context) -> UIPageViewController {
    let pageViewController = UIPageViewController(
        transitionStyle: .scroll,
        navigationOrientation: .horizontal)
    return pageViewController
  }
  func updateUIViewController(_ pageViewController: UIPageViewController, context: Context) {
    pageViewController.setViewControllers(
        [controllers[0]], direction: .forward, animated: true)
  }
}
struct BarChartVM: UIViewRepresentable {
  @Binding var arg: [String: Double]
  var title: String = ""
  let chart = BarChartView()

  func makeUIView(context: UIViewRepresentableContext<BarChartVM>) -> BarChartView {
    setUpChart()
    return chart
  }

  func updateUIView(_ uiView: BarChartView, context: UIViewRepresentableContext<BarChartVM>) {
    updateChartData()
  }

  func setUpChart() {
    chart.noDataText = "No data available"
    let pointArray = arg.sorted(by: <).map { $0.key }        
  }
  func updateChartData() {
    var entries = [BarChartDataEntry]()
    let valueArray = arg.sorted(by: <).map { $1 }
    for i in 0..<valueArray.count {
        let entry = BarChartDataEntry(x: Double(i), yValues: [valueArray[i]])
        entries.append( entry)
    }
    let set = BarChartDataSet(entries: entries, label: title)
    set.colors = ChartColorTemplates.material()      
    let data = BarChartData(dataSet: set)
    chart.data = data
  }
}