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

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

新規登録して質問してみよう
ただいま回答率
85.50%
Ruby

Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。

Ruby on Rails

Ruby on Railsは、オープンソースのWebアプリケーションフレームワークです。「同じことを繰り返さない」というRailsの基本理念のもと、他のフレームワークより少ないコードで簡単に開発できるよう設計されています。

Ajax

Ajaxとは、Webブラウザ内で搭載されているJavaScriptのHTTP通信機能を使って非同期通信を利用し、インターフェイスの構築などを行う技術の総称です。XMLドキュメントを指定したURLから読み込み、画面描画やユーザの操作などと並行してサーバと非同期に通信するWebアプリケーションを実現することができます。

Q&A

解決済

1回答

4183閲覧

【Rails】Ajaxの部分テンプレートが1つめだけ更新されない

Romay

総合スコア40

Ruby

Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。

Ruby on Rails

Ruby on Railsは、オープンソースのWebアプリケーションフレームワークです。「同じことを繰り返さない」というRailsの基本理念のもと、他のフレームワークより少ないコードで簡単に開発できるよう設計されています。

Ajax

Ajaxとは、Webブラウザ内で搭載されているJavaScriptのHTTP通信機能を使って非同期通信を利用し、インターフェイスの構築などを行う技術の総称です。XMLドキュメントを指定したURLから読み込み、画面描画やユーザの操作などと並行してサーバと非同期に通信するWebアプリケーションを実現することができます。

0グッド

0クリップ

投稿2019/04/09 06:23

編集2019/04/20 00:44

前提・実現したいこと

RailsでAjax実装している部分テンプレートが、最初の1つめだけAjaxとして機能していません。
2つめ以降は、入力フォームから入力し送信を押すごとに、Ajaxとして機能し遷移なしに表示されます。
1つめは、ページをリロードすると表示されるものの、Ajaxとして表示されていません。

edit.html.erbの画面に入力フォームとその下にAjax表示部分のスペースがあり、入力して「追加」を押すごとに、そのスペースに部分テンプレートが追加されていくイメージです。

Ruby 2.4.5
Rails 4.2.8
mySQL(開発環境)
postgreSQL(本番環境)
Heroku

該当のソースコード

ユーザーがフォームから入力するフォームと、Ajaxの表示部分
users/edit.html.erb

ruby

1<div class="content content-item"> 2 <ol> 3 <!-- アイテムAjaxで送信 --> 4 <%= form_tag('/items', method: :post, remote: true) do %> 5 <input type="text" name="text" pattern=".*\S+.*" required placeholder="23文字以内で" size="23" maxlength="23" autocomplete="off" class="header-btn"> 6 <%= submit_tag "追加" ,:class => "header-btn margin-top10px" %> 7 <% end %> 8 </ol> 9</div> 10 11<div class="content content-item"> 12 <ol> 13 <% if @items.present? %> 14 <div id="item-content"> 15 <% @items.each do |item| %> 16 <%= render partial: "item", locals: { item: item } %> 17 <% end %> 18  </div> 19 <% end %> 20 </ol> 21</div>

item_controllerの該当部

Ruby

1 def create 2 @item = Item.new(text: item_params[:text], user_id: current_user.id, check: 0) 3 4 #Ajax仕様 5 respond_to do |format| 6 if @item.save 7 format.html { redirect_to @item, notice: '作成完了'} 8 format.json { render :show, status: :created, location: @item } 9 10 format.js 11 else 12 format.html { render :new } 13 format.json { render json: @item.errors, status: :unprocessable_entity } 14 end 15 end 16 end

_item.html.erb 部分テンプレートです。

ruby

1<li class="idea"> 2 <div class="idea-text"> 3 <div class="checked-item_<%= "#{item.id}"%>" > 4 <%= item.text %> 5 </div> 6 </div> 7 <div class="item-menu-right"> 8 <%= link_to item_path(item), method: :delete , remote: true do %> 9 <i class="fa fa-trash fa-fw" aria-hidden="true"></i> 10 <% end %> 11 <%= link_to edit_item_path(item), method: :get ,class: "fa-edit-margin", item_id: item.id do %> 12 <i class="fa fa-edit fa-fw" aria-hidden="true" ></i> 13 <% end %> 14 15 <%= check_box_tag '', '', item.check, {'data-id' => item.id, 'data-user-id' => item.user_id ,class: "check_#{item.id} option-input"} %> 16 17 </div> 18</li> 19

試したこと

部分テンプレートの書き方、<%= render partial: "item", locals: { item: item } %>の部分によって、挙動が変わるのでしょうか。

補足情報(FW/ツールのバージョンなど)

##追記

コントローラーの処理後,create.js.erbに処理が渡っているためこちらを追記しておきます。

create.js.erb

erb

1$("<%= escape_javascript(render partial: '/users/item', locals: { item: @item }) %>").appendTo("#item-content"); 2 3//submitの後に、フォームの中身を空にする 4$('form').find("textarea, :text, select").val("").end().find(":checked").prop("checked", false);

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

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

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

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

Romayさんが実施しているrailsのajaxのやり方は私が知らないだけなのかも知れませんが、私は「これは動かないだろうな。むしろなぜ2回目は以降動いているんだろう?」と感じました。


私が知っている方式

以下のサイトにあるように、create.js.erb がないとajaxが動作しないと私は認識しておりました。
Railsで remote: true と js.erbを使って簡単にAjax(非同期通信)を実装しよう!(いいね機能のデモ付)

このファイル名の"create"はアクション名ですね。
そして、このファイルがあればcreateアクション内の render は一切不要です。

処理の流れは以下のような感じになります。

  1. edit.html.erb: remote: true なフォームからリクエスト
  2. controller: create アクション

(editなのにcreate!? updateじゃなくて?)
0. create.js.erb:

js

1// create.js.erb 2// タイプミスとか勘違いで動かなかったらごめんなさい 3$('.content.content-item ol').html("<%= j(render partial: 'item', locals: {item: @item}) %>") 4 5# .html の部分を .append にすれば期待された動作になりそう

4.<div class="content content-item"><ol>の中に _item.html.erb 部分が描画される


Romayさん方式についての考察

Romayさん方式がどうやって動作しているのか予想したのですが、おそらくcontroller内のcreateアクションの location: で描画場所を指定し、:show を描画しているのではないかと推測しております。

少なくとも、描画されるのは :show と考えられますので、正常な動作は望めないのかなと感じました。

また location: @item だけでどうやって描画すべき場所をrailsが認識しているのか不思議にもなりました。

投稿2019/04/20 00:34

siruku6

総合スコア1382

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

Romay

2019/04/20 00:42

回答ありがとうございます。create.js.erbは作成しております!こちらのミスで質問に載せ忘れておりました。create.js.erbを追記しておきます!
siruku6

2019/04/20 01:02 編集

これならかなり動きそうな感じですね! 動かない理由もわかりました。 初回は"#item-content"が存在しないから動いていません。 次の修正をしてみてください! # users/edit.html.erb 二つ目の <div class="content content-item"> になにかidを追加してください。 この場所をcreate.js.erbで指定したいからです。 私だったら <div id="created-items" class="content content-item"> などにします。 # create.js.erb $("<%= escape_javascript(render partial: '/users/item', locals: { item: @item }) %>").appendTo("#item-content"); の部分を $('#created-items ol').append("<%= escape_javascript(render partial: '/users/item', locals: { item: @item }) %>") に変えれば動くような気がします。 私のコードに Syntaxエラー などのくだらないミスがあるかもしれませんので、そこはご容赦ください。
Romay

2019/04/21 05:21

"初回は#item-contentが存在しない"ご指摘の通りですね!直したところ上手く動きました!ありがとうございます!
siruku6

2019/04/26 11:00

役に立てたようでうれしいです^^
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.50%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問