【问题标题】:constantly changing JSlider, event on user interaction?不断变化的JSlider,用户交互事件?
【发布时间】:2014-12-11 14:59:40
【问题描述】:

我目前正在使用 Java Swing 开发一个小型 MP3 播放器。我想实现一个允许用户跳到随机位置的进度条。我认为使用 JSlider 可能是最好的选择。

现在的问题是,我使用 Swing Timer 每 200m 更新进度条。这样,滑块的更改事件每次都会触发。有什么方法可以过滤掉不是由用户引起的事件调用?

我目前正在计时器中尝试使用 setValueIsAdjusting(true),但这似乎非常不一致...

Timer progressTimer = new Timer(200, e -> {
    updateProgress(player.getPosition());
});
progressTimer.start();

progressSlider.addChangeListener(new ChangeListener() {
    @Override
    public void stateChanged(ChangeEvent e) {
        if (progressSlider.getValueIsAdjusting()) {
            return;
        }
        System.out.println("Skipping to "+progressSlider.getValue());
        player.skipTo(progressSlider.getValue());
    }
});

System.out 有时会打印用户单击的位置,有时会打印滑块随计时器更新的位置。有什么帮助吗?

【问题讨论】:

  • 为什么你不能在 stateChanged 事件中更新进度条?
  • 嘿!您是否尝试过编写自己的模型类来扩展 BoundedRangeModel 并将其传递给 JSpinner 构造函数?你可以让模型从玩家本身获取信息,这样你就不再需要计时器了
  • 谢谢,塔巴基。我看了看,我没有找出你的确切意思。覆盖内部方法似乎对我没有帮助。

标签: java swing jslider


【解决方案1】:

如果滑块没有调整,我只通过计时器更新滑块来解决问题:

Timer progressTimer = new Timer(200, e -> {
    if (!progressSlider.getValueIsAdjusting()) {
        updateProgress(player.getPosition());
    }
});
progressTimer.start();

progressSlider.addChangeListener(new ChangeListener() {
    @Override
    public void stateChanged(ChangeEvent e) {
        if (progressSlider.getValueIsAdjusting()) {
            return;
        }
        System.out.println("Skipping to "+progressSlider.getValue());
        player.skipTo(progressSlider.getValue());
    }
});

【讨论】:

    【解决方案2】:

    我一直为 JSlider 解决这个问题的方法是通过私有布尔标志。就在我以编程方式设置 JSlider 值之前,我将标志设置为 true。在JSlider事件代码的开头,如果标志为真,我立即返回。

    将标志设置为 false 是棘手的部分,因为如果您在以编程方式设置 JSlider 值后立即执行此操作,则更改事件甚至还没有执行,所以当它实际执行时,标志将为 false 并且什么都没有会得到的。

    摘自SwingUtilities.invokeLater()的JavaDocs:

    “如果从事件调度线程调用 invokeLater [...] 仍将被推迟,直到处理完所有未决事件。”

    由于您已经在 AWT EDT 中,并且目标是让 JSlider 对更改做出反应(并立即返回)并 然后 将标志设置为 false,因此解决方案非常出色简单,从来没有让我失望过:

    ignoreEvents = true;
    jSlider.setValue(newValue);
    SwingUtilities.invokeLater(() -> ignoreEvents = false);
    


    编辑:

    如果情况真的很简单(您只想忽略一个事件。),您可以这样做:

    像上面一样,在调用 setValue 之前设置布尔标志。在 JSlider 事件代码中,如果设置了标志,清除标志并返回。这里不需要 SwingUtilities.invokeLater()。



    【讨论】:

    • 我不能再测试这个了,但它听起来很聪明而且很有前途。不过,我会将其标记为解决方案,谢谢!
    • 谢谢,我第一个接受的解决方案。 :D 顺便说一句,我没有在非 Windows 系统上测试过,但我引用的文档表明行为可靠。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2012-12-06
    • 1970-01-01
    • 2015-04-14
    • 2017-08-27
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多