2017-02-16 22 views
0

Я разрабатываю приложение для управления клиентами своими машинами. Я создал все необходимые таблицы, схемы, контроллеры и т. Д.Spring MVC + Thymeleaf - сохранение отношения @ManyToOne

У клиента есть отношение List и Machine к клиенту (bidiretional, non-optional).

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

Так вот некоторый короткий надрез коды:

@Controller 
public class MachineController { 

... 
    @GetMapping("/machines/add/{clientId}") 
    public String addMachine(@PathVariable("clientId") int clientId, Model model) throws ClientNotFoundException { 
     model.addAttribute("machineTypes", MachineType.values()); 
     model.addAttribute("machine", new Machine()); 
     model.addAttribute("client", clientService.find(clientId)); 
     return "machines/form"; 
    } 
} 

    @PostMapping("/machines/save") 
    public String saveMachine(@ModelAttribute @Valid Machine machine, BindingResult bindingResult, Model model) 
      throws ClientNotFoundException { 

     model.addAttribute("machineTypes", MachineType.values()); 

     int clientId = machine.getClient().getId(); 
     LOG.debug("ClientId:{}", clientId); 
     // Client object is not filled here ! clientId is 0 (new client). 
} 

Проблема с функцией сохранения - я не знаю, как передать exisitng объекта клиента для Machine объекта, который отправляется по HTTP POST. Мой контроллер жалуется, что клиент не отправлять и BindingResult бросает ошибку:

Field error in object 'machine' on field 'client.address.city: rejected value [null]; Field error in object 'machine' on field 'client.address.zipCode: rejected value [null]; Field error in object 'machine' on field 'client.name': rejected value [null];

Я loooking вперед любой помощи.

HTML форма представлена ​​ниже:

       <form class="form-horizontal" th:action="@{/machines/save}" th:object="${machine}" method="post" id="machineForm"> 
           <input type="hidden" th:field="*{id}"/> 

            <!-- Panel for machine --> 
           <div th:class="${#fields.hasErrors('machineType')}? 'form-group has-error has-feedback' : 'form-group'"> 
            <label class="control-label col-sm-4" for="manufacturer">Rodzaj:*</label> 
            <div class="col-sm-8"> 

             <select th:field="${machine.machineType}" class="form-control" id="machineTypeSelect"> 
              <option value="" disabled="disabled" selected="selected">Wybierz rodzaj</option> 
              <option th:each="type: ${machineTypes}" th:value="${type.name()}" th:text="${type}" th:attr="data-has-car=${type.hasCar()}"></option> 
             </select> 

             <div class="help-block" th:if="${#fields.hasErrors('machineType')}" 
              th:errors="*{machineType}"></div> 
            </div> 
           </div> 

           <div th:class="${#fields.hasErrors('manufacturer')}? 'form-group has-error has-feedback' : 'form-group'"> 
            <label class="control-label col-sm-4" for="manufacturer">Producent:*</label> 
            <div class="col-sm-8"> 
             <input type="text" 
               class="form-control" 
               id="manufacturer" 
               placeholder="Podaj producenta" 
               th:field="*{manufacturer}" /> 
             <div class="help-block" th:if="${#fields.hasErrors('manufacturer')}" 
              th:errors="*{manufacturer}"></div> 
            </div> 
           </div> 

           <div th:class="${#fields.hasErrors('model')}? 'form-group has-error has-feedback' : 'form-group'"> 
            <label class="control-label col-sm-4" for="model">Model:</label> 
            <div class="col-sm-8"> 
             <input type="text" 
               class="form-control" 
               id="model" 
               placeholder="Podaj model" 
               th:field="*{model}"/> 
             <div class="help-block" th:if="${#fields.hasErrors('model')}" 
              th:errors="*{model}"></div> 
            </div> 
           </div> 

           <div th:class="${#fields.hasErrors('productionYear')}? 'form-group has-error has-feedback' : 'form-group'"> 
            <label class="control-label col-sm-4" for="productionYear">Rok produkcji:*</label> 
            <div class="col-sm-8"> 
             <input type="number" 
               class="form-control" 
               id="productionYear" 
               placeholder="Podaj rok produkcji" 
               th:field="*{productionYear}"/> 
             <div class="help-block" th:if="${#fields.hasErrors('productionYear')}" 
              th:errors="*{productionYear}"></div> 
            </div> 
           </div> 

           <div th:class="${#fields.hasErrors('factoryNo')}? 'form-group has-error has-feedback' : 'form-group'"> 
            <label class="control-label col-sm-4" for="factoryNo">Numer fabryczny:</label> 
            <div class="col-sm-8"> 
              <input type="number" 
                class="form-control" 
                id="factoryNo" 
                placeholder="Podaj numer fabryczny" 
                th:field="*{factoryNo}"/> 
             <div class="help-block" th:if="${#fields.hasErrors('factoryNo')}" 
              th:errors="*{factoryNo}"></div> 
            </div> 
           </div> 

           <div th:class="${#fields.hasErrors('maxLoad')}? 'form-group has-error has-feedback' : 'form-group'"> 
            <label class="control-label col-sm-4" for="maxLoad">Max udżwig:</label> 
            <div class="col-sm-8"> 
             <div class="input-group"> 
             <input type="number" 
               class="form-control" 
               id="maxLoad" 
               placeholder="Max. udźwig" 
               aria-describedby="measure" 
               th:field="*{maxLoad}"/> 
              <span class="input-group-addon" id="measure">kg</span> 
             </div> 
             <div class="help-block" th:if="${#fields.hasErrors('maxLoad')}" 
              th:errors="*{maxLoad}"></div> 
            </div> 
           </div> 

           <div th:object="${machine.client}"> 
            <div th:class="${#fields.hasErrors('id')}? 'form-group has-error has-feedback' : 'form-group'"> 
             <label class="control-label col-sm-4" for="clientId">Wybrany klient:</label> 
             <div class="col-sm-8"> 
              <span class="form-control-static" id="selectedClient">Nie wybrano! Wyszukaj po prawej</span> 
              <input type="hidden" th:field="${machine.client.id}" id="clientId" /> 
              <div class="help-block" th:if="${#fields.hasErrors('id')}" 
               th:errors="${machine.client.id}"></div> 
             </div> 
            </div> 
           </div> 

           <div id="machineCar" th:object="${machine.car}"> 

            <div th:class="${#fields.hasErrors('make')}? 'form-group has-error has-feedback' : 'form-group'"> 
             <label class="control-label col-sm-4" for="carMake">Marka pojazdu:*</label> 
             <div class="col-sm-8"> 
              <input type="text" 
                class="form-control" 
                id="carMake" 
                placeholder="Podaj markę pojazdu" 
                th:field="*{make}"/> 
              <div class="help-block" th:if="${#fields.hasErrors('make')}" 
               th:errors="*{make}"></div> 
             </div> 
            </div> 

            <div th:class="${#fields.hasErrors('vin')}? 'form-group has-error has-feedback' : 'form-group'"> 
             <label class="control-label col-sm-4" for="factoryNo">VIN:</label> 
             <div class="col-sm-8"> 
              <input type="number" 
                class="form-control" 
                id="vin" 
                placeholder="Podaj numer VIN" 
                th:field="*{vin}"/> 
              <div class="help-block" th:if="${#fields.hasErrors('vin')}" 
               th:errors="*{vin}"></div> 
             </div> 
            </div> 
           </div> 
            <div class="form-group"> 
             <div class="col-xs-12"> 
              <button type="submit" class="btn btn-primary">Zapisz dane</button> 
             </div> 
            </div> 
           </form> 

ответ

0

Я решил свою проблему. Не уверен, что это правильное решение, но работает.

Таким образом, просто в машиностроении лица:

public class Machine { 

    @Id 
    @GeneratedValue(strategy = GenerationType.SEQUENCE, generator = "MACHINE_SEQUENCE") 
    private int id; 

    // ... 
    @ManyToOne(optional = false, fetch = FetchType.LAZY, cascade = { CascadeType.PERSIST, CascadeType.MERGE}) 
    @JoinColumn(name = "client_id", nullable = false) 
    @Valid // REMOVED 
    private Client client; 

Я удалил @Valid аннотацию на верхней части поля клиента и поэтому Spring MVC не проверяющего формы Thymeleaf больше.

Другим решением является включение скрытых входов с клиентскими (имя, название компании и адрес дочернего объекта), так thymeleaf может передать весь объект и BindingResult не будет жаловаться ...

0

пытаются добавить

<input type="hidden" name="client.id" value="${client.id}" /> 

в виде в HTML, те клиента объект с идентификатором значение будет создан, а затем оставить остальное на репозиторий, ему просто нужен идентификатор, чтобы связать запись.

+0

я делал ранее такие скрытые вход, но до сих пор не работает. – mlewandowski

+0

Не могли бы вы поделиться своей формой HTML? –

+0

Я разместил мою форму html в описании вопроса выше. – mlewandowski