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


99

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

Моя проблема: как передать идентификатор строки действию, запускаемому ссылкой на команду, чтобы я знал, с какой строкой действовать? Я пробовал использовать <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недоступна при рендеринге атрибута (она работает, когда я использую фиксированное значение).

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

Ответы:


215

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

Есть несколько способов выполнить это требование.

  1. Если ваш servletcontainer поддерживает как минимум Servlet 3.0 / EL 2.2, то просто передайте его как аргумент метода действия / слушателя UICommand компонента или AjaxBehaviorтега. Например

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

    В комбинации с:

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

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

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

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

    с участием:

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

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


  2. Использование <f:param>в UICommandкомпоненте. Он добавляет параметр запроса.

     <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>в целевом представлении, см. Также f: viewParam vs @ManagedProperty :

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

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


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

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

    В комбинации с

     private Long id; // +setter

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


  4. Вместо этого привяжите значение 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и ленивое создание экземпляра в получателе является обязательным, если вы используете это в bean-компоненте представления или сеанса, поскольку DataModelне реализует Serializable)

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

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

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


  5. Используйте Application#evaluateExpressionGet()для программной оценки тока #{item}.

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

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


1
+1, хотя я предпочитаю №2 (если нужно поддерживать 2.5).
Bozho

Спасибо за исчерпывающий ответ. К сожалению, я должен сообщить, что №1 было единственной вещью, которая работала в отфильтрованной таблице данных с примерами (это как раз тот сценарий, для которого мне это нужно). Все остальные работали только на нефильтрованном столе. Я считаю, что это больше ошибка в праймфейсах, чем в вашем ответе.
Майкл Боргвардт

Ограничен ли запрос или представление bean-компонента?
BalusC

2
Под "фильтром" вы имеете в виду, как в этом примере витрины ? Симптомы указывают на то, что действие фильтра выполняется только на стороне клиента и что модель на стороне сервера не поддерживается. Не уверен, намеренно ли это. Вы всегда можете оставить отчет о проблеме.
BalusC

Ваш пост находится между самыми полезными постами, которые я когда-либо читал. Я использовал метод 5, потому что я вынужден использовать сервлет 2.5. Теперь у меня вопрос: можно ли отправить параметр с помощью commandLink (как в вашем примере), но с использованием ajax?
Aditzu 05

11

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


6
Эта функция характерна не только для JSF 2.0 (который сам по себе может работать в контейнерах Servlet 2.5), но для EL 2.2 (который является частью Servlet 3.0).
BalusC

11

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

<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>

поддерживающий боб

 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;
        }
    }
}

-1

Благодаря этому сайту 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.


1
У вас другая проблема, чем здесь. Вы хотите сохранить параметры запроса из #{param}карты для последующих запросов, а не передавать произвольный параметр. Ваши вопросы и ответы описаны в stackoverflow.com/questions/17734230
BalusC
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.