在JSF中按ID查找组件

问题描述 投票:46回答:5

我想通过提供的ID从托管bean中找到一些UIComponent

我编写了以下代码:

private UIComponent getUIComponent(String id) {  
      return FacesContext.getCurrentInstance().getViewRoot().findComponent(id) ;  
}

我将p:inputTextarea定义为:

<p:inputTextarea id="activityDescription" value="#{adminController.activityDTO.activityDescription}" required="true" maxlength="120"
    autoResize="true" counter="counter" counterTemplate="{0} characters remaining." cols="80" rows="2" />

[现在,如果以getUIComponent("activityDescription")的形式调用该方法,则返回null,但是如果以getUIComponent("adminTabView:activityForm:activityDescription")的形式调用它,则可以获得org.primefaces.component.inputtextarea.InputTextarea实例。

是否有任何方法可以获取仅具有ID(即“ activityDescription”)而不是绝对ID(即“ adminTabView:activityForm:activityDescription”的组件?

jsf jsf-2 primefaces uicomponents
5个回答
44
投票

您可以使用以下代码:

public UIComponent findComponent(final String id) {

    FacesContext context = FacesContext.getCurrentInstance(); 
    UIViewRoot root = context.getViewRoot();
    final UIComponent[] found = new UIComponent[1];

    root.visitTree(new FullVisitContext(context), new VisitCallback() {     
        @Override
        public VisitResult visit(VisitContext context, UIComponent component) {
            if (component != null 
                && component.getId() != null 
                && component.getId().equals(id)) {
                found[0] = component;
                return VisitResult.COMPLETE;
            }
            return VisitResult.ACCEPT;              
        }
    });

    return found[0];

}

此代码将只在您传递了id的树中找到第一个组件。如果树中有2个具有相同名称的组件,则必须执行一些自定义操作(如果它们在2个不同的命名容器下,则可能是这样做的)。


2
投票

我尝试此代码,对您有帮助:

private static UIComponent getUIComponentOfId(UIComponent root, String id){
    if(root.getId().equals(id)){
        return root;
    }
    if(root.getChildCount() > 0){
        for(UIComponent subUiComponent : root.getChildren()){
                UIComponent returnComponent = getUIComponentOfId(subUiComponent, id);
                if(returnComponent != null){
                    return returnComponent;
            }
        }
    }
    return null;
}

谢谢


1
投票

也许不可能。 FacesContext.getCurrentInstance().getViewRoot().findComponent(id)方法仅返回一个UIComponent。 ViewRoot构造为一棵树,因此如果您在视图中有两种形式,每个形式都有一个带有id="text"的组件,则它们会将其父组件添加到id中,因此不会发生冲突。如果将两个id="text"组件放在同一表格中,则会抛出java.lang.IllegalStateException

如果要查找具有搜索ID的所有组件,则可以编写一个实现以下方法的方法:

List<UIComponent> foundComponents = new ArrayList();
for(UIComponent component: FacesContext.getCurrentInstance().getViewRoot().getChildren()) {
    if(component.getId().contains("activityDescription")){
        foundComponents.add(component);
    }
}

或者如果您想找到第一个匹配项:

UIComponent foundComponent;
for(UIComponent component: FacesContext.getCurrentInstance().getViewRoot().getChildren()) {
    if(component.getId().contains("activityDescription")){
        foundComponent = component;
        break;
    }
}

1
投票

只需将prependId="false"放在此文本区域所在的表单中。


0
投票

是,在所有为NamingContainers的父组件中,您必须添加属性prependId="false"-它肯定会在<h:form>中起作用,而在其他组件中也应起作用。如果无法通过.xhtml文件中的属性进行设置,则必须以编程方式设置该值。

问题后作者的建议:

如果您正在使用的组件中没有这样的属性,请尝试像下面这样写查找方法:

private UIComponent findComponent(String id, UIComponent where) {
if (where == null) {
   return null;
}
else if (where.getId().equals(id)) {
   return where;
}
else {
   List<UIComponent> childrenList = where.getChildren();
   if (childrenList == null || childrenList.isEmpty()) {
      return null;
   }
   for (UIComponent child : childrenList) {
      UIComponent result = null;
      result = findComponent(id, child);
      if(result != null) {
         return result;
   }
   return null;
}   

接下来只是调用

UIComponent iamLookingFor = findComponent(myId, FacesContext.getCurrentInstance().getViewRoot());

有帮助吗?

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