bootstrap_form icon indicating copy to clipboard operation
bootstrap_form copied to clipboard

How to display selected file name?

Open iraszl opened this issue 6 years ago • 17 comments
trafficstars

Without the bootstrap_form gem by default the rails forms show the filename or the number of files when selected, so the user knows they successfully selected a file to be uploaded in the form. Screenshot 2019-03-26 at 12 01 02 PM

However, when I use bootstrap_form, the file name and count is gone. Is there a way to bring it back, otherwise the users think that they failed to select the file and try again, and sometimes give up thinking the file select in the form is broken.

iraszl avatar Mar 26 '19 04:03 iraszl

Thanks for the feedback. I know that the file upload control is implemented in a different way than some of the other controls. I don't know why, as it was done before I was a maintainer. I'm sure there was a good reason. I will add this to my list of issues to look at.

In the meantime, you can implement your own fields to give feedback. That's what I've done.

lcreid avatar Mar 27 '19 15:03 lcreid

Thanks for the kind comment. What do you mean by your own fields? Do you have an example of what you've done please?

iraszl avatar Mar 28 '19 02:03 iraszl

I have one example I can share:

<%= bootstrap_form_with(model: @document, local: true) do |f| %>
  <div class="row">
    <div class="col-sm-6">
      <%= f.text_field(:uri, label: "File Name", disabled: true) %>
    </div>
    <div class="col-sm-6">
      <%= content_tag(:div, "&#8203;".html_safe, class: "mb-2 d-none d-sm-block") %>
      <%= f.file_field(:uri, hide_label: true, label: "URI") %>
    </div>
  </div>
  <div class="row">
    <div class="col">
      <%= link_to "Cancel", :back, class: "btn btn-primary" %>
    </div>
    <div class="col">
      <%= f.primary "Upload", class: "btn btn-primary float-right", disabled: true %>
    </div>
  </div>
<% end %>

The text field named uri (never mind why I called it that) actually is the file name of the file that was uploaded. If you were uploading the document with an AJAX request you'd have to add some JavaScript to the solution. In my case, it was simply doing a round trip to the server, so when the page re-rendered, the file name was shown.

I don't have any examples where I was uploading multiple files and showing a count on the screen. Sorry.

I hope this helps.

lcreid avatar Mar 28 '19 15:03 lcreid

Thank you for being so helpful!

iraszl avatar Mar 29 '19 05:03 iraszl

@iraszl Here's a naive way of showing the name of the file immediately after the file has been selected. No need for custom fields.

// Workaround for displaying selected file
$(document).on('ready turbolinks:load', function() {
  $('.custom-file-input').change(function(){
    $('.custom-file-label').text(this.value);
  });
});

This works for my file field:

<%= bootstrap_form_for @listing do |form| %>
...
 <%= form.file_field :image, direct_upload: true, accept: "image/png,image/gif,image/jpeg", label: "Add an image" %>
...
<% end %>

Before file has been selected: Screenshot 2019-04-15 at 16 09 06

After file has been selected: Screenshot 2019-04-15 at 16 35 46

codeandclay avatar Apr 15 '19 15:04 codeandclay

@codeandclay Thank you so much! I'm going to test this out ASAP!

iraszl avatar Apr 15 '19 15:04 iraszl

This is the recommended solution by the bootstrap folks

import bsCustomFileInput from 'bs-custom-file-input';
$(document).ready(function () {
  bsCustomFileInput.init();
})

hwhelchel avatar Apr 20 '19 03:04 hwhelchel

https://www.npmjs.com/package/bs-custom-file-input

hwhelchel avatar Apr 20 '19 03:04 hwhelchel

@codeandclay Solution works. @hwhelchel Doesn't seem to work. Do I need to add some class to the form to make it work?

iraszl avatar Apr 20 '19 08:04 iraszl

@iraszl If you're using Turbolinks (by default Rails does), perhaps @hwhelchel 's solution will work if you change the second line as follows:

import bsCustomFileInput from 'bs-custom-file-input';
$(document).on('ready turbolinks:load', function() {
  bsCustomFileInput.init();
})

You'll also have to make sure you include the package. Follow the link in this comment for instructions.

lcreid avatar Apr 20 '19 15:04 lcreid

This is an elegant solution. At the moment, bootstrap_form doesn't ship any JavaScript. Do you think it's good enough to document this solution in the README, rather than change the code?

lcreid avatar Apr 20 '19 15:04 lcreid

@lcreid Works great. But note that this line is not needed in the javascript if you load the library: import bsCustomFileInput from 'bs-custom-file-input'; Can you confirm if I'm correct or not?

iraszl avatar Apr 21 '19 05:04 iraszl

FWIW, while this issue is open, you can solve the issue by sprinkling a bit of script into your form:

<script type="application/javascript">
    $('input[type="file"]').change(function(e){
        var fileName = e.target.files[0].name;
        $('.custom-file-label').html(fileName);
    });
</script>

Halloran avatar Sep 21 '19 03:09 Halloran

If you have multiple file fields, you could do something like this so they all don't get updated to show the same value:

$('.custom-file-input').change(function(e){
  var fileName = e.target.files[0].name;
  $(`.custom-file-label[for=${e.currentTarget['id']}]`).html(fileName);
});

davidray avatar Sep 23 '19 01:09 davidray

@davidray nice catch. Your solution works both when multiple: true and multiple: false Thanks

Halloran avatar Sep 23 '19 12:09 Halloran

Here is the solution that worked for me on a Rails 6 app. A big thanks to @hwhelchel and @lcreid for their comments above. From the terminal run:

yarn add bs-custom-file-input

Add the following to your project's javascript/packs/application.js file:

import bsCustomFileInput from 'bs-custom-file-input';
$(document).on('ready turbolinks:load', function() {
  bsCustomFileInput.init();
})

Use one of these in your view:

<%= f.file_field(:image) %>
<%= f.file_field(:images, multiple: true, placeholder: "Choose files") %>

barriault avatar Jan 22 '20 22:01 barriault

For novices like me and who work just with plain JS (in case you use turbolinks) [based on the solution by @codeandclay]:

Add this function to app/assets/javascript/custom/[file_name].js:

document.addEventListener('turbolinks:load', function() {
  var fileInput = document.querySelector('.custom-file-input');
  var fileLabel = document.querySelector('.custom-file-label');
  fileInput.addEventListener("change", (e) => {fileLabel.textContent = e.target.value.replace(/^.*[\\\/]/, ''); });
}, false);

And in your file_field you don't need to change a thing

<%= bootstrap_form_for(@project, local: true) do |form| %>
    <%= form.file_field :attachments, label_class: "text-muted", label: "Add attachment"  %>
    <%= form.submit %>
<% end %>

After file was selected you'll see this:

Screen Shot 2020-07-07 at 10 13 20 PM

anastaszi avatar Jul 08 '20 05:07 anastaszi