Active Storage利用して複数の画像を投稿できる機能とJavaScriptを用いてプレビューできる機能を実装します。
今回も初心者向けにレシピ投稿アプリを例に作成していきます。
※なお今回は前回の記事の発展的な内容になっております。
Active Storageの導入方法についてこちら
プレビュー機能についてはこちら
今回は上記の機能が実装されている前提で話を進めていきます。
##Active Storage関連ファイルの修正
まずはActive Storage周りのファイルから修正していきます。
####アソシエーションの修正
レコードとファイルを1対1の関係で紐づけるためにhas_one_attachedメソッドを使用していましたが、今回は1対多の関係に変更するので**has_many_attached**メソッドに修正します。Railsガイド
また、imageを複数形のimagesに変更します。
class Recipe < ApplicationRecord
has_many_attached :images #ここを修正
end
####投稿フォームの修正
file_fieldのimageをimagesに修正します。
また、name属性を追加し送信する際に必要な画像の配列を設定します。
<%= form_with model: @recipe, local: true do |f| %>
#中略
<div class="form-group">
<label class="text-secondary">画像</label><br>
<%= f.file_field :images, name: 'recipe[images][]' %> #ここを修正
</div>
#以下略
<% end %>
####コントローラーの修正
画像の配列を受け取れるようにストロングパラメーターを修正します。
class RecipesController < ApplicationController
#中略
private
def recipe_params
params.require(:recipe).permit(:title, :text, :category_id, :time_required_id, images: []) #images: []に修正
end
end
####バリデーションの設定
今回、投稿できる画像を3枚までにしたいので、独自のバリデーションメソッドを作成していきます。Railsガイド
class Recipe < ApplicationRecord
has_many_attached :images
#ここから追加
validate :image_length #カスタムメソッドなので"validate"
private
def image_length
if images.length >= 4
errors.add(:images, "は3枚以内にしてください")
end
end
#ここまで追加
end
以上でActive Storage関連の修正は完了です。
##preview.jsの修正
今回は以下のGIFのようにひとつ画像を選択すると新しく画像選択フォームが出現する仕様にします。
まずは、前回作成したpreview.jsを確認してみましょう。
if (document.URL.match(/new/)){
document.addEventListener('DOMContentLoaded', () => {
const createImageHTML = (blob) => {
const imageElement = document.getElementById('new-image');
const blobImage = document.createElement('img');
blobImage.setAttribute('class', 'new-img')
blobImage.setAttribute('src', blob);
imageElement.appendChild(blobImage);
};
document.getElementById('recipe_image').addEventListener('change', (e) => {
const imageContent = document.querySelector('img');
if (imageContent){
imageContent.remove();
}
const file = e.target.files[0];
const blob = window.URL.createObjectURL(file);
createImageHTML(blob);
});
});
}
こちらを修正していきます。
今回は、以下のようなHTMLを生成することを前提にpreview.jsを編集していきます。
<div id="new-image">
<!-- ここから -->
<div class="image-element">
<img class="new-img" src="xxxxxxx...">
<input id="recipe_image_n" class="recipe-images" name="recipe[images][]" type="file">
</div>
<!-- ここまでを複製していくイメージ -->
</div>
前回まではid="new-image"のdiv要素に直接プレビュー画像を挿入していましたが、今回は新たなdiv要素を作成しその中に画像を挿入していきます。
そして、classにimage-elementを設定し**querySelectorAll**メソッドとlengthメソッドで作成された要素の数を定数imageElementNumに格納します。
if (document.URL.match(/new/)){
document.addEventListener('DOMContentLoaded', () => {
const ImageList = document.getElementById('new-image'); //追記
const createImageHTML = (blob) => {
const imageElement = document.createElement('div'); //new-imageからdivに変更
//ここから
imageElement.setAttribute('class', "image-element")
let imageElementNum = document.querySelectorAll('.image-element').length
//ここまで追記
const blobImage = document.createElement('img');
blobImage.setAttribute('class', 'new-img')
blobImage.setAttribute('src', blob);
imageElement.appendChild(blobImage);
ImageList.appendChild(imageElement); //追記
});
};
document.getElementById('recipe-images').addEventListener('change', (e) => { //recipe-imagesに修正
//ここから
//const imageContent = document.querySelector('img');
//if (imageContent){
//imageContent.remove();
//}
//ここまで削除
const file = e.target.files[0];
const blob = window.URL.createObjectURL(file);
createImageHTML(blob);
});
});
}
次に新しく画像選択フォームを作成する記述をしていきます。
createElementメソッドでinput要素を生成し、setAttributeで属性を追加していきます。
そして、inputのidに先ほど宣言したimageElementNumを使いinput要素が何番目の要素かを判別します。
//中略
document.addEventListener('DOMContentLoaded', () => {
const ImageList = document.getElementById('new-image');
const createImageHTML = (blob) => {
const imageElement = document.createElement('div');
imageElement.setAttribute('class', "image-element")
let imageElementNum = document.querySelectorAll('.image-element').length
const blobImage = document.createElement('img');
blobImage.setAttribute('class', 'new-img')
blobImage.setAttribute('src', blob);
//ここから
const inputHTML = document.createElement('input');
inputHTML.setAttribute('id', `recipe_image_${imageElementNum}`);
inputHTML.setAttribute('class', 'recipe-images');
inputHTML.setAttribute('name', 'recipe[images][]');
inputHTML.setAttribute('type', 'file');
//ここまで追記
imageElement.appendChild(blobImage);
imageElement.appendChild(inputHTML); //追記
ImageList.appendChild(imageElement);
//以下略
これで、画像を選択すると新しく画像選択フォームが出現するようになりました。
2枚目以降にもイベント発火するよう処理を記述していきます。
1度目に発火するイベントとほとんど同じ記述になります。
//中略
const createImageHTML = (blob) => {
const imageElement = document.createElement('div');
imageElement.setAttribute('class', "image-element")
let imageElementNum = document.querySelectorAll('.image-element').length
const blobImage = document.createElement('img');
blobImage.setAttribute('class', 'new-img')
blobImage.setAttribute('src', blob);
const inputHTML = document.createElement('input');
inputHTML.setAttribute('id', `recipe_image_${imageElementNum}`);
inputHTML.setAttribute('class', 'recipe-images');
inputHTML.setAttribute('name', 'recipe[images][]');
inputHTML.setAttribute('type', 'file');
imageElement.appendChild(blobImage);
imageElement.appendChild(inputHTML);
ImageList.appendChild(imageElement);
//ここから
inputHTML.addEventListener('change', (e) => {
const file = e.target.files[0];
const blob = window.URL.createObjectURL(file);
createImageHTML(blob);
});
//ここまで追記
};
//以下略
ここまでで複数の画像投稿に対応したプレビュー機能の実装が完了しました。
しかし、これでは1枚画像を選択するたびに新しく画像選択フォームが出現してしまうので何枚でも画像選択が可能になってしまいます。
モデルにバリデーションをかけているので投稿は保存されません。
####条件分岐で制限をかけよう
input要素の数える定数imageElementNumを使い、3枚選択すると新しく画像選択フォームが出現しないように条件分岐していきましょう。
//中略
const inputHTML = document.createElement('input');
inputHTML.setAttribute('id', `recipe_image_${imageElementNum}`);
inputHTML.setAttribute('class', 'recipe-images');
inputHTML.setAttribute('name', 'recipe[images][]');
inputHTML.setAttribute('type', 'file');
imageElement.appendChild(blobImage);
if (imageElementNum < 2) { //追記
imageElement.appendChild(inputHTML);
} //追記
ImageList.appendChild(imageElement);
inputHTML.addEventListener('change', (e) => {
const file = e.target.files[0];
const blob = window.URL.createObjectURL(file);
createImageHTML(blob);
});
//以下略
if (document.URL.match(/new/)){
document.addEventListener('DOMContentLoaded', () => {
const ImageList = document.getElementById('new-image');
const createImageHTML = (blob) => {
const imageElement = document.createElement('div');
imageElement.setAttribute('class', "image-element")
let imageElementNum = document.querySelectorAll('.image-element').length
const blobImage = document.createElement('img');
blobImage.setAttribute('class', 'new-img')
blobImage.setAttribute('src', blob);
const inputHTML = document.createElement('input');
inputHTML.setAttribute('id', `recipe_image_${imageElementNum}`);
inputHTML.setAttribute('class', 'recipe-images');
inputHTML.setAttribute('name', 'recipe[images][]');
inputHTML.setAttribute('type', 'file');
imageElement.appendChild(blobImage);
if (imageElementNum < 2) {
imageElement.appendChild(inputHTML);
}
ImageList.appendChild(imageElement);
inputHTML.addEventListener('change', (e) => {
const file = e.target.files[0];
const blob = window.URL.createObjectURL(file);
createImageHTML(blob);
});
};
document.getElementById('recipe-images').addEventListener('change', (e) => {
const file = e.target.files[0];
const blob = window.URL.createObjectURL(file);
createImageHTML(blob);
});
});
}
次回は投稿された複数の画像をスライド形式で表示する実装を行っていきます。