質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

87.80%

【rails5】form_for内に複数のfile_fieldがあるとajax通信後に勝手に順番がズレる

受付中

回答 0

投稿

  • 評価
  • クリップ 0
  • VIEW 352

score 99

遭遇している不具合はタイトルの通りなのですが、これだけだとよく分からないと思いますので、順を追って説明します。

まず、以下のような二つの画像が投稿できるformをajaxで実装しています。

イメージとしては、Itemモデルがあり、そのモデルは画像①(必須)、画像②(任意)を持つという形になっています。

formのコードイメージ(画像投稿部分だけの簡易ver)
= form_for(item, url: path , html:{id:'item_form', data_parsley_validate:"" ,multipart: true}, remote: true) do |f|

  .form-group
    .image_box
      label for="top_image"
        | 画像①(必須)
        = f.file_field :top_image, accept: 'image/*', id: "top_image", class: "image_item pointer"

  .form-group
    .image_box
      label for="top_image_2"
        | 画像②(任意)
        = f.file_field :top_image_2, accept: 'image/*', id: "top_image_2", class: "image_item pointer"

  .form-group
    = f.submit 'Submit', class:"btn", data: { disable_with: "Sending..." }

その上で、一点よく分からない不具合が発生しています。

例)
1・すでに投稿したアイテムの画像②を追加投稿しようとする
2・画像容量など何らかのバリデーションにかかる(不具合発生させるためにわざとです。)
3・投稿画面にエラーが表示される
--ここからが問題---
4・前述の投稿フォームの画像①のfile_field(表示時はinputタグ)が画像②のfile_fieldの位置に動き、画像②のfile_field(inputタグ)がHTML上から消えている。以下のイメージです。

エラー発生後のformのコードイメージ
= form_for(item, url: path , html:{id:'item_form', data_parsley_validate:"" ,multipart: true}, remote: true) do |f|

  .form-group
    .image_box
      label for="top_image"
        | 画像①(必須)

  .form-group
    .image_box
      label for="top_image_2"
        | 画像②(任意)
        = f.file_field :top_image, accept: 'image/*', id: "top_image", class: "image_item pointer"

  .form-group
    = f.submit 'Submit', class:"btn", data: { disable_with: "Sending..." }

ちなみにページ自体を再読み込みしてあげると最初の正しいformに戻ります。

Controller側は以下の通りです。シンプルな処理しかしてません。

更新時のControllerのupdateアクション
def update
  @item = Item.find(params[:id])
  if @item.update(item_params)
    @sussess = true
    respond_to do |format|
      format.js { flash[:info] = '情報が更新されました。' }
    end
  else
    respond_to do |format|
      format.js
    end
  end
end
Controller側で処理した後のupdate.js.slim

ここもシンプルです

 - if @success == true
  | window.location.href = "/managements/articles/#{@article.id}/edit";

- else 
  | $('.alert').remove(); 
  | $(" #{ j( render 'managements/shared/error_messages', object: @article )}").insertAfter(".x_title");

自分で把握できていること

1・file_field(inputタグ)の場所が勝手に動くタイミングは、update.js.slimが実行された直後

2・試しにinputタグを画像②のfile_fieldの後に直書きした場合、name属性の有無で挙動が変わった。

  .form-group
    .image_box
      label for="top_image"
        | 画像①(必須)
        = f.file_field :top_image, accept: 'image/*', id: "top_image", class: "image_item pointer"

  .form-group
    .image_box
      label for="top_image_2"
        | 画像②(任意)
        = f.file_field :top_image_2, accept: 'image/*', id: "top_image_2", class: "image_item pointer"

  .form-group
    .image_box
    input.image_item accept="image/*" name=""


【name属性のあるinputタグ】画像①のinputタグが画像②のinputタグに移り、画像②のinputタグがこの直書きしたinputタグに移った

  .form-group
    .image_box
      label for="top_image"
        | 画像①(必須)


  .form-group
    .image_box
      label for="top_image_2"
        | 画像②(任意)
        = f.file_field :top_image, accept: 'image/*', id: "top_image", class: "image_item pointer"


  .form-group
    .image_box
    = f.file_field :top_image_2, accept: 'image/*', id: "top_image_2", class: "image_item pointer"


 
【name属性のないinputタグ】画像①のinputタグが画像②のinputタグに移り、画像②のinputタグが消失。追加したname属性がないinputタグはそのまま。

  .form-group
    .image_box
      label for="top_image"
        | 画像①(必須)


  .form-group
    .image_box
      label for="top_image_2"
        | 画像②(任意)
        = f.file_field :top_image, accept: 'image/*', id: "top_image", class: "image_item pointer"

  .form-group
    .image_box
    input.image_item accept="image/*"

3・このフォームがあるページのjsを完全にコメントアウトしても、この事象は発生。

4・type="file"ではないinputタグでは発生しない。

数時間悪戦苦闘したものの、結局原因は分からず。。。form_forが勝手に処理しているのでしょうか?どなたか原因らしきことが分かれば教えて頂けると嬉しいです。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

まだ回答がついていません

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 87.80%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る