如何明确设置 QChart 轴刻度?
Posted
技术标签:
【中文标题】如何明确设置 QChart 轴刻度?【英文标题】:How do I set QChart axis ticks explicitly? 【发布时间】:2017-03-24 17:31:44 【问题描述】:使用Zoom Line Example 我制作了一个Python QChartView
类,它可以使用箭头键滚动并使用加号和减号键进行缩放。 (见下面我的代码)。
当我向左滚动时,我希望网格线和轴刻度与数据滚动的量相同。但是,只有数据(QLineSeries)向左滚动。 5 条网格线保持在相同的位置,但它们的刻度值已更新。这是不可取的,因为新的刻度值可以是任何值。
我查看了文档,但找不到如何使网格与数据一起滚动。我错过了什么吗?
我还希望能够将刻度设置为显式值(这样我也许可以自己实现滚动行为)。是否可以将轴刻度值设置为特定值?
我的示例代码:
import sys
from math import pi, sin, sqrt
from PyQt5.QtChart import QLineSeries, QChart, QChartView
from PyQt5.QtCore import Qt
from PyQt5.QtWidgets import QApplication
class ZoomPanChartView(QChartView):
""" QChartView that can zoom/pan with the keys
"""
def __init__(self, chart):
super().__init__(chart)
self.zoomFactor = sqrt(2) # QCharts default is 2
self.panPixels = 10
def keyPressEvent(self, keyEvent):
""" Panning (scrolling) is done with the arrow keys.
Zooming goes with the plus and minus keys.
The '=' key resets.
"""
key = keyEvent.key()
if key == Qt.Key_Equal:
self.chart().zoomReset()
if key == Qt.Key_Plus:
self.chart().zoom(self.zoomFactor)
elif key == Qt.Key_Minus:
self.chart().zoom(1/self.zoomFactor)
elif key == Qt.Key_Left:
self.chart().scroll(-self.panPixels, 0)
elif key == Qt.Key_Right:
self.chart().scroll(+self.panPixels, 0)
elif key == Qt.Key_Up:
self.chart().scroll(0, +self.panPixels)
elif key == Qt.Key_Down:
self.chart().scroll(0, -self.panPixels)
elif key == Qt.Key_0:
self.chart().axisX().applyNiceNumbers() # changes the range
else:
super().keyPressEvent(keyEvent)
def main():
app = QApplication(sys.argv)
chart = QChart()
series = QLineSeries()
for i in range(0, 100):
x = i * pi / 20
y = sin(x)
series.append(x, y)
chart.addSeries(series)
chart.createDefaultAxes()
chart.axisY().setRange(-1, 1)
chart.legend().hide()
chartView = ZoomPanChartView(chart)
chartView.show()
chartView.resize(400, 300)
sys.exit(app.exec_())
if __name__ == "__main__":
main()
【问题讨论】:
我遇到了同样的问题。为了使网格易于阅读,我希望设置和固定网格步骤,独立于滚动和平移。所以,我也想自己设置网格刻度值。这样它会随着你的移动而移动。 QChart 的网格根本不是人类可读的。我会尝试一些事情,如果其中一个有效,我会在这里说。 【参考方案1】:您可以使用 QCategoryAxis 将刻度放置在您想要的位置:
初始化:
ch = self.chView.chart()
self.chartAxisX = QCategoryAxis(labelsPosition=QCategoryAxis.AxisLabelsPositionOnValue, startValue=0.0)
ch.setAxisX(self.chartAxisX)
self.chartAxisY = QCategoryAxis(labelsPosition=QCategoryAxis.AxisLabelsPositionOnValue, startValue=0.0)
ch.setAxisY(self.chartAxisY)
添加系列:
ch.addSeries(s)
s.attachAxis(self.chartAxisX)
s.attachAxis(self.chartAxisY)
将刻度设置为 5 的倍数:
for s in self.chartAxisX.categoriesLabels():
self.chartAxisX.remove(s)
for i in range(0, int(max_x_value) + 1, 5):
self.chartAxisX.append(str(i), i)
self.chartAxisX.setRange(0.0, max_x_value)
或者对任何区间使用这个通用函数:
def format_axis(axis, min_value, max_value, step):
for s in axis.categoriesLabels():
axis.remove(s)
axis.setStartValue(min_value)
for i in range(ceil(min_value / step), floor(max_value / step) + 1):
v = i * step
axis.append('%g' % v, v)
axis.setRange(min_value, max_value)
format_axis(self.chartAxisX, -1.1, 0.98, 0.25)
【讨论】:
感谢您的回答。与此同时,我决定使用PyQtGraph,但谁知道呢,我可能会在未来的项目中使用 QtCharts。【参考方案2】:我能找到的最好方法是将QValueAxis
设置为QChart
上的轴并调用QValueAxis::applyNiceNumbers()
来调整范围,即当前比例的最大值和最小值,以便显示的数字更人性化可读。但这会改变数据的位置而不是网格线的位置。您可以在 horizontalBarChart 示例中检查函数的行为。
我曾想过使用 QLineSeries 数据集自己制作网格,但我需要更改刻度在轴上的位置,据我所知,目前的 QChart 不容易制作。
简短的回答:你不能用 QCharts 做到这一点..
我使用Qwt 库已经有一段时间了,我可以证明那里的网格行为符合预期,其他行为也更加成熟。平移移动手柄,缩放使网格逐步调整大小以保持人类可读。也许值得检查。
【讨论】:
以上是关于如何明确设置 QChart 轴刻度?的主要内容,如果未能解决你的问题,请参考以下文章