EDIT1 :
正如Valentin Jacquemin所建议的,我用List<Entry<String,String>>
实现了它
但是,当我现在更改一个值时,tomcat会抛出一个错误:
错误讯息:
SEVERE: /bvDesktop_RuleOverviewAddActionNode.xhtml at line 111 and column 115 value="#{paramListKVs.value}": Property 'value' not writable on type java.lang.Object
javax.el.PropertyNotFoundException: /bvDesktop_RuleOverviewAddActionNode.xhtml at line 111 and column 115 value="#{paramListKVs.value}": Property 'value' not writable on type java.lang.Object
我的xhtml中的相应行:
<p:dataTable id="paramListKV" var="paramListKVs" value="#{ruleTreeBeanAddActionNode.paramListKV}" editable="true" editMode="cell">
<p:column>
<f:facet name="header">List Key</f:facet>
<h:outputText value="#{paramListKVs.key}"/>
</p:column>
<p:column>
<p:cellEditor id="paramListKVsEditTest">
<f:facet name="header">List Value</f:facet>
<f:facet name="output"><h:outputText value="#{paramListKVs.value}"/></f:facet>
--><f:facet name="input"><p:inputText value="#{paramListKVs.value}" style="width:96%" /></f:facet><--
</p:cellEditor>
</p:column>
</p:dataTable>
标记为 - &gt;和&lt; - 。
我不知道java试图编写代码。我甚至不确定value="#{paramListKVs.value}"
是否正确。有什么想法吗?
我有一个特殊的问题我能够解决,但我觉得我的解决方案更像是一种肮脏的方式。我希望有更优雅的方式来做到这一点,但我找不到一个(搜索如此)。所以这是我的问题:
我目前正在创建一个基于规则的框架来自动处理某些文件,我想让编辑们在一个漂亮而简单的Web界面中创建这些规则。
我已经拥有所需的网站,数据模型等。然而,一个特殊问题是,行动规则可以有Set of Properties Key-Value
明智的。我选择TreeMap
。现在我想能够add
,delete
和edit
代表数据表中的单元格。
爪哇: createActioNode:
@ManagedBean
public class RuleTreeBeanAddActionNode {
...
private TreeMap<String,String> actionParamMap = new TreeMap<String,String>();
private String paramKeyToAdd = "";
private String paramValueToAdd = "";
private int paramKeyIndex;
private String valueToChange;
private String keyForInfo;
private UIComponent datatable;
public void addKeyValueToMap() {
if((paramKeyToAdd != null || !paramKeyToAdd.equals("")) && (paramValueToAdd != null) || !paramValueToAdd.equals("")) {
unusedActionParamKeys.remove(paramKeyToAdd);
usedActionParamKeys.add(paramKeyToAdd);
actionParamsAsListKeys.add(paramKeyToAdd);
actionParamsAsListValues.add(paramValueToAdd);
actionParamMap.put(paramKeyToAdd, paramValueToAdd);
actionParamMapEntries = new ArrayList<Entry<Integer, String>>((Collection<? extends Entry<Integer, String>>) actionParamMap.entrySet());
userSession.setAttribute("actionParamMap", actionParamMap);
userSession.setAttribute("unusedActionParamKeys", unusedActionParamKeys);
userSession.setAttribute("usedActionParamKeys", usedActionParamKeys);
userSession.setAttribute("tempSelectorWorkOn", "");
userSession.setAttribute("tempRuleId", "");
userSession.setAttribute("actionParamsAsListKeys", actionParamsAsListKeys);
userSession.setAttribute("actionParamsAsListValues", actionParamsAsListValues);
paramKeyToAdd = "";
paramValueToAdd = "";
}
}
public void setValueToChange(String valueToChange) {
DataTable dt = (DataTable)datatable;
String id = String.valueOf(paramKeyIndex);
Entry<String, String> temp = (Entry<String, String>)dt.getRowData();
actionParamMap.put((String) temp.getKey(), valueToChange);
}
//getter and setter
createActionNode.xhtml
<tr>
<td>Action Parameter </td>
<td>
<p:dataTable binding="#{ruleTreeBeanAddActionNode.datatable}"
rowKey="test"
rowIndexVar="paramKeyIndex"
id="paramKeysValues"
var="params"
value="#{ruleTreeBeanAddActionNode.actionParamMapEntries}"
editable="true"
editMode="cell">
<p:column id="keys">
<f:facet name="header">Param Key</f:facet>
<h:outputText value="#{params.key}"/>
</p:column>
<p:column id="values">
<f:facet name="header">Param Value</f:facet>
<p:cellEditor id="paramValueEdit">
<f:facet name="output"><h:outputText value="#{params.value}"/></f:facet>
<f:facet name="input"><p:inputText id="evinput" value="#{ruleTreeBeanAddActionNode.valueToChange}" style="width:96%"/></f:facet>
</p:cellEditor>
</p:column>
</p:dataTable>
</td>
</tr>
<tr><td><br/></td></tr>
<tr>
<td>Param To Value</td>
<td>
<table>
<tr>
<td>
<p:selectOneMenu id="paramKeys" value="#{ruleTreeBeanAddActionNode.paramKeyToAdd}">
<f:selectItems value="#{ruleTreeBeanAddActionNode.unusedActionParamKeys}"/>
</p:selectOneMenu>
</td>
<td>
<p:inputText id="paramValue" value="#{ruleTreeBeanAddActionNode.paramValueToAdd}"/>
</td>
</tr>
<tr>
<td></td>
<td>
<p:commandButton id="addParam" value="+" action="#{ruleTreeBeanAddActionNode.addKeyValueToMap}" update="paramKeysValues,paramKeys,paramValue"/>
</td>
</tr>
</table>
</td>
</tr>
我明白我不需要返回地图而是返回条目集,以便数据表可以迭代通过它。所以当我在单元格中编辑记录时,功能
变更会调用#{ruleTreeBeanAddActionNode.valueToChange}
。但是,要检索实际记录,从而检索地图中要更改的键和值,我必须检查UIComponent
的{{1}},检索已编辑的记录,保存地图并重建条目集。
所以我的实际问题是,这是推荐的方法吗?我是否监督过让我的世界变得更容易的事情?有其他人实现了这样的事情吗?
答案 0 :(得分:1)
以下是UIData的文档,其中任何数据表都是以下内容:
UIData是一个支持数据绑定到的集合的UIComponent 由DataModel实例表示的数据对象,它是当前的 该组件本身的价值(通常通过 ValueExpression)。在对数据行进行迭代处理期间 在数据模型中,当前行的对象作为请求公开 var属性指定的键下的属性。
因此,正如您所说,您需要一个可迭代对象,以便数据表可以遍历它并在构建表时公开每个元素。
另一方面,你没有完全利用绑定功能。只需让组件完成其工作,并将value=#{ruleTreeBeanAddActionNode.actionParams}
,actionParams
设置为List
或任何其他supported type。完成后,您的数据表的模型将引用您的ruleTreeBeanAddActionNode.actionParams
元素。您不需要任何其他机制来进行同步。
另见:
更新
我猜您尝试使用AbstractMap.SimpleEntry作为列表的条目。尽管setValue
不是void
类型,但此类与JavaBean不兼容。创建自己的对象,保存行的状态,以便您拥有JavaBean兼容对象,EL将能够正确解析读/写方法:
public class Entry{
private String key;
private String value;
public Entry(String key, String value) {
this.key = key;
this.value = value;
}
public String getKey() {return key;}
public String getValue() {return value;}
public void setKey(String key) {this.key = key;}
public void setValue(String value) {this.value = value;}
}
然后在你的支持bean中:
@ManagedBean
public class RuleTreeBeanAddActionNode {
private List<Entry> entries;
@PostConstruct
public void init() {
entries = new ArrayList<Entry>();
entries.add(new Entry("foo", "bar"));
}
public List<Entry> getEntries() { return entries; }
}