Imagem múltipla do Rails 4 ou upload de arquivo usando carrierwave

86

Como posso fazer upload de várias imagens de uma janela de seleção de arquivo usando Rails 4 e CarrierWave? Eu tenho um post_controllere um post_attachmentsmodelo. Como posso fazer isso?

Alguém pode dar um exemplo? Existe uma abordagem simples para isso?

SSR
fonte

Respostas:

195

Esta é a solução para fazer upload de várias imagens usando carrierwave no rails 4 do zero

Ou você pode encontrar uma demonstração de trabalho: Multiple Attachment Rails 4

Para fazer, basta seguir estas etapas.

rails new multiple_image_upload_carrierwave

No arquivo gem

gem 'carrierwave'
bundle install
rails generate uploader Avatar 

Criar post scaffold

rails generate scaffold post title:string

Criar andaime post_attachment

rails generate scaffold post_attachment post_id:integer avatar:string

rake db:migrate

Em post.rb

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

Em post_attachment.rb

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

Em 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

Em views / posts / _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 %>

Para editar um anexo e uma lista de anexos para qualquer postagem. Em 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 %>

Atualize o formulário para editar um anexo views / 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 %>

Modifique o método de atualização em 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

No rails 3, não há necessidade de definir parâmetros fortes e, como você pode definir attribute_accessible no modelo e accept_nested_attribute para postar o modelo, o atributo acessível está obsoleto no rails 4.

Para editar um anexo, não podemos modificar todos os anexos de uma vez. portanto, substituiremos o anexo um por um, ou você pode modificar de acordo com sua regra. Aqui, apenas mostro como atualizar qualquer anexo.

SSR
fonte
2
no show action do post controller, acho que você esqueceu @post = Post.find (params [: id])
wael
1
@SSR Por que você percorre cada anexo de postagem em createação? Rails e carrierwave são inteligentes o suficiente para salvar coleções automaticamente.
falcão
3
Adoraria ver a edição (especialmente a :_destroyparte de manuseio )
Tun
5
@SSR - Sua resposta é muito útil. Você também poderia atualizar sua resposta com a ação de edição.
raj_on_rails
2
Quando adiciono validações ao modelo post_attachment, elas não evitam que o modelo de postagem seja salvo. Em vez disso, a postagem é salva e, em seguida, o erro inválido ActiveRecord é lançado apenas para o modelo de anexo. Acho que isso é por causa da criação! método. mas, em vez disso, usar criar falha silenciosamente. Alguma ideia de como fazer a validação acontecer no poste chegar aos anexos?
dchess
32

Se dermos uma olhada na documentação da CarrierWave, isso é realmente muito fácil agora.

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

Vou usar o Produto como modelo e quero adicionar as fotos, como exemplo.

  1. Obtenha o branch master Carrierwave e adicione-o ao seu Gemfile:

    gem 'carrierwave', github:'carrierwaveuploader/carrierwave'
  2. Crie uma coluna no modelo pretendido para hospedar uma série de imagens:

    rails generate migration AddPicturesToProducts pictures:json
  3. Execute a migração

    bundle exec rake db:migrate
  4. Adicionar fotos ao modelo do produto

    app/models/product.rb
    
    class Product < ActiveRecord::Base
      validates :name, presence: true
      mount_uploaders :pictures, PictureUploader
    end
  5. Adicionar imagens a parâmetros fortes em ProductsController

    app/controllers/products_controller.rb
    
    def product_params
      params.require(:product).permit(:name, pictures: [])
    end
  6. Permita que seu formulário aceite várias fotos

    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. Em suas visualizações, você pode fazer referência às imagens analisando a matriz de imagens:

    @product.pictures[1].url

Se você escolher várias imagens de uma pasta, a ordem será a ordem exata em que você as está tirando de cima para baixo.

drjorgepolanco
fonte
9
A solução da CarrierWave para esse problema me faz estremecer. Envolve colocar todas as referências aos arquivos em um campo em uma matriz! Certamente não seria considerado o "caminho dos trilhos". E se você quiser remover alguns ou adicionar arquivos extras à postagem? Não estou dizendo que não seria possível, só estou dizendo que seria feio. Uma tabela de junção é uma ideia muito melhor.
Toby 1 Kenobi,
3
Eu não poderia concordar mais, Toby. Você faria a gentileza de fornecer essa solução?
drjorgepolanco
2
Essas soluções já são fornecidas pelo SSR. Outro modelo é colocado em prática para manter o arquivo carregado, então o que precisa de muitos arquivos carregados se relaciona em uma relação um-para-muitos ou muitos-para-muitos com esse outro modelo. (a tabela de junção que mencionei em meu comentário anterior seria no caso de um relacionamento muitos para muitos)
Toby 1 Kenobi
Obrigado @ Toby1Kenobi, queria saber como o método de matriz de coluna contabilizaria versões de imagem (não vejo como pode). Sua estratégia é factível.
chaostheory
Implementei esse recurso do Carrierwave com Rails 5.xx, github.com/carrierwaveuploader/carrierwave/blob/master/… Mas não consigo executá-lo com sucesso e está gerando erro. UndefinedConversionError ("\x89" from ASCII-8BIT to UTF-8) Para solução SSR, ele funciona bem com Rails 4.xx, mas estou enfrentando desafios (com Rails 5.xx) ou seja, seu armazenamento ActionDispatch::Http::UploadedFileem banco de dados ao invés de nome de arquivo. Também não armazena arquivos em pastas públicas para determinado caminho no uploader.
Mansi Shah
8

Algumas pequenas adições à resposta SSR :

aceita_nested_attributes_for não requer que você altere o controlador do objeto pai. Então, se corrigir

name: "post_attachments[avatar][]"

para

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

então, todas essas mudanças de controlador como essas se tornam redundantes:

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

Além disso, você deve adicionar PostAttachment.newao formulário do objeto pai:

Em views / posts / _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 %>

Isso tornaria redundante essa mudança no controlador dos pais:

@post_attachment = @post.post_attachments.build

Para obter mais informações, consulte Rails fields_for form not show up, anested form

Se você usa Rails 5, mude o Rails.application.config.active_record.belongs_to_required_by_defaultvalor detrue para false(em config / initializers / new_framework_defaults.rb) devido a um bug dentro de accept_nested_attributes_for (caso contrário, accept_nested_attributes_for não funcionará geralmente no Rails 5).

EDIT 1:

Para adicionar sobre destruir :

Em models / post.rb

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

Em views / posts / _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 %>

Dessa forma, você simplesmente não precisa ter um controlador de objeto filho! Quero dizer, nenhum PostAttachmentsControlleré mais necessário. Quanto ao controlador do objeto pai ( PostController), você também quase não o altera - a única coisa que você altera lá é a lista de parâmetros da lista de permissões (para incluir os parâmetros relacionados ao objeto filho) como este:

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

É por isso que accepts_nested_attributes_foré tão incrível.

programas
fonte
Essas são, na verdade, grandes adições à resposta @SSR, não pequenas :) accept_nested_attributes_for é bastante. Na verdade, não há necessidade de um controlador filho. Seguindo sua abordagem, a única coisa que não consigo fazer é exibir mensagens de erro de formulário para a criança quando algo dá errado com o upload.
Luis Fernando Alen
Obrigado pela sua contribuição. Consegui fazer o upload funcionar, mas gostaria de saber como poderia adicionar atributos adicionais ao campo do formulário post_attachments em views / posts / _form.html.erb? <%= d.text_field :copyright, name: "album[diapos_attributes][][copyright]", class: 'form-field' %>grava os direitos autorais apenas para o último registro e não para todos eles.
SEJU
6

Também descobri como atualizar o upload de vários arquivos e também o refatorou um pouco. Este código é meu, mas você entendeu.

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
Chris Habgood
fonte
1
Obrigado por compartilhar seu código. quando tiver tempo, atualize o código em meu repositório gihub e não se esqueça de comentar cada método para que todos possam entender o código facilmente.
SSR
1
Eu clonei os repos, você me dá permissão para fazer um PR?
Chris Habgood
Sim claro. Qual é o seu nome de usuário do github
SSR
Você teve a chance de me dar acesso?
Chris Habgood
2

Aqui está minha segunda refatoração no modelo:

  1. Mova métodos privados para o modelo.
  2. Substitua @motherboard por self.

Controlador:

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

No modelo da placa-mãe:

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
Chris Habgood
fonte
2

Ao usar a associação, @post.post_attachmentsvocê não precisa definir o post_id.

Chris Habgood
fonte