A suite to track Project Diva score statistics and ratings / D4DJ event data.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
projectdivar/server/node_modules/get-pixels/node-pixels.js

194 lines
4.3 KiB

'use strict'
var ndarray = require('ndarray')
var path = require('path')
var PNG = require('pngjs').PNG
var jpeg = require('jpeg-js')
var pack = require('ndarray-pack')
var GifReader = require('omggif').GifReader
var Bitmap = require('node-bitmap')
var fs = require('fs')
var request = require('request')
var mime = require('mime-types')
var parseDataURI = require('parse-data-uri')
function handlePNG(data, cb) {
var png = new PNG();
png.parse(data, function(err, img_data) {
if(err) {
cb(err)
return
}
cb(null, ndarray(new Uint8Array(img_data.data),
[img_data.width|0, img_data.height|0, 4],
[4, 4*img_data.width|0, 1],
0))
})
}
function handleJPEG(data, cb) {
var jpegData
try {
jpegData = jpeg.decode(data)
}
catch(e) {
cb(e)
return
}
if(!jpegData) {
cb(new Error("Error decoding jpeg"))
return
}
var nshape = [ jpegData.height, jpegData.width, 4 ]
var result = ndarray(jpegData.data, nshape)
cb(null, result.transpose(1,0))
}
function handleGIF(data, cb) {
var reader
try {
reader = new GifReader(data)
} catch(err) {
cb(err)
return
}
if(reader.numFrames() > 0) {
var nshape = [reader.numFrames(), reader.height, reader.width, 4]
try {
var ndata = new Uint8Array(nshape[0] * nshape[1] * nshape[2] * nshape[3])
} catch(err) {
cb(err)
return
}
var result = ndarray(ndata, nshape)
try {
for(var i=0; i<reader.numFrames(); ++i) {
reader.decodeAndBlitFrameRGBA(i, ndata.subarray(
result.index(i, 0, 0, 0),
result.index(i+1, 0, 0, 0)))
}
} catch(err) {
cb(err)
return
}
cb(null, result.transpose(0,2,1))
} else {
var nshape = [reader.height, reader.width, 4]
var ndata = new Uint8Array(nshape[0] * nshape[1] * nshape[2])
var result = ndarray(ndata, nshape)
try {
reader.decodeAndBlitFrameRGBA(0, ndata)
} catch(err) {
cb(err)
return
}
cb(null, result.transpose(1,0))
}
}
function handleBMP(data, cb) {
var bmp = new Bitmap(data)
try {
bmp.init()
} catch(e) {
cb(e)
return
}
var bmpData = bmp.getData()
var nshape = [ bmpData.getHeight(), bmpData.getWidth(), 4 ]
var ndata = new Uint8Array(nshape[0] * nshape[1] * nshape[2])
var result = ndarray(ndata, nshape)
pack(bmpData, result)
cb(null, result.transpose(1,0))
}
function doParse(mimeType, data, cb) {
switch(mimeType) {
case 'image/png':
handlePNG(data, cb)
break
case 'image/jpg':
case 'image/jpeg':
handleJPEG(data, cb)
break
case 'image/gif':
handleGIF(data, cb)
break
case 'image/bmp':
handleBMP(data, cb)
break
default:
cb(new Error("Unsupported file type: " + mimeType))
}
}
module.exports = function getPixels(url, type, cb) {
if(!cb) {
cb = type
type = ''
}
if(Buffer.isBuffer(url)) {
if(!type) {
cb(new Error('Invalid file type'))
return
}
doParse(type, url, cb)
} else if(url.indexOf('data:') === 0) {
try {
var buffer = parseDataURI(url)
if(buffer) {
process.nextTick(function() {
doParse(type || buffer.mimeType, buffer.data, cb)
})
} else {
process.nextTick(function() {
cb(new Error('Error parsing data URI'))
})
}
} catch(err) {
process.nextTick(function() {
cb(err)
})
}
} else if(url.indexOf('http://') === 0 || url.indexOf('https://') === 0) {
request({url:url, encoding:null}, function(err, response, body) {
if(err) {
cb(err)
return
}
type = type;
if(!type){
if(response.getHeader !== undefined){
type = response.getHeader('content-type');
}else if(response.headers !== undefined){
type = response.headers['content-type'];
}
}
if(!type) {
cb(new Error('Invalid content-type'))
return
}
doParse(type, body, cb)
})
} else {
fs.readFile(url, function(err, data) {
if(err) {
cb(err)
return
}
type = type || mime.lookup(url)
if(!type) {
cb(new Error('Invalid file type'))
return
}
doParse(type, data, cb)
})
}
}