使用Vaadin和Spring Boot实现依赖注入的良好实践

时间:2017-05-10 21:28:08

标签: java spring-mvc spring-boot dependency-injection vaadin

我有以下问题:我有一个扩展UI的Mainview类。在这个类中依赖注入工作。但是这个班级拥有一个带有视图的导航器。此视图包含也保存视图的Tabsheet。现在我需要注入一个Spring @Service类。但是DI链被破坏了。我已经知道依赖注射链不应该被打断(我在这里通过另一个问题找到了这个)。但我无法找到一种方法来保持这条链的活力。谁有一个良好的tipp如何实现观点也注入和尽管这样我可以随处使用导航器?我想我做得太复杂甚至错了。这是我的Mainview(扩展UI)类:

@Theme("valo")
@SpringUI
public class MainView extends UI {

private static final long serialVersionUID = 1L;
private static final Logger LOGGER = LoggerFactory.getLogger(MainView.class);
public static Navigator navigator;
private List<String> availableRealms = new ArrayList<>();
private ComboBox<String> select =new ComboBox<>();

@Autowired
private SpringViewProvider viewProvider;


@Override
protected void init(VaadinRequest request) {
    LOGGER.info("Initializing MainView container...");
    createMainView();

}

public void createMainView(){
    HorizontalLayout comboBoxLayout = new HorizontalLayout();

    comboBoxLayout.addComponent(select);
    availableRealms.add("Master-Realm");
    select.setTextInputAllowed(false);
    select.setItems(availableRealms);
    select.setSelectedItem(availableRealms.get(0));
    LOGGER.info("Building mainView area...");
    final VerticalLayout root = new VerticalLayout();
    root.setSizeFull();
    root.setMargin(true);
    root.setSpacing(true);
    setContent(root);
    final CssLayout navigationBar = new CssLayout();
    //      navigationBar.addComponent(comboBoxLayout);
    navigationBar.addStyleName(ValoTheme.LAYOUT_COMPONENT_GROUP);
    navigationBar.addComponent(createNavigationButton("Dashboard",DashBoardView.VIEW_NAME));
    navigationBar.addComponent(createNavigationButton("Realm Settings",RealmSettingsView.VIEW_NAME));
    navigationBar.addComponent(createNavigationButton("Client Management",ClientManagementView.VIEW_NAME));
    navigationBar.addComponent(createNavigationButton("User Management",UserManagementView.VIEW_NAME));
    navigationBar.addComponent(createNavigationButton("Group Management",GroupManagementView.VIEW_NAME));
    comboBoxLayout.addComponent(navigationBar);

    root.addComponent(comboBoxLayout);
    //      root.addComponent(navigationBar);

    final Panel viewContainer = new Panel();
    viewContainer.setSizeFull();
    root.addComponent(viewContainer);
    root.setExpandRatio(viewContainer, 1.0f);
    LOGGER.info("Initializing navigator...");
    navigator = new Navigator(this, viewContainer);
    navigator.addProvider(viewProvider);
    navigator.addView("", DashBoardView.class);

}

/**
 * Creates a new button with listener that uses the navigator.
 * @param caption
 * @param viewName
 * @return <b>Button</b>
 */
private Button createNavigationButton(String caption, final String viewName) {
    Button button = new Button(caption);
    button.addStyleName(ValoTheme.BUTTON_SMALL);
    // If you didn't choose Java 8 when creating the project, convert this to an anonymous listener class
    button.addClickListener(event -> getUI().getNavigator().navigateTo(viewName));
    return button;
}

}

这是持有标签页的ClientManagementView:

@SpringView(name = ClientManagementView.VIEW_NAME)
@ViewScope
public class ClientManagementView extends VerticalLayout implements View {

private final Logger LOGGER = LoggerFactory.getLogger(ClientManagementView.class);

private static final long serialVersionUID = 1L;

public static final String VIEW_NAME = "clientmanagementview";

@Autowired
ClientOverview clientOverview;

@Autowired
ClientCreationView clientCreationView;

@Autowired
private TabSheet tabsheet;

@PostConstruct
void init() {
    LOGGER.info("Initializing ClientView...");
    setSizeFull();
    setSpacing(true);
    setMargin(true);
    //      TabSheet tabsheet = new TabSheet();
    tabsheet.addStyleName(ValoTheme.TABSHEET_FRAMED);
    this.addComponent(tabsheet);
    tabsheet.addTab(clientOverview);
    tabsheet.addTab(clientCreationView);
}

@Override
public void enter(ViewChangeEvent event) {
    // do nothing.

}

}

现在是客户端创建视图:

@SpringView(name = ClientCreationView.VIEW_NAME)
@ViewScope
public class ClientCreationView extends VerticalLayout implements View {


private static final long serialVersionUID = 1L;
public static final String VIEW_NAME = "client_creationview";
private final Logger LOGGER = LoggerFactory.getLogger(ClientCreationView .class);

@Autowired
private TabSheet tabsheet;

@Autowired
private ClientCredentialFormular clientCredentialFormular;

@Autowired
private ClientRegisterFormular clientRegisterFormular;

public ClientCreationView(){
    LOGGER.info("Initializing ClientCreationView...");
    this.setCaption("Register New Client");
    LOGGER.info("Initializing ClientView...");
    setSizeFull();
    setSpacing(true);
    setMargin(true);
    //      TabSheet tabsheet = new TabSheet();
    this.addComponent(tabsheet);
    tabsheet.addTab(clientRegisterFormular);
    tabsheet.addTab(clientCredentialFormular);

}

@Override
public void enter(ViewChangeEvent event) {
    // Do nothing...

}

}

此视图的选项卡需要注入服务类:

@SpringView(name = ClientRegisterFormular.VIEW_NAME)
@ViewScope
public class ClientRegisterFormular extends VerticalLayout implements View {

private static final long serialVersionUID = 1L;
public static final String VIEW_NAME = "client_data_formular";
private static final Logger LOGGER = LoggerFactory.getLogger(ClientRegisterFormular.class);

@Autowired
IClientService clientService;

private TextField clientIdField = new TextField("Client-Id");
private TextField rootUriField = new TextField("Root-URI");
private TextField redirectUriField = new TextField("Redirect-URI");
private Button saveButton = new Button("Save");
private Button cancelButton = new Button("Cancel");
private FormLayout form = new FormLayout();
private SaveButtonListener saveButtonListener = new SaveButtonListener(this);
private CancelButtonListener cancelButtonListener = new CancelButtonListener(this);


public ClientRegisterFormular(){
    LOGGER.info("Initializing Client Data Formular...");
    this.setCaption("Client Data");
    this.setSizeFull();
    this.setSpacing(true);
    this.setMargin(true);
    /*
     * Panel for Formular:
     */
    Panel panel = new Panel("Register Client");
    panel.addStyleName("regFormPanel");
    panel.setSizeUndefined(); // Shrink to fit content

    //Formular:
    this.form.setMargin(true);
    this.form.setSizeFull();
    this.clientIdField.setWidth("900px");
    this.clientIdField.setRequiredIndicatorVisible(true);
    form.addComponent(clientIdField);
    this.rootUriField.setRequiredIndicatorVisible(true);
    this.rootUriField.setWidth("900px");
    form.addComponent(rootUriField);
    this.redirectUriField.setRequiredIndicatorVisible(true);
    this.redirectUriField.setWidth("900px");
    form.addComponent(redirectUriField);
    //TODO Checkboxes for

    //Button-Layout:
    HorizontalLayout buttonLayout = new HorizontalLayout();
    this.saveButton.addClickListener(saveButtonListener);
    buttonLayout.addComponent(this.saveButton);
    this.cancelButton.addClickListener(this.cancelButtonListener);
    buttonLayout.addComponent(this.cancelButton);
    buttonLayout.setComponentAlignment(this.saveButton, Alignment.TOP_LEFT);

    form.addComponent(buttonLayout);
    panel.setContent(form);
    this.addComponent(panel);
    //      this.addComponent(buttonPanel);
    this.setComponentAlignment(panel, Alignment.TOP_LEFT);

}

@Override
public void enter(ViewChangeEvent event) {
    // Do nothing..

}


private static class SaveButtonListener implements Button.ClickListener{

    private static final long serialVersionUID = 1L;
    private ClientRegisterFormular clientRegisterFormular;

    public SaveButtonListener(ClientRegisterFormular clientRegisterFormular){
        this.clientRegisterFormular = clientRegisterFormular;
    }

    @Override
    public void buttonClick(ClickEvent event) {
        LOGGER.info("Saving client...");
        Client newClient = new Client();
        newClient.setClientId(clientRegisterFormular.clientIdField.getValue());
        newClient.setRootUri(this.clientRegisterFormular.rootUriField.getValue());
        newClient.setRedirectUri(this.clientRegisterFormular.redirectUriField.getValue());
        newClient.setClientId(UUID.randomUUID().toString());
        this.clientRegisterFormular.clientService.createClient(newClient);
        refreshView();

    }

    public void refreshView() {
        MainView.navigator.navigateTo(ClientManagementView.VIEW_NAME);
    }

}

private static class CancelButtonListener implements Button.ClickListener{

    private static final long serialVersionUID = 1L;
    private ClientRegisterFormular clientRegisterFormular;

    public CancelButtonListener(ClientRegisterFormular clientRegisterFormular){
        this.clientRegisterFormular = clientRegisterFormular;
    }

    @Override
    public void buttonClick(ClickEvent event) {
        LOGGER.info("Canceling Client Creation...");
        refreshView();

    }


    public void refreshView() {
        MainView.navigator.navigateTo(ClientManagementView.VIEW_NAME);
    }

}

}

2 个答案:

答案 0 :(得分:1)

我解决了。问题不在于标签页不是弹簧组件(因为我在应用程序类中声明它是@bean)。问题是我没有被构造函数自动装配。通过将tabheet设置为构造函数的参数并设置autowire,我解决了它。

答案 1 :(得分:0)

我认为问题出在ClientManagementView中,您在其中创建两个视图:

tabsheet.addTab(new ClientOverview());
tabsheet.addTab(new ClientCreationView());

但它应该是Spring应该创建那些视图,因为你将它们声明为@SpringView并且你想要注入其他bean。

我认为您应该在ClientManagementView中自动装配它们:

@Autowired
ClientOverview clientOverview;

@Autowired
ClientCreationView clientCreationView;


tabsheet.addTab(clientOverview);
tabsheet.addTab(clientCreationView);