I have implemented several xamdatacharts throughout my app. A new requirement to run my app on a tablet was given to me and basically the xamdatachart make this nearly impossible. I have 2 issues that have plagued me since I wrote the app in v11.1 then 12.2.
I have recently upgraded to 14.1 and have not seen any performance boost in the XamDataChart.
1) I have a chart using a LineSeries and a CategoryXAxis. The source is an observable collection of (DateTime, float). I attempt to simulate an EEG graph by updating that collection by index (so no FIFO). Collection is 4000 points for 8 seconds worth of data. I update the collection every 50ms so I am updating ~25 points of data plus 250 blank points to simulate a moving gap about 20 times a second. I have even used an custom implementation of an ObservableCollection that turns off notifications when I start the update and then sends a reset after all 275 points are updated vs just updating as they come in. The cpu was no different. Point is that this graph is the biggest cpu hog and on the tablet causes the UI to eventually become unresponsive. I am hoping that someone can tell me if there is a more efficient way to create an EEG simulated graph using any of the Infragistics controls?
2) I have another chart that can have N LineSeries but in this case they all use the CategoryDateTimeXAxis. Each series will have points coming in at no faster than 1/second. However, this graph show all the history. What I find is after several hours of running each point added takes longer and longer. I believe this is because the CategoryDateTimeXAxis sorts the data on every insert even though I know it is already sorted. Is there anything in the new version that I can do to optimize that xaxis type or a new way to use it to make this more efficient? I have had to limit each series to only hold 28800 (8 hours) points as beyond that the entire UI gets noticeably choppy each second. I have posted this question before about 18 months ago as a support ticket and the suggestion was to use a scatterline series instead but this had a similar jittery effect, slightly reduced, but happened all the time even when the series only had a few points in it (plus the implementation was much uglier).
Any new suggestions to optimize either of these scenarios?
Hi Mike,
Most of your performance for #1 is probably going to measure and arrange calls for the geometry needed to render a 4000 point LineSeries. Even if you only update a small subset of the data points, that update is going to trigger the entire series to be re-rendered. I'm looking into this specific requirement to see if there is a better way to do it.
For #2, is a CategoryDateTimeXAxis necessary here? If your data points have a fixed interval between them then a CategoryXAxis might be a better choice. It doesn't presort the data before rendering like the CategoryDateTimeXAxis so it should perform better. I'll take a look at your previous support tickets to see if I can find the one dealing with this.
I will get back to you shortly with more information.
Thanks Rob.
I kept using the CategoryDateTimeXAxis because I cannot guarantee all the points will be equally spaced. It just made my life a little easier if it took care of the spacing for me. I just wish I could tell it to not sort as an optional optimization.
I look forward to anything I can do to optimize the EEG chart as well. I have a C++ guy I work with that is giving me a lot of flack about .NET graphing performance... :)
Hi Valerie,
I just verified that the collection objects do not implement INotifyPropertyChanged so they cannot be firing off any events. Also, I am replacing the collection element, not updating a property of the object, so it shouldn't matter if they did implement that interface right?
I have actually been working on this chart performance for some time. I have implemented a few other chart controls to see if I can get better performance and so far only one has met my expectations (LightningChart). It uses about half the CPU to draw the same graph as Infragistics but as I have started using other Infragistics controls I don't want to include another 3rd party control into our product. I am still hoping we can find a more efficient way to create a sweeping EEG that doesn't kill a tablet.
Thanks,
Mike
I was under the impression that you were just updating the data points by updating their values rather than replacing them, based on your initial description of your implementation. Since you are just replacing the items, INotifyPropertyChanged will have no bearing on this.
While I don't have an EEG example, our engineers have given me code for an EKG and I imagine that the update strategy would be similar so I'm throwing a sample together. Out of curiosity, what kind of tablet are you using? And just to make sure, as I have seen other posts by you on the Silverlight forums, this is for WPF correct? I should have asked this initially.
Sorry for the confusion.
This is the code I use to update the collection:
waveForm.WaveData.SuspendCollectionChangeNotification(); //_logger.Info(String.Format("Enter OC lock - UpdateWaveFormCollections - {0}", waveForm.WaveChannelName)); // add all the points from the buffer to the data list that are older than the updated LastDisplayedTimeStamp WaveFormDataPoint data; while (waveForm.WaveDataBuffer.Count > 0 && waveForm.WaveDataBuffer.TryPeek(out data) && data != null && data.TimeStamp < waveForm.LastDisplayedTimeStamp) { if (waveForm.CurrentPosition >= collectionSize) { waveForm.CurrentPosition = 0; } WaveFormDataPoint queueitem; if (waveForm.WaveDataBuffer.TryDequeue(out queueitem)) { waveForm.WaveData[waveForm.CurrentPosition] = queueitem; waveForm.CurrentPosition++; } }
// Add the moving space if (waveForm.WaveDataBuffer.Count > 0) { int spacesize = (int)(0.5 / waveForm.SecondsBetweenPoints); // half second. quan of samples to use for moving space int last = 0; for (int p = 0; p < spacesize; p++) { var emptywavedata = new WaveFormDataPoint { TimeStamp = waveForm.LastDisplayedTimeStamp.AddSeconds(waveForm.SecondsBetweenPoints * p), Value = float.NaN }; if (waveForm.CurrentPosition + p >= collectionSize) { waveForm.WaveData[p - last] = emptywavedata; } else { waveForm.WaveData[waveForm.CurrentPosition + p] = emptywavedata; last = p; } }
// If the control is not visible the do not notify // the observable collection of changes if (IsVisible) { waveForm.WaveData.NotifyChanges(); }
EKG is perfect as well. Just need a really efficient method to plot several graphs at 500Hz.
This is a crappy (not all that bad just annoyed with it right now) little Windows tablet using an Atom N2600 @ 1.6GHz
I put together a sample that used our developers data and it turns out this looks more like an EEG than an EKG. Looking at your update code though, I believe we're doing it similarly, except for the while loop and the wave data buffer part. Although in my case I'm not pulling datapoints from a queue to override the data source, but I'm instantiating new data points entirely. It shouldn't matter though, yours may be more memory efficient that way.
One thing I notice in our developers sample though, is that the chart looks like a normal EEG without needing to use 4000 data points. Now I don't know what your EEG looks like with 4000 data points but when I compare my sample with pictures I see on the internet, it looks pretty close. I attached the sample so you can take a look. If I bump up the amount of data points to 4000, I do notice a jump in my cpu usage. Task Manager isn't really the best tool to use for this but on my laptop it was staying around 13% with 4000 data points. I then tried it out on a Surface Pro tablet (i5 1.7GHz so a bit more powerful than yours) and Task Manager showed between 30 and 40% cpu usage. From these tests I definitely can see how it could be worse on your tablet.
My conclusion from all of this is that I don't think there is anything wrong with the way you are updating the data. I think it just comes down to the chart itself trying to render 4000 constantly changing data points on a tablet. For the tablet, you may need to try playing with the Resolution property on the LineSeries to try and cut down on the number of geometry that is rendered. Or you'll have to see if you can get away with lowering the data point count so the chart doesn't have as much to render.
Rob,
I was under the impression that the XamDataChart would generate a "best fit" line graph given a data set so I fed all the data I get to the chart which is roughly 500 points per second. Regardless of how many points I add I still have to "update" the chart around every 50ms in order to attain a smooth scrolling affect so whether I update 275 points (25 of data + 250 for a "moving gap") or 55 points (5 of data and 50 for moving gap after some data processing) I didn't thinnk it would make much difference. I will try this and see if it makes helps with the cpu.
empty
Hello Mike,
Thanks for clarifying. The information in this thread should prove useful to the community.
Let me know if you have any further questions.
We have a tablet with an Intel N2600 Atom processor so it is very low power. I believe it has GMA 3600 graphics chip.
Interesting. I'm curious as to what kind of graphics processor the tablet has. I haven't really seen cases where software rendering is actually better than hardware rendering but if the graphics processor is just that bad then I guess it could give you a boost. I never even thought about considering that. Nice find!
I believe I understand. Thanks.
On my desktop I do see acceptable performance. I have been working with the resolution and decreasing the number of points I plot but I do not see significant gains in performance vs. the loss of accuracy in the graph. I did find out that if I force software rendering mode I get a nice performance boost on the tablet so this may make the app acceptable.
Thanks for the help.