-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathStreamSaver.js
120 lines (105 loc) · 3.49 KB
/
StreamSaver.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
;((name, definition) => {
'undefined' != typeof module ? module.exports = definition() :
'function' == typeof define && 'object' == typeof define.amd ? define(definition) :
this[name] = definition()
})('streamSaver', () => {
'use strict'
let
iframe, loaded,
secure = location.protocol == 'https:' || location.hostname == 'localhost',
streamSaver = {
createWriteStream,
supported: false,
version: {
full: '1.0.0',
major: 1, minor: 0, dot: 0
}
}
streamSaver.mitm = 'https://jimmywarting.github.io/StreamSaver.js/mitm.html?version=' +
streamSaver.version.full
try {
// Some browser has it but ain't allowed to construct a stream yet
streamSaver.supported = 'serviceWorker' in navigator && !!new ReadableStream() && !!new WritableStream()
} catch(err) {
// if you are running chrome < 52 then you can enable it
// `chrome://flags/#enable-experimental-web-platform-features`
}
function createWriteStream(filename, queuingStrategy, size) {
// normalize arguments
if (Number.isFinite(queuingStrategy))
[size, queuingStrategy] = [queuingStrategy, size]
let channel = new MessageChannel,
popup,
setupChannel = () => new Promise((resolve, reject) => {
channel.port1.onmessage = evt => {
if(evt.data.download) {
resolve()
if(!secure) popup.close() // don't need the popup any longer
let link = document.createElement('a')
let click = new MouseEvent('click')
link.href = evt.data.download
link.dispatchEvent(click)
}
}
if(secure && !iframe) {
iframe = document.createElement('iframe')
iframe.src = streamSaver.mitm
iframe.hidden = true
document.body.appendChild(iframe)
}
if(secure && !loaded) {
let fn;
iframe.addEventListener('load', fn = evt => {
loaded = true
iframe.removeEventListener('load', fn)
iframe.contentWindow.postMessage(
{filename, size}, '*', [channel.port2])
})
}
if(secure && loaded) {
iframe.contentWindow.postMessage({filename, size}, '*', [channel.port2])
}
if(!secure) {
popup = window.open(streamSaver.mitm, Math.random())
let onready = evt => {
if(evt.source === popup){
popup.postMessage({filename, size}, '*', [channel.port2])
removeEventListener('message', onready)
}
}
// Another problem that cross origin don't allow is scripting
// so popup.onload() don't work but postMessage still dose
// work cross origin
addEventListener('message', onready)
}
})
return new WritableStream({
start(error) {
// is called immediately, and should perform any actions
// necessary to acquire access to the underlying sink.
// If this process is asynchronous, it can return a promise
// to signal success or failure.
return setupChannel()
},
write(chunk) {
// is called when a new chunk of data is ready to be written
// to the underlying sink. It can return a promise to signal
// success or failure of the write operation. The stream
// implementation guarantees that this method will be called
// only after previous writes have succeeded, and never after
// close or abort is called.
// TODO: Kind of important that service worker respond back when
// it has been written. Otherwise we can't handle backpressure
channel.port1.postMessage(chunk)
},
close() {
channel.port1.postMessage('end')
console.log('All data successfully read!')
},
abort(e) {
channel.port1.postMessage('abort')
}
}, queuingStrategy)
}
return streamSaver
})