Skip to content
  • Categories
  • Recent
  • Tags
  • Popular
  • Users
  • Groups
  • Search
  • Get Qt Extensions
  • Unsolved
Collapse
Brand Logo
  1. Home
  2. Qt Development
  3. QML and Qt Quick
  4. QT charts extremely slow - QLineSeries
QtWS25 Last Chance

QT charts extremely slow - QLineSeries

Scheduled Pinned Locked Moved Solved QML and Qt Quick
qtchartqtchartsqlineseriespython
25 Posts 4 Posters 6.6k Views
  • Oldest to Newest
  • Newest to Oldest
  • Most Votes
Reply
  • Reply as topic
Log in to reply
This topic has been deleted. Only users with topic management privileges can see it.
  • W Witc

    @JonB When add console.log(serie.count()) it prints out again just some addres,
    when I type mySerie.count() in python (before emitting the signal) the output is 10 - so guess the X and Y data are presented in the serie
    129232c9-64ee-4379-8eb8-40a1efa5ea25-image.png

    JonBJ Offline
    JonBJ Offline
    JonB
    wrote on last edited by JonB
    #13

    @Witc said in QT charts extremely slow - QLineSeries:

    When add console.log(serie.count()) it prints out again just some addres,

    That does not sound good. I know your series is OK at the Python side, the question is how to get it at the QML side.

    I said I know nothing about QML. At this point I think you should read e.g. https://stackoverflow.com/questions/54459713/pass-c-lineseries-to-qml-charts

    You cannot pass QLineSeries to QML since ChartView has no possibility to add series. There is ChartView.createSeries() only. So the only way is to create a series in QML and pass array of points from C++ so you could add the points to the series using XYSeries.append(x,y).

    W 2 Replies Last reply
    0
    • JonBJ JonB

      @Witc said in QT charts extremely slow - QLineSeries:

      When add console.log(serie.count()) it prints out again just some addres,

      That does not sound good. I know your series is OK at the Python side, the question is how to get it at the QML side.

      I said I know nothing about QML. At this point I think you should read e.g. https://stackoverflow.com/questions/54459713/pass-c-lineseries-to-qml-charts

      You cannot pass QLineSeries to QML since ChartView has no possibility to add series. There is ChartView.createSeries() only. So the only way is to create a series in QML and pass array of points from C++ so you could add the points to the series using XYSeries.append(x,y).

      W Offline
      W Offline
      Witc
      wrote on last edited by
      #14

      @JonB BTW when I call myChart.addSeries(serie)in qml file - then the output is
      TypeError: Property 'addSeries' of object DeclarativeChart(0x17574328e80) is not a function

      1 Reply Last reply
      0
      • JonBJ JonB

        @Witc said in QT charts extremely slow - QLineSeries:

        When add console.log(serie.count()) it prints out again just some addres,

        That does not sound good. I know your series is OK at the Python side, the question is how to get it at the QML side.

        I said I know nothing about QML. At this point I think you should read e.g. https://stackoverflow.com/questions/54459713/pass-c-lineseries-to-qml-charts

        You cannot pass QLineSeries to QML since ChartView has no possibility to add series. There is ChartView.createSeries() only. So the only way is to create a series in QML and pass array of points from C++ so you could add the points to the series using XYSeries.append(x,y).

        W Offline
        W Offline
        Witc
        wrote on last edited by
        #15

        @JonB OK, so they say I should pass array of points from python to QML (It looks like we are getting to my first post here). I can pass X and Y point by point, but have no idea how to pass all my array or list

        x = [0,1,2,3,4,5,6,7,8,9]
        y = [10,20,10,5,23,15,12,2,9,21]
        
        JonBJ 1 Reply Last reply
        0
        • W Witc

          @JonB OK, so they say I should pass array of points from python to QML (It looks like we are getting to my first post here). I can pass X and Y point by point, but have no idea how to pass all my array or list

          x = [0,1,2,3,4,5,6,7,8,9]
          y = [10,20,10,5,23,15,12,2,9,21]
          
          JonBJ Offline
          JonBJ Offline
          JonB
          wrote on last edited by JonB
          #16

          @Witc
          My final word: I think QML ChartView/XYSeries etc. are not the same thing as their Qt QChartView/QXYSeries etc. counterparts. So when you said

          I found similar problem on the forum - they solved it (Creating the series first, and adding it to the Chart after all of the appends solved the problem)

          but they use C instead of python - Is it possible to bring the solution to python? I could not find a solution :-(

          I don't think it has anything to do with C++/Python, you picked a Qt widgets issue/solution which cannot be applied to QML....

          Which means I only see QML XYSeries.append(real x, real y) method to add one point at a time (I do not see a way to create a series on its own and then add it to a QML ChartView). Which you can call in a loop given the two arrays of points.

          If that is "too slow", I don't know what you can do. Maybe you can hide the chart view and re-show it after all points have been added, maybe that would make it faster.

          Otherwise have you read e.g. https://stackoverflow.com/questions/38467769/add-c-qabstractseries-to-qml-chartview ? Does that help for C++ -> QML?

          1 Reply Last reply
          0
          • B Offline
            B Offline
            Bob64
            wrote on last edited by
            #17

            I only have experience with QML charts and a C++ back end, but the approach I have taken is to create the series in QML and expose functions from C++ that accept a series so that it can be filled from the C++ side.

            I don't know how this translates to Python. In particular I do not know what the mechanism is for exposing Python-side functions into QML. I presume there is some sort of equivalent way of doing this (in C++ it is necessary to declare a function as Q_INVOKABLE in a QObject-derived class). Also I do not know how efficient this is going to be as there will be additional marshalling between Python and the underlying C++ layer.

            This might be one of those cases where it is necessary to bite the bullet and drop down to C/C++ to handle this from the Python.

            1 Reply Last reply
            0
            • fcarneyF Offline
              fcarneyF Offline
              fcarney
              wrote on last edited by
              #18

              I don't know how to do this in Python, but this is how I solve this issue with QML charts. I create a function that updates all the data in the series at once that is callable from QML:

              class PointUpdater : public QObject
              {
                  Q_OBJECT
              public:
                  PointUpdater(QObject* parent=nullptr)
                      : QObject(parent)
                  {
                  }
              
              public slots:
                  void update(QAbstractSeries *series, QVariantList points){
                      if(!series){
                          return;
                      }
                      QVector<QPointF> newpoints;
                      for(QVariant point: points){
                          auto list = point.toList();
                          newpoints.push_back(QPointF(list.at(0).toDouble(), list.at(1).toDouble()));
                      }
              
                      QXYSeries *xySeries = static_cast<QXYSeries *>(series);
                      xySeries->replace(newpoints);
                  }
              };
              

              C++ is a perfectly valid school of magic.

              W 1 Reply Last reply
              1
              • fcarneyF fcarney

                I don't know how to do this in Python, but this is how I solve this issue with QML charts. I create a function that updates all the data in the series at once that is callable from QML:

                class PointUpdater : public QObject
                {
                    Q_OBJECT
                public:
                    PointUpdater(QObject* parent=nullptr)
                        : QObject(parent)
                    {
                    }
                
                public slots:
                    void update(QAbstractSeries *series, QVariantList points){
                        if(!series){
                            return;
                        }
                        QVector<QPointF> newpoints;
                        for(QVariant point: points){
                            auto list = point.toList();
                            newpoints.push_back(QPointF(list.at(0).toDouble(), list.at(1).toDouble()));
                        }
                
                        QXYSeries *xySeries = static_cast<QXYSeries *>(series);
                        xySeries->replace(newpoints);
                    }
                };
                
                W Offline
                W Offline
                Witc
                wrote on last edited by Witc
                #19

                @fcarney looks nice, but also do not know how to conver it to the python - you are passing pointer of *series (which was created in QML)
                to C++ and then fill it - in python maybe impossible?!?

                1 Reply Last reply
                0
                • W Offline
                  W Offline
                  Witc
                  wrote on last edited by
                  #20

                  @JonB @fcarney @Bob64
                  Probably with my best solution below, it takes about 5 seconds to plot 5000 values

                  Python file - the slot fillSigChart is called from QML

                  @Slot(QLineSeries)
                      def fillSigChart(self,serie1):
                         
                          start = time.time()
                          for i in self.x:
                              serie1.append(i, self.y[i])   # filling serie with my prepared data
                          end = time.time()
                          print(end - start)
                  

                  In QML file: I call the slot above

                  Connections{ 
                    target: backend
                    function onSigPlotData(x,y){
                      myChart.removeAllSeries();
                      myChart.zoomReset()
                      var serie1 = myChart.createSeries(ChartView.SeriesTypeLine, "Signal ", axisX, axisY);
                      serie1.useOpenGl = true
                              
                      backend.fillSigChart(serie1)       // pass serie1 to python and then fill it with data
                     } 
                   }
                  
                  
                  JonBJ 1 Reply Last reply
                  0
                  • W Witc

                    @JonB @fcarney @Bob64
                    Probably with my best solution below, it takes about 5 seconds to plot 5000 values

                    Python file - the slot fillSigChart is called from QML

                    @Slot(QLineSeries)
                        def fillSigChart(self,serie1):
                           
                            start = time.time()
                            for i in self.x:
                                serie1.append(i, self.y[i])   # filling serie with my prepared data
                            end = time.time()
                            print(end - start)
                    

                    In QML file: I call the slot above

                    Connections{ 
                      target: backend
                      function onSigPlotData(x,y){
                        myChart.removeAllSeries();
                        myChart.zoomReset()
                        var serie1 = myChart.createSeries(ChartView.SeriesTypeLine, "Signal ", axisX, axisY);
                        serie1.useOpenGl = true
                                
                        backend.fillSigChart(serie1)       // pass serie1 to python and then fill it with data
                       } 
                     }
                    
                    
                    JonBJ Offline
                    JonBJ Offline
                    JonB
                    wrote on last edited by
                    #21

                    @Witc
                    OK, so at least we understand how to/that you can go from QML to Python and back passing a series around.

                    It might be that this is as good as it gets. However, you are still appending points one by one and this may be "expensive", we don't know.

                    There is an overload of append() which accepts a pre-built list of points: PySide2.QtCharts.QtCharts.QXYSeries.append(points). In C++ this is void QXYSeries::append(const QList<QPointF> &points).

                    You should try this. So instead of serie1.append(i, self.y[i]) you want to first build all the points into a list in the loop and then append them in one go afterwards. I would guess something like:

                    points = []
                    for i in self.x:
                        points.append(QPointF(i, self.y[i]))   # filling points with my prepared data
                    serie.append(points)  # append list of points in one call
                    

                    Does that make it any faster?

                    Even better, given that you start with no points you need to retain and just want the newly created points, might be serie.replace(points)

                    Note: This is much faster than replacing data points one by one, or first clearing all data, and then appending the new data. Emits QXYSeries::pointsReplaced() when the points have been replaced.

                    W 1 Reply Last reply
                    2
                    • B Offline
                      B Offline
                      Bob64
                      wrote on last edited by Bob64
                      #22

                      I concur with @JonB regarding appending multiple points at once, or replacing the data vs appending points one at a time. I found it made a massive difference.

                      One point at a time is slow in C++, but you will be paying a double penalty in Python as each individual call has to go through a wrapper layer. The Python overhead of the calls taking a list will mainly be the conversion of the Python list to the QList. I would hope that conversions like Python list to QList are optimised as much as they can be as they will be pervasive operations in a Python Qt application.

                      1 Reply Last reply
                      0
                      • JonBJ JonB

                        @Witc
                        OK, so at least we understand how to/that you can go from QML to Python and back passing a series around.

                        It might be that this is as good as it gets. However, you are still appending points one by one and this may be "expensive", we don't know.

                        There is an overload of append() which accepts a pre-built list of points: PySide2.QtCharts.QtCharts.QXYSeries.append(points). In C++ this is void QXYSeries::append(const QList<QPointF> &points).

                        You should try this. So instead of serie1.append(i, self.y[i]) you want to first build all the points into a list in the loop and then append them in one go afterwards. I would guess something like:

                        points = []
                        for i in self.x:
                            points.append(QPointF(i, self.y[i]))   # filling points with my prepared data
                        serie.append(points)  # append list of points in one call
                        

                        Does that make it any faster?

                        Even better, given that you start with no points you need to retain and just want the newly created points, might be serie.replace(points)

                        Note: This is much faster than replacing data points one by one, or first clearing all data, and then appending the new data. Emits QXYSeries::pointsReplaced() when the points have been replaced.

                        W Offline
                        W Offline
                        Witc
                        wrote on last edited by
                        #23

                        @JonB

                        Your idea to serie1.append(points) had no improvement - it took also about 5 seconds, but your second idea with serie1.replace(points) has great result!

                        Solution in python:

                        start = time.time()
                        points = []
                        for i in self.x:
                             points.append(QPointF(i, self.y[i]))   # filling points with my prepared data 
                        serie1.replace(points)  # fill list of points in one call
                        end = time.time()
                        print(end - start)
                        

                        for 50000 values it takes only 126 ms, for 5000 values it takes 13 ms

                        I think we can consider this thread as solved - or any other ideas?
                        Thank you all @JonB @Bob64 @fcarney for help!

                        JonBJ 1 Reply Last reply
                        2
                        • W Witc

                          @JonB

                          Your idea to serie1.append(points) had no improvement - it took also about 5 seconds, but your second idea with serie1.replace(points) has great result!

                          Solution in python:

                          start = time.time()
                          points = []
                          for i in self.x:
                               points.append(QPointF(i, self.y[i]))   # filling points with my prepared data 
                          serie1.replace(points)  # fill list of points in one call
                          end = time.time()
                          print(end - start)
                          

                          for 50000 values it takes only 126 ms, for 5000 values it takes 13 ms

                          I think we can consider this thread as solved - or any other ideas?
                          Thank you all @JonB @Bob64 @fcarney for help!

                          JonBJ Offline
                          JonBJ Offline
                          JonB
                          wrote on last edited by
                          #24

                          @Witc
                          :) I guess the append() maybe just appends one at a time, even if it's empty. The replace() presumably does not have to do that.

                          1 Reply Last reply
                          0
                          • fcarneyF Offline
                            fcarneyF Offline
                            fcarney
                            wrote on last edited by
                            #25

                            "append" causes a redraw. There is no way to disable the redraw that I can find. The QML api for charts is kinda crappy for adding data points. Which is why I had to use "replace" in C++. I have done thousands of points this way and it doesn't cause delays in the UI interface.

                            C++ is a perfectly valid school of magic.

                            1 Reply Last reply
                            1
                            • JonBJ JonB referenced this topic on
                            • JonBJ JonB referenced this topic on
                            • JonBJ JonB referenced this topic on

                            • Login

                            • Login or register to search.
                            • First post
                              Last post
                            0
                            • Categories
                            • Recent
                            • Tags
                            • Popular
                            • Users
                            • Groups
                            • Search
                            • Get Qt Extensions
                            • Unsolved