2017-02-16 63 views
0

我正在开发一个管理客户机和应用程序的应用程序。 我已经创建了所有必要的表格,模式,控制器等。Spring MVC + Thymeleaf - 保存关系@ManyToOne

客户实体具有列表和机器具有客户关系(双向,非可选)。

我所遇到的问题与向现有客户端添加全新机器(提供此服务器已存在)有关。

因此,这里是一些代码短剪断:

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

我已经取消了对客户端领域的顶部,因此Spring MVC的不确认Thymeleaf形式@Valid注解了。

另一个解决办法是包括与客户的详细资料隐藏的输入(姓名,公司和子地址对象),所以thymeleaf可以传输完整的对象和BindingResult不会抱怨...

0

尝试在HTML表单中添加

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

,与ID的客户对象值将被创建,然后离开,其余的存储库,它只需要id来关联记录。

+0

我已经做以前,这些隐藏的输入,但仍不起作用。 – mlewandowski

+0

你介意分享你的HTML表单吗? –

+0

我在上面的问题描述中发布了我的html表单。 – mlewandowski

相关问题