Как передать выбранную строку в commandLink внутри dataTable или ui: repeat?

Я использую Primefaces в приложении JSF 2. У меня есть <p:dataTable>, и вместо того, чтобы выбирать строки, я хочу, чтобы пользователь мог выполнять различные действия по отдельным строкам. Для этого у меня есть несколько <p:commandLink> в последнем столбце.

Моя проблема: как передать идентификатор строки в действие, запущенное по ссылке команды, чтобы я знал, к какой строке действовать? Я попытался использовать <f:attribute>:

<p:dataTable value="#{bean.items}" var="item">
    ...
    <p:column>
        <p:commandLink actionListener="#{bean.insert}" value="insert">
            <f:attribute name="id" value="#{item.id}" />
        </p:commandLink>
    </p:column>
</p:dataTable>

Но он всегда дает 0 - видимо, переменная строки f недоступна, когда атрибут визуализируется (он работает, когда я использую фиксированное значение).

У кого-то есть альтернативное решение?

Ответ 1

Что касается причины, то <f:attribute> относится к самому компоненту (заполняется во время создания времени просмотра), а не к итерированной строке (заполняется во время отображения времени просмотра).

Существует несколько способов достижения этого требования.

  • Вместо этого используйте <f:param>. Он добавляет параметр запроса.

    <h:commandLink action="#{bean.insert}" value="insert">
        <f:param name="id" value="#{item.id}" />
    </h:commandLink>
    

    Если ваш bean - это область запроса, пусть JSF установил ее @ManagedProperty

    @ManagedProperty(value="#{param.id}")
    private Long id; // +setter
    

    Или, если ваш bean имеет более широкую область видимости или требуется более мелкая проверка/преобразование, используйте <f:viewParam> в целевом view, см. также f: viewParam vs @ManagedProperty:

    <f:viewParam name="id" value="#{bean.id}" required="true" />
    

    В любом случае это имеет то преимущество, что datamodel не обязательно нужно сохранять для отправки формы (для случая, когда ваш bean является областью действия запроса).


  • Вместо этого используйте <f:setPropertyActionListener>. Преимущество заключается в том, что это устраняет необходимость доступа к карте параметров запроса, когда bean имеет более широкую область действия, чем область запроса.

    <h:commandLink action="#{bean.insert}" value="insert">
        <f:setPropertyActionListener target="#{bean.id}" value="#{item.id}" />
    </h:commandLink>
    

    В сочетании с

    private Long id; // +setter
    

    Он будет доступен только с помощью свойства id в методе действий. Это требует только того, что датамодель сохраняется для запроса на отправку формы. Лучше всего поставить bean в область видимости на @ViewScoped.


  • Если ваш servletcontainer поддерживает Servlet 3.0/EL 2.2, просто передайте его как аргумент метода. Это также требует, чтобы датамодель сохранялась для запроса на отправку формы. Лучше всего поставить bean в область видимости на @ViewScoped.

    <h:commandLink action="#{bean.insert(item.id)}" value="insert" />
    

    В сочетании с:

    public void insert(Long id) {
        // ...
    }
    

    Вы даже можете передать весь объект объекта:

    <h:commandLink action="#{bean.insert(item)}" value="insert" />
    

    с:

    public void insert(Item item) {
        // ...
    }
    

    В контейнерах Servlet 2.5 это также возможно, если вы обеспечиваете реализацию EL, которая поддерживает это, например JBoss EL. Подробную информацию о конфигурации см. В этом ответе.


  • Привяжите значение datatable к DataModel<E>, которое, в свою очередь, обертывает элементы.

    <h:dataTable value="#{bean.model}" var="item">
    

    с

    private transient DataModel<Item> model;
    
    public DataModel<Item> getModel() {
        if (model == null) {
            model = new ListDataModel<Item>(items);
        }
        return model;
    }
    

    (делает его transient и лениво инстанцирует его в getter, является обязательным, когда вы используете его в представлении или сеансе с областью bean, поскольку DataModel не реализует Serializable)

    Затем вы сможете получить доступ к текущей строке DataModel#getRowData(), не передавая ничего вокруг (JSF определяет строку на основе запроса имя параметра щелкнутой командной строки/кнопки).

    public void insert() {
        Item item = model.getRowData();
        Long id = item.getId();
        // ...
    }
    

    Это также требует, чтобы датамодель сохранялась для запроса на отправку формы. Лучше всего поставить bean в область видимости на @ViewScoped.


  • Вы можете использовать Application#evaluateExpressionGet() для программной оценки текущего #{item}.

    public void insert() {
        FacesContext context = FacesContext.getCurrentInstance();
        Item item = context.getApplication().evaluateExpressionGet(context, "#{item}", Item.class);
        Long id = item.getId();
        // ...
    }
    

Какой способ выбрать, зависит от функциональных требований и того, обладает ли тот или иной преимущества для других целей. Я лично продолжу # 3 или, если вы хотите также поддерживать контейнеры с сервлетом 2.5, с № 2.

Ответ 2

В JSF 1.2 это было сделано <f:setPropertyActionListener> (внутри командного компонента). В JSF 2.0 (EL 2.2, если быть точным, благодаря BalusC), можно сделать это следующим образом: action="${filterList.insert(f.id)}

Ответ 3

На моей странице просмотра:

<p:dataTable  ...>
<p:column>
<p:commandLink actionListener="#{inquirySOController.viewDetail}" 
               process="@this" update=":mainform:dialog_content"
           oncomplete="dlg2.show()">
    <h:graphicImage library="images" name="view.png"/>
    <f:param name="trxNo" value="#{item.map['trxNo']}"/>
</p:commandLink>
</p:column>
</p:dataTable>

поддержка bean

 public void viewDetail(ActionEvent e) {

    String trxNo = getFacesContext().getRequestParameterMap().get("trxNo");

    for (DTO item : list) {
        if (item.get("trxNo").toString().equals(trxNo)) {
            System.out.println(trxNo);
            setSelectedItem(item);
            break;
        }
    }
}

Ответ 4

Благодаря этот сайт Mkyong, единственным решением, которое на самом деле работало для нас, чтобы передать параметр, было

<h:commandLink action="#{user.editAction}">
    <f:param name="myId" value="#{param.id}" />
</h:commandLink>

с

public String editAction() {

  Map<String,String> params = 
            FacesContext.getExternalContext().getRequestParameterMap();
  String idString = params.get("myId");
  long id = Long.parseLong(idString);
  ...
}

Технически, что вы не можете перейти непосредственно к самому методу, но к JSF request parameter map.