如何根据 javafx 中的 Action 为同一个 tableview 提供上下文菜单?
How to give a Contextmenu to same tableview according to Action in javafx?
我有一个组合框和一个 table。组合框有两个项目,它们是 type1 和 type2。当我 select 来自组合的 type1 时,table 上下文菜单显示 "type1 menu",当我 select 来自组合的 type2 时,table 上下文菜单显示 "type2 menu"。如何弄清楚这个数字?
这是我的实验代码....但它不能正常工作..!
import java.util.Arrays;
import javafx.application.Application;
import javafx.beans.binding.Bindings;
import javafx.beans.property.SimpleStringProperty;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.event.ActionEvent;
import javafx.geometry.Insets;
import javafx.scene.Group;
import javafx.scene.Scene;
import javafx.scene.control.ComboBox;
import javafx.scene.control.ContextMenu;
import javafx.scene.control.Label;
import javafx.scene.control.MenuItem;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableRow;
import javafx.scene.control.TableView;
import javafx.scene.control.cell.PropertyValueFactory;
import javafx.scene.layout.VBox;
import javafx.scene.text.Font;
import javafx.stage.Stage;
public class TableViewSample extends Application {
private final TableView<Person> table = new TableView<>();
private final ComboBox<String> combo = new ComboBox<>();
private final ObservableList<Person> data =
FXCollections.observableArrayList(
new Person("Jacob", "Smith", "jacob.smith@example.com"),
new Person("Isabella", "Johnson", "isabella.johnson@example.com"),
new Person("Ethan", "Williams", "ethan.williams@example.com"),
new Person("Emma", "Jones", "emma.jones@example.com"),
new Person("Michael", "Brown", "michael.brown@example.com")
);
public static void main(String[] args) {
launch(args);
}
@Override
public void start(Stage stage) {
Scene scene = new Scene(new Group());
stage.setTitle("Table View Sample");
stage.setWidth(450);
stage.setHeight(500);
final Label label = new Label("Right Click a table Row");
label.setFont(new Font("Arial", 20));
combo.getItems().setAll("Type1","Type2");
combo.getSelectionModel().select(0);
table.setEditable(true);
//----------------------- Add table column ------------------------------//
TableColumn<Person, String> firstNameCol = new TableColumn<>("First Name");
firstNameCol.setMinWidth(100);
firstNameCol.setCellValueFactory(
new PropertyValueFactory<>("firstName"));
TableColumn<Person, String> lastNameCol = new TableColumn<>("Last Name");
lastNameCol.setMinWidth(100);
lastNameCol.setCellValueFactory(
new PropertyValueFactory<>("lastName"));
TableColumn<Person, String> emailCol = new TableColumn<>("Email");
emailCol.setMinWidth(200);
emailCol.setCellValueFactory(
new PropertyValueFactory<>("email"));
table.setItems(data);
table.getColumns().addAll(Arrays.asList(firstNameCol, lastNameCol, emailCol));
setTableMenu(0); // You Can Comment This //
combo.setOnAction(event -> {
setTableMenu(combo.getSelectionModel().getSelectedIndex());
});
//-------------------------- Final Works ------------------------------//
final VBox vbox = new VBox();
vbox.setSpacing(5);
vbox.setPadding(new Insets(10, 0, 0, 10));
vbox.getChildren().addAll(label, combo, table);
((Group) scene.getRoot()).getChildren().addAll(vbox);
stage.setScene(scene);
stage.show();
}
//---------- Set What ContextMenu according to the combobox -----------//
private void setTableMenu(int selectedIndex) {
table.setRowFactory((TableView<Person> tableView) -> {
final TableRow<Person> row = new TableRow<>();
switch(selectedIndex){
case 0:
final ContextMenu contextMenu1 = new ContextMenu();
final MenuItem item1 = new MenuItem("Type1 menu");
item1.setOnAction((ActionEvent event) -> {
System.out.println("Type 1 menu selected");
});
contextMenu1.getItems().add(item1);
// Set context menu on row, but use a binding to make it only show for non-empty rows:
row.contextMenuProperty().bind(
Bindings.when(row.emptyProperty())
.then((ContextMenu) null)
.otherwise(contextMenu1)
);
break;
case 1:
final ContextMenu contextMenu2 = new ContextMenu();
final MenuItem item2 = new MenuItem("Type2 menu");
item2.setOnAction((ActionEvent event) -> {
System.out.println("Type 2 menu selected");
});
contextMenu2.getItems().add(item2);
// Set context menu on row, but use a binding to make it only show for non-empty rows:
row.contextMenuProperty().bind(
Bindings.when(row.emptyProperty())
.then((ContextMenu) null)
.otherwise(contextMenu2)
);
}
return row ;
});
}
public static class Person {
private final SimpleStringProperty firstName;
private final SimpleStringProperty lastName;
private final SimpleStringProperty email;
private Person(String fName, String lName, String email) {
this.firstName = new SimpleStringProperty(fName);
this.lastName = new SimpleStringProperty(lName);
this.email = new SimpleStringProperty(email);
}
public String getFirstName() {
return firstName.get();
}
public void setFirstName(String fName) {
firstName.set(fName);
}
public String getLastName() {
return lastName.get();
}
public void setLastName(String fName) {
lastName.set(fName);
}
public String getEmail() {
return email.get();
}
public void setEmail(String fName) {
email.set(fName);
}
}
}
更改 rowFactory
不会更新 TableRow
。这些行仍将是最初创建的行 rowFactory
并且上下文菜单的类型在创建行时决定并且永远不会更改。
然而,您可以在 ContextMenu
显示之前准备它,并根据 ComboBox
的状态对其进行修改:
// in start method
table.setRowFactory((TableView<Person> tableView) -> {
final TableRow<Person> row = new TableRow<>();
final ContextMenu contextMenu = new ContextMenu();
final MenuItem item1 = new MenuItem("Type1 menu");
item1.setOnAction((ActionEvent event) -> {
System.out.println("Type 1 menu selected");
});
contextMenu.getItems().add(item1);
final MenuItem item2 = new MenuItem("Type2 menu");
item2.setOnAction((ActionEvent event) -> {
System.out.println("Type 2 menu selected");
});
row.contextMenuProperty().bind(
Bindings.when(row.emptyProperty())
.then((ContextMenu) null)
.otherwise(contextMenu)
);
row.setOnContextMenuRequested(evt -> {
// update menu when requested
switch (combo.getSelectionModel().getSelectedIndex()) {
case 0:
contextMenu.getItems().setAll(item1);
break;
case 1:
contextMenu.getItems().setAll(item2);
break;
}
});
return row;
});
/*setTableMenu(0);
combo.setOnAction(event -> {
setTableMenu(combo.getSelectionModel().getSelectedIndex());
});*/
或者,您也可以对所有行使用相同的 ContextMenu
,并在 ComboBox
:
中的更改中修改它
// in start method
final MenuItem item1 = new MenuItem("Type1 menu");
item1.setOnAction((ActionEvent event) -> {
System.out.println("Type 1 menu selected");
});
final ContextMenu contextMenu = new ContextMenu(item1);
final MenuItem item2 = new MenuItem("Type2 menu");
item2.setOnAction((ActionEvent event) -> {
System.out.println("Type 2 menu selected");
});
combo.setOnAction(evt -> {
switch (combo.getSelectionModel().getSelectedIndex()) {
case 0:
contextMenu.getItems().setAll(item1);
break;
case 1:
contextMenu.getItems().setAll(item2);
break;
}
});
table.setRowFactory((TableView<Person> tableView) -> {
final TableRow<Person> row = new TableRow<>();
row.contextMenuProperty().bind(
Bindings.when(row.emptyProperty())
.then((ContextMenu) null)
.otherwise(contextMenu)
);
return row;
});
/*setTableMenu(0);
combo.setOnAction(event -> {
setTableMenu(combo.getSelectionModel().getSelectedIndex());
});*/
我有一个组合框和一个 table。组合框有两个项目,它们是 type1 和 type2。当我 select 来自组合的 type1 时,table 上下文菜单显示 "type1 menu",当我 select 来自组合的 type2 时,table 上下文菜单显示 "type2 menu"。如何弄清楚这个数字? 这是我的实验代码....但它不能正常工作..!
import java.util.Arrays;
import javafx.application.Application;
import javafx.beans.binding.Bindings;
import javafx.beans.property.SimpleStringProperty;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.event.ActionEvent;
import javafx.geometry.Insets;
import javafx.scene.Group;
import javafx.scene.Scene;
import javafx.scene.control.ComboBox;
import javafx.scene.control.ContextMenu;
import javafx.scene.control.Label;
import javafx.scene.control.MenuItem;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableRow;
import javafx.scene.control.TableView;
import javafx.scene.control.cell.PropertyValueFactory;
import javafx.scene.layout.VBox;
import javafx.scene.text.Font;
import javafx.stage.Stage;
public class TableViewSample extends Application {
private final TableView<Person> table = new TableView<>();
private final ComboBox<String> combo = new ComboBox<>();
private final ObservableList<Person> data =
FXCollections.observableArrayList(
new Person("Jacob", "Smith", "jacob.smith@example.com"),
new Person("Isabella", "Johnson", "isabella.johnson@example.com"),
new Person("Ethan", "Williams", "ethan.williams@example.com"),
new Person("Emma", "Jones", "emma.jones@example.com"),
new Person("Michael", "Brown", "michael.brown@example.com")
);
public static void main(String[] args) {
launch(args);
}
@Override
public void start(Stage stage) {
Scene scene = new Scene(new Group());
stage.setTitle("Table View Sample");
stage.setWidth(450);
stage.setHeight(500);
final Label label = new Label("Right Click a table Row");
label.setFont(new Font("Arial", 20));
combo.getItems().setAll("Type1","Type2");
combo.getSelectionModel().select(0);
table.setEditable(true);
//----------------------- Add table column ------------------------------//
TableColumn<Person, String> firstNameCol = new TableColumn<>("First Name");
firstNameCol.setMinWidth(100);
firstNameCol.setCellValueFactory(
new PropertyValueFactory<>("firstName"));
TableColumn<Person, String> lastNameCol = new TableColumn<>("Last Name");
lastNameCol.setMinWidth(100);
lastNameCol.setCellValueFactory(
new PropertyValueFactory<>("lastName"));
TableColumn<Person, String> emailCol = new TableColumn<>("Email");
emailCol.setMinWidth(200);
emailCol.setCellValueFactory(
new PropertyValueFactory<>("email"));
table.setItems(data);
table.getColumns().addAll(Arrays.asList(firstNameCol, lastNameCol, emailCol));
setTableMenu(0); // You Can Comment This //
combo.setOnAction(event -> {
setTableMenu(combo.getSelectionModel().getSelectedIndex());
});
//-------------------------- Final Works ------------------------------//
final VBox vbox = new VBox();
vbox.setSpacing(5);
vbox.setPadding(new Insets(10, 0, 0, 10));
vbox.getChildren().addAll(label, combo, table);
((Group) scene.getRoot()).getChildren().addAll(vbox);
stage.setScene(scene);
stage.show();
}
//---------- Set What ContextMenu according to the combobox -----------//
private void setTableMenu(int selectedIndex) {
table.setRowFactory((TableView<Person> tableView) -> {
final TableRow<Person> row = new TableRow<>();
switch(selectedIndex){
case 0:
final ContextMenu contextMenu1 = new ContextMenu();
final MenuItem item1 = new MenuItem("Type1 menu");
item1.setOnAction((ActionEvent event) -> {
System.out.println("Type 1 menu selected");
});
contextMenu1.getItems().add(item1);
// Set context menu on row, but use a binding to make it only show for non-empty rows:
row.contextMenuProperty().bind(
Bindings.when(row.emptyProperty())
.then((ContextMenu) null)
.otherwise(contextMenu1)
);
break;
case 1:
final ContextMenu contextMenu2 = new ContextMenu();
final MenuItem item2 = new MenuItem("Type2 menu");
item2.setOnAction((ActionEvent event) -> {
System.out.println("Type 2 menu selected");
});
contextMenu2.getItems().add(item2);
// Set context menu on row, but use a binding to make it only show for non-empty rows:
row.contextMenuProperty().bind(
Bindings.when(row.emptyProperty())
.then((ContextMenu) null)
.otherwise(contextMenu2)
);
}
return row ;
});
}
public static class Person {
private final SimpleStringProperty firstName;
private final SimpleStringProperty lastName;
private final SimpleStringProperty email;
private Person(String fName, String lName, String email) {
this.firstName = new SimpleStringProperty(fName);
this.lastName = new SimpleStringProperty(lName);
this.email = new SimpleStringProperty(email);
}
public String getFirstName() {
return firstName.get();
}
public void setFirstName(String fName) {
firstName.set(fName);
}
public String getLastName() {
return lastName.get();
}
public void setLastName(String fName) {
lastName.set(fName);
}
public String getEmail() {
return email.get();
}
public void setEmail(String fName) {
email.set(fName);
}
}
}
更改 rowFactory
不会更新 TableRow
。这些行仍将是最初创建的行 rowFactory
并且上下文菜单的类型在创建行时决定并且永远不会更改。
然而,您可以在 ContextMenu
显示之前准备它,并根据 ComboBox
的状态对其进行修改:
// in start method
table.setRowFactory((TableView<Person> tableView) -> {
final TableRow<Person> row = new TableRow<>();
final ContextMenu contextMenu = new ContextMenu();
final MenuItem item1 = new MenuItem("Type1 menu");
item1.setOnAction((ActionEvent event) -> {
System.out.println("Type 1 menu selected");
});
contextMenu.getItems().add(item1);
final MenuItem item2 = new MenuItem("Type2 menu");
item2.setOnAction((ActionEvent event) -> {
System.out.println("Type 2 menu selected");
});
row.contextMenuProperty().bind(
Bindings.when(row.emptyProperty())
.then((ContextMenu) null)
.otherwise(contextMenu)
);
row.setOnContextMenuRequested(evt -> {
// update menu when requested
switch (combo.getSelectionModel().getSelectedIndex()) {
case 0:
contextMenu.getItems().setAll(item1);
break;
case 1:
contextMenu.getItems().setAll(item2);
break;
}
});
return row;
});
/*setTableMenu(0);
combo.setOnAction(event -> {
setTableMenu(combo.getSelectionModel().getSelectedIndex());
});*/
或者,您也可以对所有行使用相同的 ContextMenu
,并在 ComboBox
:
// in start method
final MenuItem item1 = new MenuItem("Type1 menu");
item1.setOnAction((ActionEvent event) -> {
System.out.println("Type 1 menu selected");
});
final ContextMenu contextMenu = new ContextMenu(item1);
final MenuItem item2 = new MenuItem("Type2 menu");
item2.setOnAction((ActionEvent event) -> {
System.out.println("Type 2 menu selected");
});
combo.setOnAction(evt -> {
switch (combo.getSelectionModel().getSelectedIndex()) {
case 0:
contextMenu.getItems().setAll(item1);
break;
case 1:
contextMenu.getItems().setAll(item2);
break;
}
});
table.setRowFactory((TableView<Person> tableView) -> {
final TableRow<Person> row = new TableRow<>();
row.contextMenuProperty().bind(
Bindings.when(row.emptyProperty())
.then((ContextMenu) null)
.otherwise(contextMenu)
);
return row;
});
/*setTableMenu(0);
combo.setOnAction(event -> {
setTableMenu(combo.getSelectionModel().getSelectedIndex());
});*/