Right, i'm trying to create a program that allows a user to build a Space Marine (Warhammer 40k) army and theres a paticular portion of the program that i'm struggling with. This portion allows the user to select a 'Unit Type' from a ComboBox, then another ComboBox underneath is populated with all of the units for the 'Unit Type' you selected in the first ComboBox.
For example if you selected the Elites 'Unit Type' in the first ComboBox then the second ComboBox should be populated with all the units of the Elites 'Unit Type' which are as follows: "Dreadnought","Ironclad Dreadnought","Venerable Dreadnought", "Assault Terminator", "Centurion Assault", "Command", "Honour Guard", "Sternguard Veteran","Terminator","Vanguard Veterans". There are 8 'Unit Types' each with their own units.
The current setup I have to achieve this is; I have a List of 'Unit Types' called unitTypeList. This List holds all 8 'Unit Types'. I also have 8 other seperate lists (which each represent a 'Unit Type') that hold muliple units for that 'Unit Type' . Then I have one more list that holds each of the 8 seprate lists as mentioned before. So it's a List thats holds 8 other lists.
So to populate the first ComboBox with the 'Unit Types' I placed the unitTypeList in an ObservableList called unitTypeOlist. I then populated the first ComboBox with that ObservableList unitTypeOList. This works fine.
Now in order to populate the second ComboBox with the correct units, i decided to add a ChangeListener to the first ComboBox, so it can take the input of the first ComboBox and then correctly populate the second ComboBox with the units based on the selection of the 'Unit Type' in the first ComboBox.
But this is where the issue occurs. I created another ObservableList in the ChangeListener and place the List that holds the 8 seperate lists, that contain each unit for the 'Unit Type, in it. However when I try to run the program and select a 'Unit Type' from the first ComboBox the programs terminates with a gigantic error starting with; Exception in thread "JavaFX Application Thread" java.lang.ClassCastException: gui.model.Unit_Type cannot be cast to java.lang.Integer.
It's clearly something to do with the casting to an int. But i'm not sure how i would achieve what i want using a ChangeListener any other way. I need some help to meet the requirements in the first paragraph.
I've already tried searching for this, and found this question very helpful already. (My setup is heavily based off this question) But I think my requirements are slighty more complex.
Here is the code for the AddUnitPane class of my program (i've tried to comment it the best i can):
package gui.view;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import elites.terminator.Terminator;
import gui.model.Unit;
import gui.model.Unit_Type;
import javafx.beans.value.ChangeListener;
import javafx.beans.value.ObservableValue;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.event.ActionEvent;
import javafx.event.EventHandler;
import javafx.geometry.HPos;
import javafx.geometry.Pos;
import javafx.scene.control.Button;
import javafx.scene.control.ComboBox;
import javafx.scene.control.Label;
import javafx.scene.control.TextField;
import javafx.scene.layout.ColumnConstraints;
import javafx.scene.layout.GridPane;
import javafx.scene.layout.HBox;
import troops.scout.Scout;
public class AddUnitPane extends GridPane
{
private List<Unit_Type> unitTypeList; //Declare the unitTypeList, which holds a list of unit types
private List<String> elitesList, fastAtkList, heavySptList, hqList, lordsowList, specialCList, transportList, troopsList; //Declare the sublists that hold all the units for the type of unit
private List<List<String>> unitList; //Declare the unitList that holds all the sublists units
private Label unitTypeLbl, unitLbl, squadNameLbl, squadSizeLbl;
private ComboBox<Unit_Type> unitTypeCombo;
private ComboBox<String> unitCombo;
private ComboBox<Integer> squadSizeCombo;
private TextField squadNameTf;
private Button addSquadBtn;
public AddUnitPane()
{
this.setVgap(15);
this.setHgap(20);
this.setAlignment(Pos.CENTER);
ColumnConstraints col1 = new ColumnConstraints();
col1.setHalignment(HPos.RIGHT);
this.getColumnConstraints().add(col1);
//--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
unitTypeList = new ArrayList<>(); //Initialise the unitTypeList
Collections.addAll(unitTypeList, new Unit_Type("Elites"), new Unit_Type("Fast Attack"), new Unit_Type("Heavy Support"), new Unit_Type("HQ"), new Unit_Type("Lords Of War"),
new Unit_Type("Special Characters"), new Unit_Type("Transport"), new Unit_Type("Troops")); //Populate the unitTypeList
ObservableList unitTypeOList = FXCollections.observableArrayList(unitTypeList); //Add the UnitTypeList to an observableArrayList
//--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
elitesList = new ArrayList<>(); //Initialise and populate all the of the sublists that hold each unit for that type
Collections.addAll(elitesList, "Dreadnought", "Ironclad Dreadnought", "Venerable Dreadnought", "Assault Terminator", "Centurion Assault", "Command", "Honour Guard",
"Sternguard Veteran", "Terminator", "Vanguard Veterans");
fastAtkList = new ArrayList<>();
Collections.addAll(fastAtkList, "Attack Bike", "Stormhawk Interceptor", "Stormtalon Gunship", "Assault", "Bike", "Land Speeder", "Scout Bike");
heavySptList = new ArrayList<>();
Collections.addAll(heavySptList, "Hunter", "Land Raider Crusader", "Land Raider Redeemer", "Land Raider", "Predator", "Stalker", "Stormraaven Gunship", "Vindicator",
"Whirlwind", "Centurion Devastator", "Devastator", "Thunderfire Cannon");
hqList = new ArrayList<>();
Collections.addAll(hqList, "Captain", "Chaplain", "Librarian", "Techmarine");
lordsowList = new ArrayList<>();
Collections.addAll(lordsowList, "Marneus Calger", "Roboute Guilliman");
specialCList = new ArrayList<>();
Collections.addAll(specialCList, "Antaro Chronus", "Cato Sicarius", "Ortan Cassius", "Torias Telion", "Varro Tigurius");
transportList = new ArrayList<>();
Collections.addAll(transportList, "Drop Pod", "Land Speeder Storm", "Razorback", "Rhino");
troopsList = new ArrayList<>();
Collections.addAll(troopsList, "Scout", "Tactical");
//--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
unitList = new ArrayList<>(); //Initialise and populate the List thats holds other Lists that hold the different units
Collections.addAll(unitList, elitesList, fastAtkList, heavySptList, hqList, lordsowList, specialCList, transportList, troopsList);
//--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
unitTypeLbl = new Label("Select The Unit Class: "); //Initialise all of the labels
unitLbl = new Label("Select The Unit: ");
squadNameLbl = new Label("Squad Name: ");
squadSizeLbl = new Label("Squad Size");
//--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
unitTypeCombo = new ComboBox<Unit_Type>(); //Initialise the unitTypeCombo
unitTypeCombo.setItems(unitTypeOList); //Populate the unitTypeCombo with the UnitTypeOList (observable list) from line 56
unitTypeCombo.getSelectionModel().selectFirst(); //Set the unitTypeCombo to show the first item
unitCombo = new ComboBox<>(); //Initialise the unitCombo
unitTypeCombo.getSelectionModel().selectedItemProperty().addListener(new ChangeListener() //add a change listener to the unitTypeCombo
{
@Override
public void changed(ObservableValue arg0, Object arg1, Object arg2)
{
ObservableList unitOList = FXCollections.observableArrayList(unitList.get((int) arg2)); //add the unitList from line 88 to an observable list and cast arg2 to an int
unitCombo.setItems(unitOList); //set the unitCombo items with the observable list from above
}
});
unitCombo = new ComboBox<>();
squadNameTf = new TextField();
squadSizeCombo = new ComboBox<>();
addSquadBtn = new Button("Add Squad");
this.add(unitTypeLbl, 0, 1);
this.add(unitTypeCombo, 1, 1);
this.add(unitLbl, 0, 2);
this.add(unitCombo, 1, 2);
this.add(squadNameLbl, 0, 3);
this.add(squadNameTf, 1, 3);
this.add(squadSizeLbl, 0, 4);
this.add(squadSizeCombo, 1, 4);
this.add(new HBox(), 0, 5);
this.add(addSquadBtn, 1, 5);
}
public ComboBox<Unit_Type> getUnitClass()
{
return unitTypeCombo;
}
public ComboBox<String> getUnit()
{
return unitCombo;
}
public String getSquadName()
{
return squadNameTf.getText();
}
public ComboBox<Integer> getSquadSize()
{
return squadSizeCombo;
}
public void AddUnitHandler(EventHandler<ActionEvent> handler)
{
addSquadBtn.setOnAction(handler);
}
}
This is one of the reason why you should avoid using the raw type.
Your
ChangeListener
is written as if you were listening to aIntegerProperty
instead of aProperty<Unit_Type>
. A cast form aUnit_Type
object toInteger
is not possible so you get a exception in this expression:Your compiler would have complained about this issue, if a
ChangeListener<Integer>
was used instead.You could listen to the
selectedIndex
property instead (checking for valid indices of course --1
is also a possible value).It would be easier to simply add the list as property to
Unit_Type
though. This would allow you to write something like this:BTW: Since you never resize the lists, the following way of initializing them would be much simpler:
Even if you want to use a specific list type it would be a good idea to reduce boilerplate code and introduce a helper method: