2017-02-16 8 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). 
} 

問題はセーブ機能である - 私は、HTTP POSTによって送信されたマシンのオブジェクトのためのexisitng Clientオブジェクトを渡す方法がわかりません。

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: 私のコントローラは、クライアントが送信されていないとBindingResultがエラーを投げていると文句を言い。

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フォームを検証していません。私は以前、このような非表示の入力を行って、それでもいる

0

試みは、HTMLのフォームで

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

を追加するために、IDを持つそれらのクライアントオブジェクトは、値を作成し、その後に残りの部分を残すことになりますリポジトリには、レコードを関連付けるためにidだけが必要です。

+0

...

別の解決策は、クライアントの詳細を隠し入力(名前、会社と子アドレスオブジェクト)を含めることがあるので、thymeleafは、完全なオブジェクトを転送することができますし、BindingResultは文句はありません動作しません。 – mlewandowski

+0

あなたのHTMLフォームを共有してもよろしいですか? –

+0

上記の質問の説明に自分のhtmlフォームを掲載しました。 – mlewandowski

関連する問題