2
* A saveAs() FileSaver implementation.
5
* By Eli Grey, http://eligrey.com
7
* See https://github.com/eligrey/FileSaver.js/blob/master/LICENSE.md
11
/*jslint bitwise: true, indent: 4, laxbreak: true, laxcomma: true, smarttabs: true, plusplus: true */
13
/*! @source http://purl.eligrey.com/github/FileSaver.js/blob/master/FileSaver.js */
15
var saveAs = saveAs || (function(view) {
17
// IE <10 is explicitly unsupported
18
if (typeof navigator !== "undefined" && /MSIE [1-9]\./.test(navigator.userAgent)) {
23
// only get URL when necessary in case Blob.js hasn't overridden it yet
24
, get_URL = function() {
25
return view.URL || view.webkitURL || view;
27
, save_link = doc.createElementNS("http://www.w3.org/1999/xhtml", "a")
28
, can_use_save_link = "download" in save_link
29
, click = function(node) {
30
var event = new MouseEvent("click");
31
node.dispatchEvent(event);
33
, is_safari = /Version\/[\d\.]+.*Safari/.test(navigator.userAgent)
34
, webkit_req_fs = view.webkitRequestFileSystem
35
, req_fs = view.requestFileSystem || webkit_req_fs || view.mozRequestFileSystem
36
, throw_outside = function(ex) {
37
(view.setImmediate || view.setTimeout)(function() {
41
, force_saveable_type = "application/octet-stream"
43
// See https://code.google.com/p/chromium/issues/detail?id=375297#c7 and
44
// https://github.com/eligrey/FileSaver.js/commit/485930a#commitcomment-8768047
45
// for the reasoning behind the timeout and revocation flow
46
, arbitrary_revoke_timeout = 500 // in ms
47
, revoke = function(file) {
48
var revoker = function() {
49
if (typeof file === "string") { // file is an object URL
50
get_URL().revokeObjectURL(file);
51
} else { // file is a File
58
setTimeout(revoker, arbitrary_revoke_timeout);
61
, dispatch = function(filesaver, event_types, event) {
62
event_types = [].concat(event_types);
63
var i = event_types.length;
65
var listener = filesaver["on" + event_types[i]];
66
if (typeof listener === "function") {
68
listener.call(filesaver, event || filesaver);
75
, auto_bom = function(blob) {
76
// prepend BOM for UTF-8 XML and text/* types (including HTML)
77
if (/^\s*(?:text\/\S*|application\/xml|\S*\/\S*\+xml)\s*;.*charset\s*=\s*utf-8/i.test(blob.type)) {
78
return new Blob(["\ufeff", blob], {type: blob.type});
82
, FileSaver = function(blob, name, no_auto_bom) {
84
blob = auto_bom(blob);
86
// First try a.download, then web filesystem, then object URLs
90
, blob_changed = false
93
, dispatch_all = function() {
94
dispatch(filesaver, "writestart progress write writeend".split(" "));
96
// on any filesys errors revert to saving with object URLs
97
, fs_error = function() {
98
if (target_view && is_safari && typeof FileReader !== "undefined") {
99
// Safari doesn't allow downloading of blob urls
100
var reader = new FileReader();
101
reader.onloadend = function() {
102
var base64Data = reader.result;
103
target_view.location.href = "data:attachment/file" + base64Data.slice(base64Data.search(/[,;]/));
104
filesaver.readyState = filesaver.DONE;
107
reader.readAsDataURL(blob);
108
filesaver.readyState = filesaver.INIT;
111
// don't create more object URLs than needed
112
if (blob_changed || !object_url) {
113
object_url = get_URL().createObjectURL(blob);
116
target_view.location.href = object_url;
118
var new_tab = view.open(object_url, "_blank");
119
if (new_tab == undefined && is_safari) {
120
//Apple do not allow window.open, see http://bit.ly/1kZffRI
121
view.location.href = object_url
124
filesaver.readyState = filesaver.DONE;
128
, abortable = function(func) {
130
if (filesaver.readyState !== filesaver.DONE) {
131
return func.apply(this, arguments);
135
, create_if_not_found = {create: true, exclusive: false}
138
filesaver.readyState = filesaver.INIT;
142
if (can_use_save_link) {
143
object_url = get_URL().createObjectURL(blob);
144
setTimeout(function() {
145
save_link.href = object_url;
146
save_link.download = name;
150
filesaver.readyState = filesaver.DONE;
154
// Object and web filesystem URLs have a problem saving in Google Chrome when
155
// viewed in a tab, so I force save with application/octet-stream
156
// http://code.google.com/p/chromium/issues/detail?id=91158
157
// Update: Google errantly closed 91158, I submitted it again:
158
// https://code.google.com/p/chromium/issues/detail?id=389642
159
if (view.chrome && type && type !== force_saveable_type) {
160
slice = blob.slice || blob.webkitSlice;
161
blob = slice.call(blob, 0, blob.size, force_saveable_type);
164
// Since I can't be sure that the guessed media type will trigger a download
165
// in WebKit, I append .download to the filename.
166
// https://bugs.webkit.org/show_bug.cgi?id=65440
167
if (webkit_req_fs && name !== "download") {
170
if (type === force_saveable_type || webkit_req_fs) {
177
fs_min_size += blob.size;
178
req_fs(view.TEMPORARY, fs_min_size, abortable(function(fs) {
179
fs.root.getDirectory("saved", create_if_not_found, abortable(function(dir) {
180
var save = function() {
181
dir.getFile(name, create_if_not_found, abortable(function(file) {
182
file.createWriter(abortable(function(writer) {
183
writer.onwriteend = function(event) {
184
target_view.location.href = file.toURL();
185
filesaver.readyState = filesaver.DONE;
186
dispatch(filesaver, "writeend", event);
189
writer.onerror = function() {
190
var error = writer.error;
191
if (error.code !== error.ABORT_ERR) {
195
"writestart progress write abort".split(" ").forEach(function(event) {
196
writer["on" + event] = filesaver["on" + event];
199
filesaver.abort = function() {
201
filesaver.readyState = filesaver.DONE;
203
filesaver.readyState = filesaver.WRITING;
207
dir.getFile(name, {create: false}, abortable(function(file) {
208
// delete file if it already exists
211
}), abortable(function(ex) {
212
if (ex.code === ex.NOT_FOUND_ERR) {
221
, FS_proto = FileSaver.prototype
222
, saveAs = function(blob, name, no_auto_bom) {
223
return new FileSaver(blob, name, no_auto_bom);
226
// IE 10+ (native saveAs)
227
if (typeof navigator !== "undefined" && navigator.msSaveOrOpenBlob) {
228
return function(blob, name, no_auto_bom) {
230
blob = auto_bom(blob);
232
return navigator.msSaveOrOpenBlob(blob, name || "download");
236
FS_proto.abort = function() {
237
var filesaver = this;
238
filesaver.readyState = filesaver.DONE;
239
dispatch(filesaver, "abort");
241
FS_proto.readyState = FS_proto.INIT = 0;
242
FS_proto.WRITING = 1;
246
FS_proto.onwritestart =
247
FS_proto.onprogress =
251
FS_proto.onwriteend =
256
typeof self !== "undefined" && self
257
|| typeof window !== "undefined" && window
260
// `self` is undefined in Firefox for Android content script context
261
// while `this` is nsIContentFrameMessageManager
262
// with an attribute `content` that corresponds to the window
264
if (typeof module !== "undefined" && module.exports) {
265
module.exports.saveAs = saveAs;
266
} else if ((typeof define !== "undefined" && define !== null) && (define.amd != null)) {
267
define([], function() {