fix(encoder): remove nested worker

Nested worker is not supported on safari
This commit is contained in:
wryk 2019-03-15 12:59:24 +01:00
parent 57d4a8dd52
commit 388a1f4bb5
4 changed files with 96 additions and 117 deletions

View file

@ -22,7 +22,7 @@ class OutputStream {
} }
} }
export function write (indexedColorImages, imageWidth, imageHeight, delay) { export function write (imageWidth, imageHeight, delay, indexedColorImages) {
const emitter = new EventEmitter() const emitter = new EventEmitter()
// yup, this is the browser nextTick implementation we are waiting for :facepalm: // yup, this is the browser nextTick implementation we are waiting for :facepalm:

View file

@ -1,15 +1,55 @@
import EventEmitter from 'eventemitter3' import EventEmitter from 'eventemitter3'
import genericPool from 'generic-pool'
import pEvent from 'p-event'
import { write } from '/services/encode-core.js'
import { promisesProgress, calcProgress } from '/services/util.js'
const PALETTE_SIZE = 255 const PALETTE_SIZE = 255
export function encode ({ imageDataList, imageWidth, imageHeight, delayTime }) { export function encode ({ imageDataList, imageWidth, imageHeight, delayTime }) {
const emitter = new EventEmitter() const emitter = new EventEmitter()
const worker = new Worker('/services/encode.worker.js')
worker.onerror = error => emitter.emit('error', error) const quantizeColorWorkerPool = genericPool.createPool({
worker.onmessageerror = error => emitter.emit('error', error) create: () => new Worker('/services/quantize-color.worker.js'),
destroy: worker => worker.terminate()
}, {
min: 0,
max: 2
})
const indexedColorImagesP = imageDataList
.map(async imageData => {
const worker = await quantizeColorWorkerPool.acquire()
const indexedColorImage = await new Promise((resolve, reject) => {
worker.onerror = reject
worker.onmessageerror = reject
worker.onmessage = event => { worker.onmessage = event => {
resolve(event.data)
}
worker.postMessage({
imageData,
paletteSize: PALETTE_SIZE
})
})
await quantizeColorWorkerPool.release(worker)
return indexedColorImage
})
const progressPromises = promisesProgress(indexedColorImagesP, function (value) {
emitter.emit('progress', calcProgress(0, 0.9, value))
})
Promise.all(progressPromises).then(indexedColorImages => {
const writeWorker = new Worker('/services/write-gif.worker.js')
writeWorker.onerror = error => emitter.emit('error', error)
writeWorker.onmessageerror = error => emitter.emit('error', error)
writeWorker.onmessage = event => {
const { type, payload } = event.data const { type, payload } = event.data
switch (type) { switch (type) {
@ -18,7 +58,7 @@ export function encode ({ imageDataList, imageWidth, imageHeight, delayTime }) {
break break
case 'progress': case 'progress':
emitter.emit('progress', payload.value) emitter.emit('progress', calcProgress(0.9, 1, payload.value))
break break
case 'done': case 'done':
@ -30,13 +70,14 @@ export function encode ({ imageDataList, imageWidth, imageHeight, delayTime }) {
} }
} }
worker.postMessage({ writeWorker.postMessage({
imageDataList,
imageWidth, imageWidth,
imageHeight, imageHeight,
paletteSize: PALETTE_SIZE, indexedColorImages,
delayTime delayTime
}) })
})
.catch(error => emitter.emit('error', error))
return emitter return emitter
} }

View file

@ -1,85 +0,0 @@
import genericPool from 'generic-pool'
import pEvent from 'p-event'
import { write } from '/services/encode-core.js'
import { promisesProgress, calcProgress } from '/services/util.js'
onmessage = handleMessage
const workerPool = genericPool.createPool({
create () {
return new Worker('/services/quantize-color.worker.js')
},
destroy (worker) {
worker.terminate()
}
}, {
min: 0,
max: 4
})
async function handleMessage (event) {
const { imageDataList, imageWidth, imageHeight, paletteSize, delayTime } = event.data
postProgressMessage(0)
console.time('quantization')
const promises = imageDataList
.map(async imageData => {
const worker = await workerPool.acquire()
const indexedColorImage = await new Promise((resolve, reject) => {
worker.onerror = reject
worker.onmessageerror = reject
worker.onmessage = event => {
resolve(event.data)
}
worker.postMessage({
imageData,
paletteSize
})
})
await workerPool.release(worker)
return indexedColorImage
})
const progressPromises = promisesProgress(promises, function (value) {
postProgressMessage(calcProgress(0, 0.9, value))
})
const indexedColorImages = await Promise.all(progressPromises)
console.timeEnd('quantization')
console.time('write')
const writing = write(indexedColorImages, imageWidth, imageHeight, delayTime)
writing.on('progress', value => postProgressMessage(calcProgress(0.9, 1, value)))
const buffer = await pEvent(writing, 'done')
console.timeEnd('write')
postDoneMessage(buffer)
}
function postProgressMessage (value) {
postMessage({
type: 'progress',
payload: {
value
}
})
}
function postDoneMessage (buffer) {
postMessage({
type: 'done',
payload: {
buffer
}
})
}

View file

@ -0,0 +1,23 @@
import { write } from '/services/encode-core.js'
onmessage = handleMessage
async function handleMessage (event) {
const { imageWidth, imageHeight, delayTime, indexedColorImages } = event.data
const writing = write(imageWidth, imageHeight, delayTime, indexedColorImages)
writing.on('progress', value => postMessage({
type: 'progress',
payload: {
value
}
}))
writing.once('done', buffer => postMessage({
type: 'done',
payload: {
buffer
}
}))
}