Java Swing:JButtons和JSliders不会单独工作

问题描述 投票:0回答:3

我用Java中的Swing制作UI相对比较新,而且我在按钮和滑块在我的项目中工作时遇到了麻烦。我试图测试我的滑块和按钮是否适用于打印,但发现它们无法正常工作。

如果我不检查哪个滑块/按钮与之交互,我将得到结果。 I.E.如果按下项目中的任何滑块或按钮,我将只获得结果。但显然我希望他们都做不同的事情。

到目前为止,这是我的代码

main.Java

//Imports here, just saving space

public class Main
{
    public static void main(String[] args) throws IOException
    {
        ImageProcessor e = new ImageProcessor();
        e.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        e.initGUI();
    }
}

image processor.Java

import java.awt.*;
import java.awt.image.*;
import java.io.*;
import javax.imageio.*;
import javax.swing.*;
import java.awt.event.*;
import javax.swing.event.*;

public class ImageProcessor extends JFrame
{

    JPanel          imagePane;
    JCheckBox       xView, yView, zView, mip;
    JLabel          xSliceLabel, ySliceLabel, zSliceLabel;
    JSlider         xSliceSlider, ySliceSlider, zSliceSlider;
    JLabel          xImageIcon, yImageIcon, zImageIconl;
    BufferedImage   xImage, yImage, zImage;
    short           cthead[][][];
    short           min, max;
    double          ai, aj, ak;
    JButton         test;


    public void initGUI()
    {
        // Build the window to contain everything
        setTitle("Simple Example");
        setSize(1200, 900);

        /////// MAKE THE GUI ///////


        // Container Window
        Container container = getContentPane();
        container.setLayout(null);


        // Check boxes to select view
        JCheckBox xView = new JCheckBox("X View");
        xView.setBounds(150, 45, 65, 20);
        container.add(xView);

        JCheckBox yView = new JCheckBox("Y View");
        yView.setBounds(225, 45, 65, 20);
        container.add(yView);

        JCheckBox zView = new JCheckBox("Z View");
        zView.setBounds(300, 45, 65, 20);
        container.add(zView);


        // Panel to show CT Head image
        JPanel imagePane = new JPanel();
        imagePane.setBounds(25, 75, 512, 512);
        imagePane.setBorder(BorderFactory.createTitledBorder("Image View"));
        container.add(imagePane);

        // Image labels to display BufferedImages
        JLabel xImageIcon = new JLabel();
        imagePane.add(xImageIcon);



        // Checkbox for MIP on/off
        JCheckBox mip = new JCheckBox("MIP");
        mip.setBounds(125, 630, 65, 20);
        container.add(mip);


        // Slider labels
        JLabel xSliceLabel = new JLabel("X Slice");
        xSliceLabel.setBounds(50, 675, 60, 25);
        container.add(xSliceLabel);

        JLabel ySliceLabel = new JLabel("Y Slice");
        ySliceLabel.setBounds(50, 725, 60, 25);
        container.add(ySliceLabel);

        JLabel zSliceLabel = new JLabel("Z Slice");
        zSliceLabel.setBounds(50, 775, 60, 25);
        container.add(zSliceLabel);


        // Slice sliders
        JSlider xSliceSlider = new JSlider(0, 255, 76);
        xSliceSlider.setBounds(125, 675, 256, 25);
        container.add(xSliceSlider);

        JSlider ySliceSlider = new JSlider(0, 255, 76);
        ySliceSlider.setBounds(125, 725, 256, 25);
        container.add(ySliceSlider);

        JSlider zSliceSlider = new JSlider(0, 112, 76);
        zSliceSlider.setBounds(125, 775, 256, 25);
        container.add(zSliceSlider);

        JButton test = new JButton("Test");
        test.setBounds(50, 700, 60, 25);
        container.add(test);

        // Handler class
        GUIEventHandler handler = new GUIEventHandler();

        xSliceSlider.addChangeListener(handler);
        ySliceSlider.addChangeListener(handler);
        zSliceSlider.addChangeListener(handler);

        test.addActionListener(handler);


        setLocationRelativeTo(null);
        setDefaultCloseOperation(EXIT_ON_CLOSE);
        setVisible(true);

    }

    private class GUIEventHandler implements ActionListener, ChangeListener
    {
        public void stateChanged(ChangeEvent e)
        {
            if (e.getSource() == xSliceSlider)
            {
                System.out.println("x Slider changed!");

            } else if (e.getSource() == ySliceSlider)
            {
                System.out.println("y Slice changed!");

            } else if (e.getSource() == zSliceSlider)
            {
                System.out.println("Z slice changed!");
            }

            //This will work if any of the sliders change state
            //System.out.println("Something changed");
        }

        public void actionPerformed(ActionEvent e)
        {
            if (e.getSource() == test)
            {
                System.out.println("Testing...");
            }

            //This will also work if any button is pressed
            //System.out.println("Testing again...");
        }
    }
}

我非常感谢任何有知识的人看着这个并指出我正确的方向。

非常感谢大家!

java swing actionlistener eventhandler changelistener
3个回答
4
投票

你定义了:

  • 按钮和滑块的字段
  • 同名的局部变量

在处理程序中,您可以访问字段,但它们永远不会被初始化!

initGUI中,重写一些行:

JSlider xSliceSlider = new JSlider(0, 255, 76); // define a local variable

如:

xSliceSlider = new JSlider(0, 255, 76); // initialize the field

2
投票

我敢打赌,这不是你的GUI在计算机中的样子......是吗?

enter image description here

文字被裁剪,但是......为什么?你可能想知道。

那么,这是因为你使用null-layout并手动设置每个组件的边界及其大小。

但是......我该如何解决这个问题呢?

首先是一些背景阅读:qazxsw poi,qazxsw poi,qazxsw poi

现在,对于你的实际问题,你可以关注@MadProgrammer对Null layout is evil的回答,他说:

实际上,如果可以,你应该给每个滑块它自己的监听器并直接从源处理它...

Why is it frowned upon to use a null layout in Swing?

按照上面的提示,您可以按如下方式修复GUI,让this answer为您处理窗口大小,因此GUI在每台计算机中都是正确的,例如:

this question

JSlider slider = new JSlider();
slider.addChangeListener(new ChangeListener() {
    public void stateChanged(ChangeEvent e) {
        JSlider slider= (Slider)e.getSource();
        // Do funky stuff
    }
});

以下是Layout managers调用打印的示例:

enter image description here

2
投票

不是为多个GUI对象使用同一个事件处理程序,而是为每个GUI对象使用单独的事件处理程序是很常见的。这通常通过实现事件侦听器接口的匿名类来完成。

在您的应用程序中,您将替换您的代码

import java.awt.Dimension;
import java.awt.GridLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

import javax.swing.BorderFactory;
import javax.swing.BoxLayout;
import javax.swing.JButton;
import javax.swing.JCheckBox;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JSlider;
import javax.swing.SwingConstants;
import javax.swing.SwingUtilities;

public class ImageProcessingCorrected {
    private JFrame frame;

    private JPanel pane;
    private JPanel imagePane;
    private JPanel boxesPane;
    private JPanel controlsPane;
    private JPanel slidersPane;

    private JCheckBox[] boxes;

    private JButton button;

    private JCheckBox mipBox;

    private JSlider[] sliders;

    private static final String[] AXIS = new String[] {"X", "Y", "Z"};

    public static void main(String[] args) {
        SwingUtilities.invokeLater(new ImageProcessingCorrected()::createAndShowGui);
    }

    @SuppressWarnings("serial")
    private void createAndShowGui() {
        frame = new JFrame(getClass().getSimpleName());

        pane = new JPanel();
        imagePane = new JPanel() {
            @Override
            public Dimension getPreferredSize() {
                return new Dimension(200, 400);
            }
        };
        boxesPane = new JPanel();
        controlsPane = new JPanel();
        slidersPane = new JPanel();

        mipBox = new JCheckBox("MIP");

        button = new JButton("Test");
        button.addActionListener(e -> {
            System.out.println("Test button pressed!");
        });

        pane.setLayout(new BoxLayout(pane, BoxLayout.PAGE_AXIS));
        boxesPane.setLayout(new GridLayout(1, 3));
        slidersPane.setLayout(new GridLayout(3, 2, 5, 15));

        boxes = new JCheckBox[3];
        sliders = new JSlider[3];

        for (int i = 0; i < boxes.length; i++) {
            boxes[i] = new JCheckBox(AXIS[i] + " View");
            boxes[i].setHorizontalAlignment(SwingConstants.CENTER);
            boxesPane.add(boxes[i]);
        }

        for (int i = 0; i < sliders.length; i++) {
            sliders[i] = new JSlider(0, 100, 50);
            sliders[i].setName(AXIS[i] + " Slice");
            JLabel label = new JLabel(AXIS[i] + " Slice");
            label.setHorizontalAlignment(SwingConstants.CENTER);

            sliders[i].addChangeListener(e -> {
                System.out.println(((JSlider) e.getSource()).getName() + " changed!");
            });

            slidersPane.add(label);
            slidersPane.add(sliders[i]);
        }

        controlsPane.add(mipBox);
        controlsPane.add(button);

        imagePane.setBorder(BorderFactory.createTitledBorder("Image View"));

        pane.add(boxesPane);
        pane.add(imagePane);
        pane.add(controlsPane);
        pane.add(slidersPane);

        frame.add(pane);

        frame.pack();
        frame.setVisible(true);
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    }
}

通过此代码

sysout

这样做你不再需要X Slice changed! X Slice changed! X Slice changed! Y Slice changed! Y Slice changed! Y Slice changed! Y Slice changed! Z Slice changed! Z Slice changed! Z Slice changed! Z Slice changed! Test button pressed! 类了,也没有必要在事件处理方法中检查 // Handler class GUIEventHandler handler = new GUIEventHandler(); xSliceSlider.addChangeListener(handler); ySliceSlider.addChangeListener(handler); zSliceSlider.addChangeListener(handler); test.addActionListener(handler);

顺便说一句:如果您运行的是Java 8+,则可以通过将 xSliceSlider.addChangeListener(new ChangeListener() { @Override public void stateChanged(ChangeEvent e) { System.out.println("x Slider changed!"); } }); ySliceSlider.addChangeListener(new ChangeListener() { @Override public void stateChanged(ChangeEvent e) { System.out.println("y Slider changed!"); } }); zSliceSlider.addChangeListener(new ChangeListener() { @Override public void stateChanged(ChangeEvent e) { System.out.println("z Slider changed!"); } }); test.addActionListener(new ActionListener() { @Override public void actionPerformed(ActionEvent e) { System.out.println("Testing..."); } }); 用于事件处理程序来更简洁地重写上面的代码:

GUIEventHandler
© www.soinside.com 2019 - 2024. All rights reserved.