7

I'm aware of the security reasons that local file cannot be set as a file programmatically to input field. Suppose I've this image as base64:

var bus = "";

I want to set it to an input type file programmatically. Please do not suggest to send this as a string to server because I cannot control the API at the other end.

I tried to convert it to blob in my code and set it to input field as:

myInputField.value = blob;

but it throws security exception:

Uncaught DOMException: Failed to set the 'value' property on 'HTMLInputElement': This input element accepts a filename, which may only be programmatically set to the empty string.

Is it possible via javascript? I, by no means, want to involve the file chooser dialog box.

Thank you.

1
  • 1
    Is it possible via javascript? - well, the error states which may only be programmatically set to the empty string - so, the conclusion is ... Commented Mar 14, 2017 at 4:36

3 Answers 3

6

Not possible. An HTML file input can only point to a file that actually exists on the computer -- it can't point to an arbitrary chunk of data.

Sign up to request clarification or add additional context in comments.

3 Comments

That's not entirely true, IIRC, on windows, you have/had the ability to enter an URL in the file upload popup's field, I can't test right now if you can pass dataURI too, but anyway.
So I tested it through a VM, and it only accepts URL, not dataURI, but still it's not necessarily on user's machine.
Hijacking accepted answer since this is where Google led me first, it's possible now: stackoverflow.com/questions/47119426/…
2

No you can't set the value of a file input*, except to clear it, but it sounds that it's not what you need anyway.
Instead, convert this dataURI to a Blob, then append this Blob to a FormData and finally post this FormData.

Your image will be sent as multipart like you seem to need.

function dataURItoBlob(data) {
  var binStr = atob(data).split(',')[1],
   len = binStr.length,
   arr = new Uint8Array(len);

  for (var i = 0; i < len; i++) {
    arr[i] = binStr.charCodeAt(i);
  }
  return new Blob(arr);
}

var dataURI = 'data:image/....';
var blob = dataURItoBlob(dataURI);

// you can even pass a <form> in this constructor to add other fields
var formData = new FormData(); 
formData.append('yourFileField', blob, 'fileName.ext');

var xhr = new XMLHttpRequest();
xhr.open('post', yourServer);
xhr.send(formData);

// now you can retrieve your image as a File/multipart with the 'yourFileField' post name

*Actually now you can do it, but it's still rather hackish.

2 Comments

I'm working on an API, rather. I just have to change the value of the input type file and the listeners will fire up. Moreover, the communication of file is going over a secured websocket whose variable reference is not available to me. So this isn't gonna help.
@MehulMohan You can't. Modify your API so that it accepts you pass directly a Blob, instead of reading a file input's files property, or ultimately, you could probably pass an object like {files:[blob]} instead of the file input, since I guess your API is only reading for it. (It may also want some name property, so in this case, just add it to the resulting Blob object, or create a new File(blob, name), but for websocket, you can just do socket.send(blob).
1

You can try like below one. Convert your sting to Blob and create file type with created Blob.

var debug = {hello: "world"};
var blob = new Blob([JSON.stringify(debug, null, 2)], {type : 'application/json'});

var outputfile=new File([blob], "filename")

1 Comment

That won't change the fact that you cannot set an file input's value. Also, a File object is just a Blob with a name and a lastModified property. All you can do with a File can be done with a Blob.

Your Answer

By clicking “Post Your Answer”, you agree to our terms of service and acknowledge you have read our privacy policy.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.