多个 JSlider 相互反应始终等于 100%

2024-04-11

我正在尝试向 java swing 应用程序添加 3 个 JSlider,以便三个滑块的总价值总和为 100。每个滑块都是一个概率,滑块 A 是将值添加到队列的概率,滑块 B是从队列中删除某个值的概率,滑块 C 是什么都不发生的概率。

示例:滑块 A 设置为 40,滑块 B 设置为 40,滑块 C 初始设置为 20。 如果用户将滑块 A 更改为 50,我希望滑块 B 和 C 的值都减少 5。这样总数仍为 100。 我知道当用户将值更改为 41 时可能会出现一些问题,因为我希望其他值保留为整数。

好的,在查看有关 BoundedRangeModel 的一些信息后,我添加了一些代码。我已经设法让滑块根据其他滑块进行更改,但我仍然不确定如何实现初始值。 此外,滑块似乎只能成对工作,而不是 3 个一组。

我将继续努力,如果我想出一个令我满意的解决方案,我会将其发布在这里。

package stackoverflow;
import java.awt.GridBagConstraints;
import java.awt.GridBagLayout;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;

import javax.swing.BoundedRangeModel;
import javax.swing.DefaultBoundedRangeModel;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JSlider;
import javax.swing.SwingUtilities;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;

public class Controller {

public static void main(String[] args)
{
    //GUI up
    Frame f = new Frame();
    f.setSize(600, 600);
    f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    f.setVisible(true);
}

@SuppressWarnings("serial")
public static class Frame extends JFrame implements PropertyChangeListener, ChangeListener
{
    //Sliders
    private JSlider s1;
    private JSlider s2;
    private JSlider s3;

    //Initial Values for fields (not sure how to implement)
    private int inis1 = 45;
    private int inis2 = 40;
    private int inis3 = 15;

    //Labels
    private JLabel s1Label;
    private JLabel s2Label;
    private JLabel s3Label;

    //Strings for labels
    private static String s1String = "Probability of adding one to queue";
    private static String s2String = "Probability of subtracting one from queue";
    private static String s3String = "Probability of doing nothing";

    public Frame()
    {
        //title of window
        super("Sliders");

        //Layout
        //Label Panel, GridBag Setup
        JPanel pane = new JPanel(new GridBagLayout());
        this.add(pane);
        GridBagConstraints c = new GridBagConstraints();

        class LimitedBoundedRangeModel extends DefaultBoundedRangeModel {
            BoundedRangeModel limit;

            public LimitedBoundedRangeModel(BoundedRangeModel limit) {
                this.limit = limit;
            }

            /** 
             * @inherited <p>
             */
            @Override
            public void setRangeProperties(int newValue, int newExtent, int newMin,
                    int newMax, boolean adjusting) {
                if (limit != null) {
                    int combined = newValue + limit.getValue();
                    if (combined > newMax) {
                        newValue = newMax - limit.getValue();
                    }
                }
                boolean invoke =
                        (adjusting != getValueIsAdjusting()) && !adjusting;
                 super.setRangeProperties(newValue, newExtent, newMin, newMax, adjusting);
                 if (invoke) {
                     SwingUtilities.invokeLater(new Runnable() {
                         public void run() {
                             fireStateChanged();   
                         }
                     });
                 }
                super.setRangeProperties(newValue, newExtent, newMin, newMax, adjusting);
            }
        }

        // use
        LimitedBoundedRangeModel firstModel = new LimitedBoundedRangeModel(null);
        LimitedBoundedRangeModel secondModel = new LimitedBoundedRangeModel(firstModel);
        LimitedBoundedRangeModel thirdModel = new LimitedBoundedRangeModel(secondModel);

        firstModel.limit = secondModel;
        secondModel.limit = thirdModel;
        thirdModel.limit = firstModel;

        s1 = new JSlider(firstModel);
        c.gridx = 0;
        c.gridy = 2;
        c.gridwidth = 5;
        pane.add(s1, c);

        s2 = new JSlider(secondModel);
        c.gridx = 0;
        c.gridy = 5;
        c.gridwidth = 5;
        pane.add(s2, c);

        s3 = new JSlider(thirdModel);
        c.gridx = 0;
        c.gridy = 7;
        c.gridwidth = 5;
        pane.add(s3, c);

        s1Label = new JLabel(s1String);
        c.gridx = 0;
        c.gridy = 2;
        c.gridwidth = 5;
        pane.add(s1Label, c);

        s2Label = new JLabel(s2String);
        c.gridx = 0;
        c.gridy = 4;
        c.gridwidth = 5;
        pane.add(s2Label, c);

        s3Label = new JLabel(s3String);
        c.gridx = 0;
        c.gridy = 6;
        c.gridwidth = 5;
        pane.add(s3Label, c);

        //enable 'tick' markers
        s1.setMajorTickSpacing(25);
        s1.setMinorTickSpacing(1);
        s1.setPaintTicks(true);
        s1.setPaintLabels(true);
        s1.setSnapToTicks(true);
        s2.setMajorTickSpacing(25);
        s2.setMinorTickSpacing(1);
        s2.setPaintTicks(true);
        s2.setPaintLabels(true);
        s2.setSnapToTicks(true);
        s3.setMajorTickSpacing(25);
        s3.setMinorTickSpacing(1);
        s3.setPaintTicks(true);
        s3.setPaintLabels(true);
        s3.setSnapToTicks(true);

        //Listening to sliders

        s1.addChangeListener(new ChangeListener()
        {
        public void stateChanged(ChangeEvent ce)
        {
            JSlider source = (JSlider)ce.getSource();
            if (!source.getValueIsAdjusting())
            {
                int P = (int)source.getValue();
                System.out.println("Probability of adding to queue is now: " + P);
                /*
                 * TODO
                 * Would setting the values of the other sliders here,
                 * instead of using the BoundedRangeModel work?
                 * Tricky math would have to be used.
                 */
               // s2Slider.setValue();

            }
           }
        });

        s2.addChangeListener(new ChangeListener()
        {
            public void stateChanged(ChangeEvent ce)
            {
                JSlider source = (JSlider)ce.getSource();
                if (!source.getValueIsAdjusting())
                {
                    int P = (int)source.getValue();
                    System.out.println("Probability of subtracting from queue is now: " + P);

                }
            }
        });
        s3.addChangeListener(new ChangeListener()
        {
            public void stateChanged(ChangeEvent ce)
            {
                JSlider source = (JSlider)ce.getSource();
                if (!source.getValueIsAdjusting())
                {
                    int P = (int)source.getValue();
                    System.out.println("Probability of doing nothing to the queue is now: " + P);
                }
            }
        });
    }   

    @Override
    public void propertyChange(PropertyChangeEvent evt) {
        // TODO Auto-generated method stub

    }

    @Override
    public void stateChanged(ChangeEvent e) {
        // TODO Auto-generated method stub
    }
}

}

一种方法,也许对某人有用:

package stackoverflow;

import java.awt.GridLayout;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.Map;

import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JSlider;
import javax.swing.SwingUtilities;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;

public class ConnectedSliders
{
    public static void main(String[] args)
    {
        SwingUtilities.invokeLater(new Runnable()
        {
            @Override
            public void run()
            {
                createAndShowGUI();
            }
        });
    }

    private static void createAndShowGUI()
    {
        JSlider s0 = new JSlider(0, 100, 30);
        JSlider s1 = new JSlider(0, 100, 40);
        JSlider s2 = new JSlider(0, 100, 30);

        SliderGroup sliderGroup = new SliderGroup();
        sliderGroup.add(s0);
        sliderGroup.add(s1);
        sliderGroup.add(s2);

        JPanel panel =new JPanel(new GridLayout(0,2));
        panel.add(s0);
        panel.add(createListeningLabel(s0));
        panel.add(s1);
        panel.add(createListeningLabel(s1));
        panel.add(s2);
        panel.add(createListeningLabel(s2));

        panel.add(createListeningLabel(s0, s1, s2));

        JFrame f = new JFrame();
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        f.getContentPane().add(panel);
        f.pack();
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }

    private static JLabel createListeningLabel(final JSlider ... sliders)
    {
        final JLabel label = new JLabel("");
        for (JSlider slider : sliders)
        {
            slider.addChangeListener(new ChangeListener()
            {
                @Override
                public void stateChanged(ChangeEvent e)
                {
                    int sum = 0;
                    for (JSlider slider : sliders)
                    {
                        sum += slider.getValue();
                    }
                    label.setText("Sum: "+sum);
                }
            });
        }
        return label;
    }

    private static JLabel createListeningLabel(final JSlider slider)
    {
        final JLabel label = new JLabel("");
        slider.addChangeListener(new ChangeListener()
        {
            @Override
            public void stateChanged(ChangeEvent e)
            {
                label.setText(String.valueOf(slider.getValue()));
            }
        });
        return label;
    }


}


class SliderGroup
{
    private final Map<JSlider, Integer> values;
    private final LinkedList<JSlider> candidates;

    private final ChangeListener changeListener;
    private boolean updating = false;

    SliderGroup()
    {
        this.values = new HashMap<JSlider, Integer>();
        this.candidates = new LinkedList<JSlider>();

        changeListener = new ChangeListener()
        {
            @Override
            public void stateChanged(ChangeEvent e)
            {
                JSlider source = (JSlider)e.getSource();
                update(source);
            }
        };
    }

    private void update(JSlider source)
    {
        if (updating)
        {
            return;
        }
        updating = true;

        int delta = source.getValue() - values.get(source);
        if (delta > 0)
        {
            distributeRemove(delta, source);
        }
        else
        {
            distributeAdd(delta, source);
        }


        for (JSlider slider : candidates)
        {
            values.put(slider, slider.getValue());
        }

        updating = false;
    }


    private void distributeRemove(int delta, JSlider source)
    {
        int counter = 0;
        int remaining = delta;
        while (remaining > 0)
        {
            JSlider slider = candidates.removeFirst();
            counter++;

            if (slider == source)
            {
                candidates.addLast(slider);
            }
            else
            {
                if (slider.getValue() > 0)
                {
                    slider.setValue(slider.getValue()-1);
                    remaining--;
                    counter = 0;
                }
                candidates.addLast(slider);
                if (remaining == 0)
                {
                    break;
                }
            }
            if (counter > candidates.size())
            {
                String message = "Can not distribute "+delta+" among "+candidates;
                //System.out.println(message);
                //return;
                throw new IllegalArgumentException(message);
            }
        }
    }

    private void distributeAdd(int delta, JSlider source)
    {
        int counter = 0;
        int remaining = -delta;
        while (remaining > 0)
        {
            JSlider slider = candidates.removeLast();
            counter++;

            if (slider == source)
            {
                candidates.addFirst(slider);
            }
            else
            {
                if (slider.getValue() < slider.getMaximum())
                {
                    slider.setValue(slider.getValue()+1);
                    remaining--;
                    counter = 0;
                }
                candidates.addFirst(slider);
                if (remaining == 0)
                {
                    break;
                }
            }
            if (counter > candidates.size())
            {
                String message = "Can not distribute "+delta+" among "+candidates;
                //System.out.println(message);
                //return;
                throw new IllegalArgumentException(message);
            }
        }
    }


    void add(JSlider slider)
    {
        candidates.add(slider);
        values.put(slider, slider.getValue());
        slider.addChangeListener(changeListener);
    }

    void remove(JSlider slider)
    {
        candidates.remove(slider);
        values.remove(slider);
        slider.removeChangeListener(changeListener);
    }

}

核心是SliderGroup类(类似于ButtonGroup): 可以添加几个 JSlider。一个滑块的更改将分布在其他滑块中。正如评论中提到的,这种分布有点棘手:当一个人反复将滑块 A 的值增加 1 时(例如,通过按右箭头光标键),则其他滑块的值必须反复减少 - 但是它不应该总是选择相同的滑块来接收此更改,否则只有一个滑块会移动。我使用接收更改的“候选”滑块的 LinkedList 解决了这个问题。当其他滑块的值必须减小时,更改会从左到右分布在这些候选滑块之间。收到更改的滑块被放置在列表的末尾,以便当它们的值必须再次减少时,收到先前减少的滑块将是候选列表中的最后一个。 (当它们的值必须增加时,也会做同样的事情,反之亦然)。似乎工作良好,但没有经过彻底测试:滑块将被强制具有“无效”值的情况下的行为(例如,当 A 和 B 具有其 MIN 值,但 C 没有其 MAX 值时,并且增加)可能取决于应用案例。目前,在这种情况下会引发异常,但简单地“忽略”这种情况也可能是合适的......

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

多个 JSlider 相互反应始终等于 100% 的相关文章

随机推荐

  • Windows Phone 7 和 Windows Phone 8 支持什么框架?

    Windows Phone 7 和 Windows Phone 8 支持什么框架 我在网上找不到太多关于此的信息 但我听说WP7不支持完整的框架 如果是的话 WP7 和 WP8 的框架有哪些限制 奖金问题 WP7和WP8的编程仅限于C 吗
  • GUI 读取 JTextField 时遇到问题

    我不知道我哪里出了问题 我尝试改变一些东西 但我就是无法让 CalculateButtonHandler 正常工作 抱歉 我列出了所有这些代码 但上次我没有具体说明足够了 S 如果有人能指出我正确的方向 那就太好了 谢谢 import ja
  • 是否应该将扩展属性添加到 C# 4.0 中? [关闭]

    就目前情况而言 这个问题不太适合我们的问答形式 我们希望答案得到事实 参考资料或专业知识的支持 但这个问题可能会引发辩论 争论 民意调查或扩展讨论 如果您觉得这个问题可以改进并可能重新开放 访问帮助中心 help reopen questi
  • QML - 不支持命令式代码

    有人可以评论这个事实吗QML 任何命令式JavaScript代码不会被执行 除非它是额外组件 http blog qt digia com blog 2011 05 05 qt quick designer in qt creator 2
  • 如何访问 UWP 应用中的注册表项?

    我想通过 UWP 应用程序访问 windows10 注册表项 键为 HKEY LOCAL MACHINE SOFTWARE MyCompanyName MyName 我找不到任何函数调用来完成它 请指示我该怎么做 谢谢你 如果您的应用程序是
  • IntPoint 没有索引整数值

    当我们尝试使用字段类型 IntPoint 对整数值进行索引时 这些值似乎没有正确传输到我们的 Lucene 索引中 我们正在使用 Lucene 6 0 根据 Lucene 文档的代码片段 doc add new IntPoint LENGT
  • 在 SwiftUI 2.0 中禁用选项卡视图滑动以更改页面

    我正在使用一个tab view在我的 SwiftUI 应用程序中 我想要disable its swipe向左写信至move到其他页面 我检查了这个答案 https stackoverflow com questions 65524458
  • Sencha Touch、OpenLayers、GeoServer:使用 Android 2.2 的设备上的编码错误

    我使用创建了一个小测试页面煎茶触摸 开放层我正在从一个接收 WMS WFS 数据地理服务器 我已将 HTML 页面上的编码设置为 UTF 8 我使用以下语句从 GeoServer 中提取一些 WFS 数据 var post new Open
  • 如何在emu 8086中用汇编语言打印0到100?

    这里我尝试了打印 10 到 0 的十进制数字emu8086 MODEL SMALL STACK 100H DATA NUM DB 58D CODE MAIN PROC MOV AX DATA MOV DS AX START CMP NUM
  • ASP.NET MVC 中需要两个不同版本的 Newtonsoft.Json.dll

    我开发了一个 MVC 应用程序 它依赖于使用 Newtonsoft Json dll v6 0 0 0 的 Connectwise SDK 和使用 Newtonsoft Json dll v7 0 0 0 的 Dropbox SDK 我需要
  • cell.imageView 可以在模拟器上运行,但不能在设备上运行

    表视图单元格 imageView 在模拟器上工作 但在设备上不起作用 有些事情我已经检查过 我没有更改代码中的任何内容 图像已添加到项目中并位于同一文件夹中 我还有一张工作正常的细胞图像 先感谢您 Iphone 设备区分大小写 而模拟器则不
  • case 语句中的 count 函数

    问题陈述 如果讲师教授的课程数量少于1 则将薪水更新为30000 否则更新为35000 使用一个命令 我编写的代码导致错误 请您告诉我为什么它不起作用以及我如何改进它 先谢谢了 更新 使用Oracle SQL schema 讲师 gt id
  • 发布到 Nexus 上的私有 NPM 存储库时出现身份验证错误

    我在发布到托管在我的私人 Nexus 上的私人 npm 注册表时遇到身份验证问题 我的 Nexus 设置是npm 代理 npm 注册表 托管 npmallowRepublish false npm 快照 托管 npmallowRepubli
  • Matplotlib Axes3D.quiver 图中箭头的长度不同

    有什么方法可以改变箭袋图中各个箭头的长度吗 我使用以下方法创建了一个绘图 lines to draw list of numpy 3d vectors xs ys zs list of coordinates us vs ws list o
  • 即使应用程序处于非活动状态,CursorLoader 如何自动更新视图?

    我一直在开发一个小型待办事项列表应用程序 我使用 CursorLoader 从内容提供商更新 ToDolistview 我写了一个函数onNewItemAdded 当用户在文本视图中输入新项目并单击 Enter 时调用 参考如下 publi
  • 在同一 pandas 数据框中交换两行(连同索引)

    我正在制定固定转售价格dataset https data gov sg dataset resale flat prices 如果您有兴趣 我使用的是 2015 年 1 月以后的数据 First I group the data by u
  • 单击警报消息中的“确定”后如何关闭浏览器中的当前窗口?

    基于页面中的某些操作 我想向用户发出警报消息 即 您的简历已上传 当用户在该警报框中单击 确定 时 我想关闭该窗口 我想仅使用警报方法而不是 JavaScript 中的确认方法来执行此操作 为什么 因为警报方法给出了唯一的选项 确定 而确认
  • MOVE-TO 期望输入是代理,但得到的是 NOBODY

    我的代码所做的是设置一个内部灰色补丁区域和一个外部黑色补丁区域 海龟可以在其中繁殖 每个补丁上有一个 一旦乌龟到达灰色和黑色区域之间的边界 我就会分配可变能量 以将乌龟的繁殖延迟一定的刻度 每个刻度能量增长一个单位 当能量达到一定数量时 我
  • 无法编译使用 boost 中的 odeint 的 C++

    我使用的是 Ubuntu 12 04 并且 usr include 中已经有一些 boost fies 我做了一个 sudo apt get install libboost all dev 并且还安装了很多文件 我不想删除这个 boost
  • 多个 JSlider 相互反应始终等于 100%

    我正在尝试向 java swing 应用程序添加 3 个 JSlider 以便三个滑块的总价值总和为 100 每个滑块都是一个概率 滑块 A 是将值添加到队列的概率 滑块 B是从队列中删除某个值的概率 滑块 C 是什么都不发生的概率 示例