Rails 4, завантаження декількох зображень або файлів за допомогою carrierwave


86

Як я можу завантажити декілька зображень із вікна вибору файлів за допомогою Rails 4 та CarrierWave? У мене є post_controllerі post_attachmentsмодель. Як я можу це зробити?

Хтось може навести приклад? Чи існує простий підхід до цього?

Відповіді:


195

Це рішення для завантаження декількох зображень за допомогою несучої хвилі в рейки 4 з нуля

Або ви можете знайти робочу демонстраційну версію: Multiple Attach Rails 4

Для цього просто виконайте ці дії.

rails new multiple_image_upload_carrierwave

У gem-файлі

gem 'carrierwave'
bundle install
rails generate uploader Avatar 

Створити пост-ешафот

rails generate scaffold post title:string

Створити риштування post_attachment

rails generate scaffold post_attachment post_id:integer avatar:string

rake db:migrate

У post.rb

class Post < ActiveRecord::Base
   has_many :post_attachments
   accepts_nested_attributes_for :post_attachments
end

У post_attachment.rb

class PostAttachment < ActiveRecord::Base
   mount_uploader :avatar, AvatarUploader
   belongs_to :post
end

У post_controller.rb

def show
   @post_attachments = @post.post_attachments.all
end

def new
   @post = Post.new
   @post_attachment = @post.post_attachments.build
end

def create
   @post = Post.new(post_params)

   respond_to do |format|
     if @post.save
       params[:post_attachments]['avatar'].each do |a|
          @post_attachment = @post.post_attachments.create!(:avatar => a)
       end
       format.html { redirect_to @post, notice: 'Post was successfully created.' }
     else
       format.html { render action: 'new' }
     end
   end
 end

 private
   def post_params
      params.require(:post).permit(:title, post_attachments_attributes: [:id, :post_id, :avatar])
   end

У переглядах / публікаціях / _form.html.erb

<%= form_for(@post, :html => { :multipart => true }) do |f| %>
   <div class="field">
     <%= f.label :title %><br>
     <%= f.text_field :title %>
   </div>

   <%= f.fields_for :post_attachments do |p| %>
     <div class="field">
       <%= p.label :avatar %><br>
       <%= p.file_field :avatar, :multiple => true, name: "post_attachments[avatar][]" %>
     </div>
   <% end %>

   <div class="actions">
     <%= f.submit %>
   </div>
<% end %>

Щоб відредагувати вкладення та список вкладень до будь-якої публікації. У views / posts / show.html.erb

<p id="notice"><%= notice %></p>

<p>
  <strong>Title:</strong>
  <%= @post.title %>
</p>

<% @post_attachments.each do |p| %>
  <%= image_tag p.avatar_url %>
  <%= link_to "Edit Attachment", edit_post_attachment_path(p) %>
<% end %>

<%= link_to 'Edit', edit_post_path(@post) %> |
<%= link_to 'Back', posts_path %>

Оновіть форму, щоб відредагувати подання вкладення / post_attachments / _form.html.erb

<%= image_tag @post_attachment.avatar %>
<%= form_for(@post_attachment) do |f| %>
  <div class="field">
    <%= f.label :avatar %><br>
    <%= f.file_field :avatar %>
  </div>
  <div class="actions">
    <%= f.submit %>
  </div>
<% end %>

Змініть метод оновлення в post_attachment_controller.rb

def update
  respond_to do |format|
    if @post_attachment.update(post_attachment_params)
      format.html { redirect_to @post_attachment.post, notice: 'Post attachment was successfully updated.' }
    end 
  end
end

У рейках 3 немає необхідності визначати сильні параметри, і, оскільки ви можете визначити атрибут_доступний як у моделі, так і accept_nested_attribute для публікації моделі, оскільки атрибут, який доступний, застарілий у рейках 4.

Для редагування вкладення ми не можемо одночасно змінювати всі вкладення. тому ми замінимо вкладення по одному, або ви можете змінити відповідно до вашого правила. Тут я просто покажу вам, як оновити будь-яке вкладення.


2
у шоу-дії контролера пошти, я думаю, ви забули @post = Post.find (params [: id])
wael

1
@SSR Чому ви переглядаєте вкладені файли кожного повідомлення в createдії? Рейки та носій хвилі досить розумні, щоб автоматично зберігати колекції.
яструб

3
Дуже хотілося б бачити редагування (особливо обробки :_destroyчастина)
Тун

5
@SSR - Ваша відповідь дуже корисна. Не могли б ви оновити свою відповідь за допомогою дії редагування.
raj_on_rails

2
Коли я додаю перевірки до моделі post_attachment, вони не заважають зберегти пост-модель. Натомість публікація зберігається, а потім помилка ActiveRecord видається лише для моделі вкладення. Я думаю, це через створення! метод. але використання create замість цього просто не вдається мовчки. Будь-яка ідея, як зробити так, щоб перевірка відбувалась у дописі?
dchess

32

Якщо ми подивимось на документацію CarrierWave, це насправді зараз дуже просто.

https://github.com/carrierwaveuploader/carrierwave/blob/master/README.md#multiple-file-uploads

Я буду використовувати Product як модель, я хочу додати малюнки, як приклад.

  1. Отримайте головну гілку Carrierwave і додайте її у свій Gemfile:

    gem 'carrierwave', github:'carrierwaveuploader/carrierwave'
    
  2. Створіть стовпець у передбачуваній моделі для розміщення масиву зображень:

    rails generate migration AddPicturesToProducts pictures:json
    
  3. Запустіть міграцію

    bundle exec rake db:migrate
    
  4. Додайте зображення до моделі продукту

    app/models/product.rb
    
    class Product < ActiveRecord::Base
      validates :name, presence: true
      mount_uploaders :pictures, PictureUploader
    end
    
  5. Додайте зображення до сильних параметрів у ProductsController

    app/controllers/products_controller.rb
    
    def product_params
      params.require(:product).permit(:name, pictures: [])
    end
    
  6. Дозвольте вашій формі приймати кілька зображень

    app/views/products/new.html.erb
    
    # notice 'html: { multipart: true }'
    <%= form_for @product, html: { multipart: true } do |f| %>
      <%= f.label :name %>
      <%= f.text_field :name %>
    
      # notice 'multiple: true'
      <%= f.label :pictures %>
      <%= f.file_field :pictures, multiple: true, accept: "image/jpeg, image/jpg, image/gif, image/png" %>
    
      <%= f.submit "Submit" %>
    <% end %>
    
  7. У своїх поданнях ви можете посилатися на зображення, що аналізують масив зображень:

    @product.pictures[1].url
    

Якщо ви виберете кілька зображень із папки, порядок буде точно таким, як ви робите їх зверху вниз.


9
Рішення цієї проблеми CarrierWave змушує мене скупитися. Він передбачає розміщення всіх посилань на файли в одному полі масиву! Це, звичайно, не буде вважатися "рейковим шляхом". Що робити, якщо ви захочете видалити деякі або додати додаткові файли до публікації? Я не кажу, що це було б неможливо, я просто кажу, що це було б потворно. Таблиця об’єднання - набагато краща ідея.
Тобі 1 Кенобі,

3
Я не міг більше погодитися Тобі. Чи могли б ви надати таке рішення?
drjorgepolanco

2
Ці рішення вже надає РСР. Інша модель встановлюється для зберігання завантаженого файлу, тоді те, що потребує багато завантажених файлів, співвідноситься один до одного або багато-до-багатьох з цією іншою моделлю. (таблиця об’єднань, яку я згадав у своєму попередньому коментарі, буде у випадку стосунків багато-до-багатьох)
Тобі 1 Кенобі

Дякую @ Toby1Kenobi, мені було цікаво, як метод масиву стовпців буде враховувати версії зображень (я не розумію, як це можна). Ваша стратегія здійсненна.
chaostheory

Я застосував цю функцію Carrierwave with Rails 5.xx , github.com/carrierwaveuploader/carrierwave/blob/master/ ... Але я не можу її успішно запустити, і це генерує помилку. UndefinedConversionError ("\x89" from ASCII-8BIT to UTF-8) Для рішення SSR він чудово працює з Rails 4.xx, але я стикаюся з проблемами (з Rails 5.xx), тобто його зберігання ActionDispatch::Http::UploadedFileв базі даних замість імені файлу. Це також не зберігає файли у загальних папках для заданого шляху у завантажувачі.
Мансі-шах

8

Деякі незначні доповнення до відповіді щодо РСР :

acceptpts_nested_attributes_for не вимагає змінити контролер батьківського об'єкта. Тож якщо виправити

name: "post_attachments[avatar][]"

до

name: "post[post_attachments_attributes][][avatar]"

тоді всі ці зміни контролера, подібні цим, стають зайвими:

params[:post_attachments]['avatar'].each do |a|
  @post_attachment = @post.post_attachments.create!(:avatar => a)
end

Також слід додати PostAttachment.newдо форми батьківського об'єкта:

У переглядах / публікаціях / _form.html.erb

  <%= f.fields_for :post_attachments, PostAttachment.new do |ff| %>
    <div class="field">
      <%= ff.label :avatar %><br>
      <%= ff.file_field :avatar, :multiple => true, name: "post[post_attachments_attributes][][avatar]" %>
    </div>
  <% end %>

Це призведе до зайвої зміни в батьківському контролері:

@post_attachment = @post.post_attachments.build

Для отримання додаткової інформації див. Rails fields_для форми, яка не відображається, вкладена форма

Якщо ви використовуєте Rails 5, то змініть Rails.application.config.active_record.belongs_to_required_by_defaultзначення на trueна false(у config / initializers / new_framework_defaults.rb) через помилку всередині accepts_nested_attributes_for (інакше acceptpts_nested_attributes_for зазвичай не працюватиме під Rails 5).

EDIT 1:

Додати про знищення :

У моделях / post.rb

class Post < ApplicationRecord
    ...
    accepts_nested_attributes_for :post_attachments, allow_destroy: true
end

У переглядах / публікаціях / _form.html.erb

 <% f.object.post_attachments.each do |post_attachment| %>
    <% if post_attachment.id %>

      <%

      post_attachments_delete_params =
      {
      post:
        {              
          post_attachments_attributes: { id: post_attachment.id, _destroy: true }
        }
      }

      %>

      <%= link_to "Delete", post_path(f.object.id, post_attachments_delete_params), method: :patch, data: { confirm: 'Are you sure?' } %>

      <br><br>
    <% end %>
  <% end %>

Таким чином, вам просто не потрібно мати контролер дочірнього об’єкта взагалі! Я маю на увазі, що жоден більше PostAttachmentsControllerне потрібен. Що стосується контролера батьківського об'єкта ( PostController), ви також майже не змінюєте його - єдине, що ви там змінюєте, - це список параметрів із білого списку (включаючи параметри, пов'язані з дочірніми об'єктами), наприклад:

def post_params
  params.require(:post).permit(:title, :text, 
    post_attachments_attributes: ["avatar", "@original_filename", "@content_type", "@headers", "_destroy", "id"])
end

Ось чому accepts_nested_attributes_forце так дивно.


Це насправді основні доповнення до відповіді @SSR, а не другорядні :) accept_nested_attributes_for - це цілком щось. Насправді взагалі не потрібен дочірній контролер. Дотримуючись вашого підходу, єдине, що я не можу зробити, це відображати повідомлення про помилки форми для дитини, коли щось не вдається із завантаженням.
Луїс Фернандо Ален

Дякуємо за ваш внесок. Я завантажив роботу, але мені цікаво, як я можу додати додаткові атрибути до поля форми post_attachments у views / posts / _form.html.erb? <%= d.text_field :copyright, name: "album[diapos_attributes][][copyright]", class: 'form-field' %>пише авторські права лише на останній запис, а не на всі.
SEJU

6

Крім того, я зрозумів, як оновити завантаження декількох файлів, і також трохи змінив його. Цей код мій, але ви отримуєте відхилення.

def create
  @motherboard = Motherboard.new(motherboard_params)
  if @motherboard.save
    save_attachments if params[:motherboard_attachments]
    redirect_to @motherboard, notice: 'Motherboard was successfully created.'
  else
    render :new
  end
end


def update
  update_attachments if params[:motherboard_attachments]
  if @motherboard.update(motherboard_params)
    redirect_to @motherboard, notice: 'Motherboard was successfully updated.'
  else
   render :edit
  end
end

private
def save_attachments
  params[:motherboard_attachments]['photo'].each do |photo|
    @motherboard_attachment = @motherboard.motherboard_attachments.create!(:photo => photo)
  end
end

 def update_attachments
   @motherboard.motherboard_attachments.each(&:destroy) if @motherboard.motherboard_attachments.present?
   params[:motherboard_attachments]['photo'].each do |photo|
     @motherboard_attachment = @motherboard.motherboard_attachments.create!(:photo => photo)
   end
 end

1
Дякуємо, що поділилися своїм кодом. коли ви отримаєте час, будь ласка, оновіть код на моєму репозиторії gihub і не забудьте коментувати кожен метод, щоб усі могли легко зрозуміти код.
SSR

1
Я клонував репо, ви дасте мені дозвіл на піар?
Кріс Хабгуд,

Так, звичайно. Яке ваше ім'я користувача github
SSR

Ви мали можливість надати мені доступ?
Кріс Хабгуд,

2

Ось мій другий рефактор моделі:

  1. Перемістіть приватні методи до моделювання.
  2. Замініть @ motherboard на self.

Контролер:

def create
  @motherboard = Motherboard.new(motherboard_params)

  if @motherboard.save
    @motherboard.save_attachments(params) if params[:motherboard_attachments]
  redirect_to @motherboard, notice: 'Motherboard was successfully created.'
  else
    render :new
  end
end

def update
  @motherboard.update_attachments(params) if params[:motherboard_attachments]
  if @motherboard.update(motherboard_params)
    redirect_to @motherboard, notice: 'Motherboard was successfully updated.'
  else
    render :edit
  end
end

У моделі материнської плати:

def save_attachments(params)
  params[:motherboard_attachments]['photo'].each do |photo|
    self.motherboard_attachments.create!(:photo => photo)
  end
end

def update_attachments(params)
  self.motherboard_attachments.each(&:destroy) if self.motherboard_attachments.present?
  params[:motherboard_attachments]['photo'].each do |photo|
    self.motherboard_attachments.create!(:photo => photo)
  end
end

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.