我正在尝试实现一个类似于 Windows 中的网络使用图。我显示了最后 30 秒的数据,新值出现在右侧并向左侧移动。
除了无法让垂直网格线随数据移动之外,我已经完成了所有工作。相反,主要价格变动始终保持静态,只有图表上的线条在移动。
即在下图中,我预计主要刻度保持在刻度单位 5 处,并显示在 55 和 60 处。65 处的标签将离开屏幕。
- 我已禁用自动量程,并手动更新轴上的 lowerBound 和 upperBound 属性。
- 启用自动量程后,您将获得 snake线条向 y 轴移动几秒钟,然后跳跃一个主要刻度距离的样式效果,依此类推。
- 我还尝试在图表和轴上启用动画属性。
我正在寻找任何基于现有内容的解决方案/解决方法,而不必从头开始编写。是否可以通过子类化来自定义行为?
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
import javafx.application.Application;
import javafx.application.Platform;
import javafx.collections.ListChangeListener;
import javafx.collections.ObservableList;
import javafx.scene.Scene;
import javafx.scene.chart.LineChart;
import javafx.scene.chart.NumberAxis;
import javafx.scene.chart.XYChart;
import javafx.scene.chart.XYChart.Data;
import javafx.scene.layout.BorderPane;
import javafx.stage.Stage;
public class StackOverflow25383566 extends Application {
@Override
public void start(Stage stage) throws Exception {
final NumberAxis xAxis = new NumberAxis();
xAxis.setAutoRanging(false);
xAxis.setForceZeroInRange(false);
xAxis.setTickUnit(5);
final NumberAxis yAxis = new NumberAxis(0, 15, 1);
yAxis.setAutoRanging(false);
final LineChart<Number, Number> chart = new LineChart<Number, Number>(
xAxis, yAxis);
chart.setAnimated(false);
chart.setCreateSymbols(false);
chart.setLegendVisible(false);
final XYChart.Series<Number, Number> series = new XYChart.Series<Number, Number>();
chart.getData().add(series);
series.getData().addListener(
new ListChangeListener<Data<Number, Number>>() {
@Override
public void onChanged(
ListChangeListener.Change<? extends Data<Number, Number>> arg0) {
ObservableList<Data<Number, Number>> data = series
.getData();
xAxis.setLowerBound(data.get(0).getXValue()
.doubleValue());
xAxis.setUpperBound(data.get(data.size() - 1)
.getXValue().doubleValue());
}
});
stage.setScene(new Scene(new BorderPane(chart), 800, 600));
stage.show();
final Runnable update = new Runnable() {
private int clock;
@Override
public void run() {
try {
ObservableList<Data<Number, Number>> data = series
.getData();
if (data.size() > 10) {
data.remove(0);
}
data.add(new XYChart.Data<Number, Number>(clock, clock % 13));
clock++;
} catch (Throwable e) {
// executors silently swallow exceptions
e.printStackTrace();
throw e;
}
}
};
Executors.newSingleThreadScheduledExecutor().scheduleAtFixedRate(
new Runnable() {
@Override
public void run() {
Platform.runLater(update);
}
}, 0, 500, TimeUnit.MILLISECONDS);
}
public static void main(String[] args) {
launch(args);
}
}
最佳答案
我能够通过子类化 ValueAxis 来解决这个问题,以便它返回始终是 tickUnit 倍数的刻度标记。
- 自从开发此解决方案以来,我已经意识到 jfxutils其中包含一个名为 StableTickAxis 的类,但是其 JavaDoc 声明“尚未准备好使用”
这张图片显示了我所追求的,即不一定以大刻度开始。
private class SpecialAxis extends ValueAxis<Number> {
private SimpleObjectProperty<Double> tickUnitProperty = new SimpleObjectProperty<Double>(
5d);
@Override
protected List<Number> calculateMinorTickMarks() {
List<Number> ticks = new ArrayList<Number>();
double tickUnit = tickUnitProperty.get() / getMinorTickCount();
double start = Math.floor(getLowerBound() / tickUnit) * tickUnit;
for (double value = start; value < getUpperBound(); value += tickUnit) {
ticks.add(value);
}
return ticks;
}
@Override
protected List<Number> calculateTickValues(double arg0, Object arg1) {
List<Number> ticks = new ArrayList<Number>();
double tickUnit = tickUnitProperty.get();
double start = Math.floor(getLowerBound() / tickUnit) * tickUnit;
for (double value = start; value < getUpperBound(); value += tickUnit) {
ticks.add(value);
}
return ticks;
}
@Override
protected Object getRange() {
// not sure how this is used??
return null;
}
@Override
protected String getTickMarkLabel(Number label) {
return label.toString();
}
@Override
protected void setRange(Object range, boolean arg1) {
// not sure how this is used??
}
public SimpleObjectProperty<Double> getTickUnitProperty() {
return tickUnitProperty;
}
public void setTickUnit(double tickUnit) {
tickUnitProperty.set(tickUnit);
}
}
关于javafx - 使用 JavaFX 滚动 XYChart,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25383566/