我正在编写一个复合组件,用于包装一个输入元素,并使用“可选字段”标识和下面的h:message元素对其进行扩充。 这是组件(在input.xhtml文件中):
<composite:interface/>
<composite:implementation>
<div>
#{component.children[1].setStyleClass(component.children[1].valid ? '' : 'inputError')}
<composite:insertChildren/>
</div>
<h:panelGroup styleClass="optional"
rendered="#{!component.parent.children[1].required}"
layout="block" >
#{msgs['common.optional.field']}
</h:panelGroup>
<h:message id="msgId"
for="#{component.parent.children[1].id}" errorClass="error"/>
</composite:implementation>
同样,该组件的预期用途是包装一个h:input *元素,该元素应该是使用页面中它的第一个和直接子元素。
在使用页面中,我会写:
<h:form id="f1">
<h:panelGrid border="0" columns="2" style="width: 80%">
<f:facet name="header">
<col width="40%" />
<col width="60%" />
</f:facet>
<h:outputLabel styleClass="sectionBody" for="i1"
value="Input 1"></h:outputLabel>
<my:input id="ii1">
<h:inputText id="i1" size="20" maxlength="20" tabindex="0"
required="true" label="Input 1">
</h:inputText>
</my:input>
<h:outputLabel styleClass="sectionBody" for="i2"
value="Input 2"></h:outputLabel>
<my:input id="ii2">
<h:inputText id="i2" size="20" maxlength="20" tabindex="0"
label="Input 2">
</h:inputText>
</my:input>
</h:panelGrid>
<br />
<h:commandButton value="Submit" tabindex="1">
<f:ajax listener="#{terminalImportBean.addTerminal}"
execute="@form" render="@form"/>
</h:commandButton>
</h:form>
这可以正常使用普通帖子。 但是,如果我为“输入1”(id =“i1”)添加f:ajax验证并尝试使用以下内容重新渲染复合(ii1):
...
<h:outputLabel styleClass="sectionBody" for="i1"
value="Input 1"></h:outputLabel>
<my:input id="ii1">
<h:inputText id="i1" size="20" maxlength="20" tabindex="0"
required="true" label="Input 1">
<f:ajax listener="#{myBean.checkInput1}"
event="blur"
render="ii1"/>
</h:inputText>
</my:input>
...
我在ajax响应处理期间在浏览器中生成错误:
malformedXML: During update: f1:ii1 not found
我尝试使用f1:ii1
,:f1:ii1
,但无济于事。当我使用msgId
或:f1:ii1:msgId
时,它有效。有人知道为什么吗?
我正在使用Mojarra 2.1.3
由于
答案 0 :(得分:3)
那是因为复合组件通过本身不会向HTML呈现任何内容。只有它的子项被渲染为HTML。生成的HTML输出中不存在任何ID为f1:i11
的HTML元素。在浏览器中打开页面,右键单击查看源。自己看看那里。这个ID没有这样的元素。 JavaScript / Ajax遇到了完全相同的问题。它无法找到元素以便更新它。
要解决此问题,您需要将整个复合体包装在HTML <div>
或<span>
中,并为其指定#{cc.clientId}
ID,基本上会打印UIComponent#getClientId()
<cc:implementation>
<div id="#{cc.clientId}">
...
</div>
</cc:implementation>
然后你可以参考如下:
<my:input id="foo" />
...
<f:ajax ... render="foo" />
您甚至可以引用特定的复合组件子项。
<f:ajax ... render="foo:inputId" />