Как получить выбранный индекс строки в JSF datatable?

У меня есть databale на index.xhtml

<h:dataTable style="border: solid 2px black;"
    value="#{IndexBean.bookList}" var="item"
    binding="#{IndexBean.datatableBooks}">

    <h:column>
        <h:commandButton value="Edit" actionListener="#{IndexBean.editBook}">
            <f:param name="index" value="#{IndexBean.datatableBooks.rowIndex}"/>
        </h:commandButton>
    </h:column>
</h:dataTable>

My bean:

@ManagedBean(name="IndexBean")
@ViewScoped
public class IndexBean implements Serializable {
    private HtmlDataTable datatableBooks;

    public HtmlDataTable getDatatableBooks() {
        return datatableBooks;
    }

    public void setDatatableBooks(HtmlDataTable datatableBooks) {
        this.datatableBooks = datatableBooks;
    }

    public void editBook() throws IOException{
        int index = Integer.parseInt(FacesContext.getCurrentInstance().getExternalContext().getRequestParameterMap().get("index").toString());
        System.out.println(index);
    }
}

Моя проблема в том, что я всегда получаю один и тот же индекс в журнале сервера, даже если я нажимаю кнопки редактирования. Представьте себе, что имеется одна коллекция, которая поставляется в виде данных. Я не показал, что в bean.

Если я изменю область видимости от ViewScope до RequestScope, она отлично работает. В чем проблема с @ViewScoped? Спасибо заранее:)

EDIT:

<h:column>
    <h:commandButton value="Edit" actionListener="#{IndexBean.editBook}" />
</h:column>

public void editBook(ActionEvent ev) throws IOException{
    if (ev.getSource() != null && ev.getSource() instanceof HtmlDataTable) {
        HtmlDataTable objHtmlDataTable = (HtmlDataTable) ev.getSource();
        System.out.println(objHtmlDataTable.getRowIndex());
    }
}

Ответ 1

Вы уже привязали компонент <h:dataTable> к bean. Все, что вам нужно сделать, это:

public void editBook() throws IOException{
    int index = datatableBooks.getRowIndex(); // Actually not interesting info.
    Book book = (Book) datatableBooks.getRowData(); // This is what you want.
}

Здесь также не нужен <f:param>. Дополнительные советы также см. В в этой статье.

Обновить. Я могу воспроизвести вашу проблему. Вероятно, это ошибка с @ViewScoped. Если для параметра bean установлено значение @RequestScoped, он работает так, как ожидалось. Также, когда вы удаляете привязку компонента и получаете компонент из режима просмотра самостоятельно, он работает так, как ожидалось. Я зарегистрировал вопрос 1658 об этом.

Ответ 2

Что вы можете сделать, так это использовать метод [getRowData()][1] на Java bean, чтобы напрямую получить объект, расположенный в строке, на которой находится кнопка, на которую пользователь нажал.

Пример кода:

public void editBook(ActionEvent evt) {
    // We get the table object
    HtmlDataTable table = getParentDatatable((UIComponent) evt.getSource());
    // We get the object on the selected line.
    Object o = table.getRowData();
    // Eventually, if you need the index of the line, simply do:
    int index = table.getRowIndex();
    // ...
}

// Method to get the HtmlDataTable.
private HtmlDataTable getParentDatatable(UIComponent compo) {
    if (compo == null) {
        return null;
    }
    if (compo instanceof HtmlDataTable) {
        return (HtmlDataTable) compo;
    }
    return getParentDataTable(compo.getParent());
}

<ч/" > Редактировать

Теперь код JSF выглядит следующим образом:

<h:commandButton value="Edit" actionListener="#{IndexBean.editBook}"/>

Кроме того, не забудьте изменить подпись метода editBook(), установив аргумент javax.faces.event.ActionEvent.

Ответ 3

Если вы используете EL 2.2, например, с Tomcat7, вы можете попробовать

<h:commandLink action="#{IndexBean.editBook(item)}" immediate="true">

Я надеюсь помочь