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.
create
ação? Rails e carrierwave são inteligentes o suficiente para salvar coleções automaticamente.:_destroy
parte de manuseio )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.
Obtenha o branch master Carrierwave e adicione-o ao seu Gemfile:
Crie uma coluna no modelo pretendido para hospedar uma série de imagens:
Execute a migração
Adicionar fotos ao modelo do produto
Adicionar imagens a parâmetros fortes em ProductsController
Permita que seu formulário aceite várias fotos
Em suas visualizações, você pode fazer referência às imagens analisando a matriz de imagens:
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.
fonte
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 armazenamentoActionDispatch::Http::UploadedFile
em 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.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
para
então, todas essas mudanças de controlador como essas se tornam redundantes:
Além disso, você deve adicionar
PostAttachment.new
ao formulário do objeto pai:Em views / posts / _form.html.erb
Isso tornaria redundante essa mudança no controlador dos pais:
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_default
valor detrue
parafalse
(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
Em views / posts / _form.html.erb
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:É por isso que
accepts_nested_attributes_for
é tão incrível.fonte
<%= 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.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.
fonte
Aqui está minha segunda refatoração no modelo:
Controlador:
No modelo da placa-mãe:
fonte
Ao usar a associação,
@post.post_attachments
você não precisa definir opost_id
.fonte