Подтвердить что ты не робот

Как отобразить индекс строки в JSF datatable

В JSF dataTable я хочу отобразить индекс строки рядом с строками... например:

Column A    Column B
1           xxx
2           yyy

Я думал, что могу использовать неявную переменную el, например # {rowIndex}, но это не работает.

Я нашел решение, чтобы создать привязку для таблицы данных и использовать привязку как:

<h:dataTable var="item" value="#{controller.items}" binding="#{controller.dataTable}">
    <h:column>#{controller.dataTable.rowIndex}</h:column>
    <h:column>value</h:column>
</h:dataTable>

но это решение является сложным и не работает, когда у меня много вложенных данных в таблице.

Любые идеи о том, как лучше решить эту проблему?

4b9b3361

Ответ 1

Существующее решение не кажется мне плохим. СтрокаIndex должна работать во вложенных таблицах, пока вы ссылаетесь на модель вложенной таблицы.

    <h:dataTable border="1" value="#{nestedDataModel}" var="nested">
        <h:column>
            <h:dataTable border="1" value="#{nested}" var="item">
                <h:column>
                    <h:outputText value="#{nested.rowIndex}" />
                </h:column>
                <h:column>
                    <h:outputText value="#{item}" />
                </h:column>
            </h:dataTable>
        </h:column>
    </h:dataTable>

Модель образца:

public class NestedDataModel extends DataModel implements Serializable {

    private List<List<String>> nestedDataModel = populateModel();
    private int index;

    private List<List<String>> populateModel() {
        List<List<String>> list = new ArrayList<List<String>>();
        for(int x=0; x<3; x++) {
            List<String> nestedTableData = new ArrayList<String>();
            for(int y=0; y<3; y++) {
                nestedTableData.add("Foo x="+x+" y="+y);
            }
            list.add(nestedTableData);
        }
        return list;
    }

    @Override
    public int getRowCount() {
        return nestedDataModel.size();
    }

    @Override
    public Object getRowData() {
        List<String> list = nestedDataModel.get(index);
        return new ListDataModel(list);
    }

    @Override
    public int getRowIndex() {
        return index;
    }

    @Override
    public Object getWrappedData() {
        return nestedDataModel;
    }

    @Override
    public boolean isRowAvailable() {
        return index >= 0 && index < nestedDataModel.size();
    }

    @Override
    public void setRowIndex(int arg0) {
        index = arg0;
    }

    @Override
    public void setWrappedData(Object arg0) {
        throw new UnsupportedOperationException();
    }

}

Вложения таблиц в большинстве случаев следует избегать - если вы не будете осторожны (например, сделайте их детьми формы), это может привести к прохождению O (N ^ 2) над детьми таблицы для каждой фазы жизненного цикла на submit (и на жизненном цикле 6 фаз).


Для того, что является внешним по отношению к модели, вы можете использовать простой счетчик в управляемом bean:

public class RowCounter implements Serializable {

    private transient int row = 0;

    public int getRow() {
        return ++row;
    }

}

Config:

<managed-bean>
    <managed-bean-name>rowCounter</managed-bean-name>
    <managed-bean-class>tablerows.RowCounter</managed-bean-class>
    <managed-bean-scope>request</managed-bean-scope>
</managed-bean>

Вид:

<f:view>
    <h:dataTable border="1" value="#{tableDataBean.tableDataModel}"
        var="rowBean">
        <h:column>
            <h:outputText value="#{rowCounter.row}" />
        </h:column>
        <h:column>
            <h:outputText value="#{rowBean}" />
        </h:column>
    </h:dataTable>
</f:view>

Это работает, потому что bean является областью запроса и привязан к элементу управления только для чтения вне формы. Он не будет работать во вложенном dataTable, если вы не хотите, чтобы счетчик строк был глобальным для представления. Но тогда я не уверен, что индекс строки должен быть функцией представления.

Для вложенного dataTable вам будет лучше предоставить индекс строки из строки bean. Это дает вам больше контроля, если вы решите делать такие вещи, как разбиение на страницы над наборами данных.

Ответ 2

Это решение было опубликовано Джейми Уильямсом на CodeRanch. Он говорит, что это работает с Томагавком. Я использую perffaces, который также поддерживает его.

<t:dataTable rowIndexVar="row" value="#{someBean.value}">  
    <h:column>  
        <h:outputText value="#{row + 1}"/>  
    </h:column>  
</t:dataTable>

Ответ 3

В RichFaces есть решение, похожее на Brent's:

<rich:dataTable value="#{backingBean.list}" var="v" rowKeyVar="index">
    <rich:column>
        <f:facet name="header">Index</f:facet>
        <h:outputText value="#{index + 1}" />
    </rich:column>
    <rich:column>
        <f:facet name="header">Name</f:facet>
        <h:outputText value="#{v.name}" />
    </rich:column>
</rich:dataTable>

Ответ 4

Это сработало.

<h:dataTable var="item" value="#{controller.items}">
    <h:column>#{controller.items.indexOf(item)}</h:column>
    <h:column>value</h:column>
</h:dataTable>

Ответ 5

Я просто зависеть от списка и позиции var в этом списке:

<h:column>
          <f:facet name="header">#</f:facet>
          #{bean.listValue.indexOf(varObject)+1}
</h:column>