Hi I am making a gui for a flight booking system and I have to make sure user selects only one flight in the JList when the query button is clicked, so I decided to make a JList of RadioButton as follows:
flightsList = new JList<JRadioButton>();
button.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
//user inputs
String takeoff = (String) from.getSelectedItem();
String destina = (String) to.getSelectedItem();
String flyDate = (String) date.getText();
try {
//get all available flights
Flight[] flights = manager.getFlights(takeoff, destina, flyDate);
//model for list
DefaultListModel<JRadioButton> model =
new DefaultListModel<JRadioButton>();
//fill model with flights found
for(int flightNum = 0; flightNum < flights.length; flightNum++) {
model.addElement(new JRadioButton(flights[flightNum].toString()));
}//for
//put model into jlist
flightsList.setModel(model);
} catch (BadQueryException bqe) {
JRadioButton[] errorMessage = {
new JRadioButton("Error: " + bqe.getMessage()) };
//put error message into list
flightsList.setListData(errorMessage);
}//try catch
}//actionPerformed
});
When I run, the JList shows lines of:
javax.swing.JRadioButton[,0,0,0x0,invalid,alignmentX=0.0,alignmentY=0.5,border=javax.swing.plaf.BorderUIResource$CompoundBorderUIResource@78092fac,flags=296,maximumSize=,minimumSize=,preferredSize=,defaultIcon=,disabledIcon=,disabledSelectedIcon=,margin=javax.swing.plaf.InsetsUIResource[top=2,left=2,bottom=2,right=2],paintBorder=false,paintFocus=true,pressedIcon=,rolloverEnabled=true,rolloverIcon=,rolloverSelectedIcon=,selectedIcon=,text=BA002 | London >> Manchester | Tue, 01/10/2019 06:30]
May I know what happened and how to solve this?
Thank you.
When I run, the JList shows lines of:
javax.swing.JRadioButton[,0,0,0x0,invalid,alignmentX=0.0,alignmentY=0.5....
]May I know what happened and how to solve this?
This happens because your JList
has a DefaultListCellRenderer. As you can see, this class extends
JLabel
. The method DefaultListCellRenderer#getListCellRendererComponent()
is getting an argument Object value
. This value's type is equals to the type of your JList
.
With that being said, your JList
has JRadioButton
as a generic type (JList< JRadioButton>
), which means the Object value arguemnt is a JRadioButton
.
Now, DefaultListCellRenderer
in order to get its text, it calls the toString()
method of the value hence you get this kind of text in your list's cells. (The toString()
method of a JRadioButton, returns its details, coordinates, sizes, etc...)
The solution:
It would be to use a custom ListCellRenderer. This way you can render any value-property of the argument named "value"i in getListCellRendererComponent()
method. In your case, you need to render JRadioButton
.
I will share an example with comments inside the code in order to understand this better.
import java.awt.BorderLayout;
import java.awt.Component;
import java.util.List;
import javax.swing.DefaultListModel;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JList;
import javax.swing.JRadioButton;
import javax.swing.ListCellRenderer;
import javax.swing.SwingUtilities;
public class JListJRadioButtonRenderer extends JFrame {
public JListJRadioButtonRenderer() {
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
addList();
setSize(300, 300);
setLocationRelativeTo(null);
}
private void addList() {
JList<JRadioButton> list = new JList<>();
DefaultListModel<JRadioButton> model = new DefaultListModel<>();
// Add the custom renderer.
list.setCellRenderer(new ListCellRenderer<JRadioButton>() {
@Override
public Component getListCellRendererComponent(JList<? extends JRadioButton> list, JRadioButton value,
int index, boolean isSelected, boolean cellHasFocus) {
// Fix background for selected cells.
value.setBackground(isSelected ? list.getSelectionBackground() : null);
// Select the JRadioButton too since it is selected in the list.
value.setSelected(isSelected);
return value;
}
});
list.setModel(model);
JRadioButton stackButton = new JRadioButton("Hello Stack");
JRadioButton overButton = new JRadioButton("Hello Over");
JRadioButton flowButton = new JRadioButton("Hello Flow");
model.addElement(stackButton);
model.addElement(overButton);
model.addElement(flowButton);
getContentPane().add(list);
JButton printSelected = new JButton("Print selected");
printSelected.addActionListener(e -> {
List<JRadioButton> selectedButtons = list.getSelectedValuesList();
for (JRadioButton r : selectedButtons)
System.out.println(r.getText());
});
getContentPane().add(printSelected, BorderLayout.PAGE_END);
}
public static void main(String[] args) {
SwingUtilities.invokeLater(() -> new JListJRadioButtonRenderer().setVisible(true));
}
}
Preview: