1

我对wicket 中 Dynamic <optgroup> 支持问题的回答包括一个选择框,该框使用 WebMarkupContainer 和 SelectOptions 组件进行样式设置。

我实际上是在尝试在扩展 Form 的类中使用它,并在超类构造函数中使用具有“选择”属性(以及其他)的 bean 作为 CompoundPropertyModel。MyBean 类型的 Bean 可以属于一个由 Group2Bean 表示的内部组,而其中的每一个又可以属于一个由 Group1Bean 表示的外部组。

代码(修改为隐藏机密和数据库检索)或多或少如下:

选择的 HTML 代码:

<select wicket:id="select">
    <wicket:container wicket:id="outerRV">
        <optgroup wicket:id="optGroup1">
          <wicket:container wicket:id="rv">
            <optgroup wicket:id="optGroup2">
              <wicket:container wicket:id="selectOptions">
                <option wicket:id="option"></option>
              </wicket:container>
            </optgroup>
          </wicket:container>
        </optgroup>
    </wicket:container>
</select>

整个表单的 Java 代码:

public class MyForm extends Form<FilterBean> {

public MyForm(String id, FilterBean filterBean) {

super(id, new CompoundPropertyModel<FilterBean>(filterBean);

/*
 * ...code to get group1Beans and group2Beans...
 */

/* Select */
Select select = new Select("select");
select.setRequired(true);
add(select);

/* markup repeater for each group1 division */
RepeatingView rvOuter = new RepeatingView("outerRV");
select.add(rvOuter);

for(Group1Bean group1Bean : group1Beans){ /* for each group1 division */

    /* container with unique wicket ID */
    WebMarkupContainer overOptgroup1 = new WebMarkupContainer(rv.newChildId());
    rv.add(overOptgroup1);

    /* outer optgroup, name taken from group1Bean */
    WebMarkupContainer optGroup1 = new WebMarkupContainer("optGroup1");
    overOptGroup.add(optGroup1);
    optGroup1.add(
        new AttributeModifier("label",true,
                              new Model<String>(group1Bean.getName()
        ))
    );

    /* markup repeater for each group2 division */
    RepeatingView rv = new RepeatingView("rv");
    select.add(rv);

    for(Group2Bean group2bean : group2Beans){ /* for each group2 division */

      /* container with unique wicket ID */
      WebMarkupContainer overOptgroup2 =
              new WebMarkupContainer(rv.newChildId());
      rv.add(overOptgroup2);

      /* inner optgroup, name taken from group2Bean and indented */
      WebMarkupContainer optGroup2 = new WebMarkupContainer("optGroup2");
      overOptGroup.add(optGroup2);

      optGroup2.add(new AttributeModifier("style",true,
          new Model<String>("padding-left:15px")));

      optGroup2.add(new AttributeModifier("label",true,
          new Model<String>(group2Bean.getName())));

      /* fetches and displays MyBean options for selected group2 division */
      optGroup2.add(
          new SelectOptions<MyBean>(
              "selectOptions",fetchMyBeansUnder(group2Bean)
              new MyBeanRenderer()
          ).add(new AttributeModifier("style",true,
                   new Model<String>("padding-left="30px")))
      );
    }
}

问题是,模型的 select 属性没有更新。按下(默认)提交按钮会给我一个“需要选择”反馈,即使我在生成的 webapp 界面上更改了我的选择,并尝试将此组件链接到进一步失败的组件,但无法检索到 select 属性的值。

尝试从具有通用 Select 的 1.4.6 更新到 1.5.6,但问题并没有得到解决。当Select 的 inputConverter() (在按下提交按钮后调用)在 option 上调用 getDefaultModelObject() 时,扩展和摆弄 Select 导致我将问题缩小到返回 null 的所选 SelectOption

有什么想法吗,伙计们?提前谢谢了。

4

1 回答 1

1

好吧,我发现了,一个愚蠢的错误。使用 SelectOption 的构造函数,模型不会为每个选项自动创建,而是由 MyBeanRenderer 的 getModel() 方法创建;我返回了一个空模型,而不是从 bean 参数创建的模型。我什至不认为它可能是渲染器......只是将一种方法更改为以下方法:

public IModel<T> getModel(T bean){
    return new Model<T>(bean);
}

仍然不敢相信我在这上面花了几个星期却没有看到。

于 2012-05-10T11:24:08.313 回答