tonic-image-builder
Advanced tools
Comparing version
@@ -9,2 +9,6 @@ var CanvasOffscreenBuffer = require('../../util/CanvasOffscreenBuffer'), | ||
export default function CompositeImageBuilder(queryDataModel, pipelineModel) { | ||
// State flags that let UI present the proper controls | ||
this.handleRecord = true; | ||
this.handleExploration = true; | ||
this.queryDataModel = queryDataModel; | ||
@@ -185,7 +189,14 @@ this.pipelineModel = pipelineModel; | ||
outputSize: [width, height], | ||
builder: this | ||
builder: this, | ||
arguments: this.queryDataModel.getQuery() | ||
}; | ||
// Add pipeline info + ts | ||
readyImage.arguments.pipeline = this.query; | ||
// Let everyone know the image is ready | ||
this.emit(IMAGE_READY_TOPIC, readyImage); | ||
// In case of exploration trigger next data fetch | ||
this.queryDataModel.nextExploration(); | ||
}; | ||
@@ -192,0 +203,0 @@ |
@@ -80,3 +80,3 @@ var CanvasOffscreenBuffer = require('../../util/CanvasOffscreenBuffer'), | ||
// Update LookupTableManager with data range | ||
this.lookupTableManager.addFields(this.metadata.ranges); | ||
this.lookupTableManager.addFields(this.metadata.ranges, this.queryDataModel.originalData.LookupTables); | ||
this.lutActiveSubscription = this.lookupTableManager.onActiveLookupTableChange( (data, envelope) => { | ||
@@ -83,0 +83,0 @@ if(this.getField() !== data) { |
@@ -6,3 +6,4 @@ var CanvasOffscreenBuffer = require('../../util/CanvasOffscreenBuffer'), | ||
PROBE_CHANGE_TOPIC = 'probe-change', | ||
TIME_DATA_READY = 'time-data-ready'; | ||
TIME_DATA_READY = 'time-data-ready', | ||
contains = require('mout/src/array/contains'); | ||
@@ -12,2 +13,6 @@ // ---------------------------------------------------------------------------- | ||
export default function FloatImageImageBuilder(queryDataModel, lutManager) { | ||
// State flags that let UI present the proper controls | ||
this.handleRecord = true; | ||
this.handleExploration = false; | ||
this.queryDataModel = queryDataModel; | ||
@@ -46,3 +51,3 @@ this.timeDataQueryDataModel = queryDataModel.clone(); | ||
// Update LookupTableManager with data range | ||
this.lookupTableManager.addFields(this.metadata.ranges); | ||
this.lookupTableManager.addFields(this.metadata.ranges, this.queryDataModel.originalData.LookupTables); | ||
@@ -324,5 +329,8 @@ // Handle events | ||
outputSize: [width, height], | ||
builder: this | ||
builder: this, | ||
arguments: this.queryDataModel.getQuery() | ||
}; | ||
// FIXME should add var for pipeline | ||
// Let everyone know the image is ready | ||
@@ -383,2 +391,8 @@ this.emit(IMAGE_READY_TOPIC, readyImage); | ||
FloatImageImageBuilder.prototype.isMultiView = function(){ | ||
return !contains(this.queryDataModel.originalData.type, 'single-view'); | ||
}; | ||
// ---------------------------------------------------------------------------- | ||
FloatImageImageBuilder.prototype.getLookupTableManager = function() { | ||
@@ -385,0 +399,0 @@ return this.lookupTableManager; |
@@ -51,3 +51,4 @@ var CanvasOffscreenBuffer = require('../../util/CanvasOffscreenBuffer'), | ||
// Update LookupTableManager with data range | ||
this.lookupTableManager.addFields(this.metadata.ranges); | ||
this.lookupTableManager.updateActiveLookupTable(this.field); | ||
this.lookupTableManager.addFields(this.metadata.ranges, this.queryDataModel.originalData.LookupTables); | ||
@@ -280,5 +281,5 @@ var maxSize = 0; | ||
if(axisIdx === 1) { | ||
var offset = px*xSize + pz*xSize*ySize; | ||
var offset = px + pz*xSize*ySize; | ||
for(var y = 0; y < ySize; y++) { | ||
idxValues.push(offset + y*xSize); | ||
idxValues.push(offset + (ySize - y - 1)*xSize); | ||
} | ||
@@ -288,6 +289,6 @@ idxValues.reverse(); | ||
if(axisIdx === 2) { | ||
var offset = px*xSize + (ySize - py - 1)*xSize*ySize, | ||
var offset = px + (ySize - py - 1)*xSize, | ||
step = xSize*ySize; | ||
for(var x = 0; x < xSize; x++) { | ||
idxValues.push(offset + x*step); | ||
for(var z = 0; z < zSize; z++) { | ||
idxValues.push(offset + z*step); | ||
} | ||
@@ -300,3 +301,4 @@ } | ||
var array = this.dataFields[name], | ||
data = []; | ||
data = [], | ||
range = this.lookupTableManager.getLookupTable(name).getScalarRange(); | ||
@@ -307,3 +309,3 @@ for(var i = 0; i < dataSize; i++) { | ||
probeData.fields.push({name, data}); | ||
probeData.fields.push({name, data, range}); | ||
}); | ||
@@ -310,0 +312,0 @@ |
@@ -105,3 +105,3 @@ var CanvasOffscreenBuffer = require('../../util/CanvasOffscreenBuffer'), | ||
this.lookupTableManager = lookupTableManager; | ||
this.lookupTableManager.addFields(this.compositePipeline.ranges); | ||
this.lookupTableManager.addFields(this.compositePipeline.ranges, this.queryDataModel.originalData.LookupTables); | ||
@@ -108,0 +108,0 @@ this.numLutSamples = 1024; |
@@ -5,2 +5,3 @@ var Monologue = require('monologue.js'), | ||
IMAGE_READY_TOPIC = 'image-ready', | ||
COLOR_CHANGE_TOPIC = 'color-change', | ||
LUT_NAME = 'VolumeScalar'; | ||
@@ -12,2 +13,3 @@ | ||
this.opacities = []; | ||
this.colors = []; | ||
this.eqCallback = null; | ||
@@ -18,3 +20,3 @@ this.useIntensity = false; | ||
this.originalRange = [ this.metadata.scalars[0], this.metadata.scalars[ this.metadata.scalars.length - 1]]; | ||
this.lutTextureData = new Uint8Array(this.metadata.layers * 3); | ||
this.lutTextureData = new Uint8Array(this.metadata.layers * 4); | ||
this.dataQuery = { name: 'data_fetch', categories: [] }; | ||
@@ -32,10 +34,16 @@ | ||
// Add Lut | ||
this.lookupTable = lookupTableManager.addLookupTable(LUT_NAME, [0, 1]); | ||
lookupTableManager.addFields({ VolumeScalar: [0, 1] }, this.queryDataModel.originalData.LookupTables); | ||
this.lookupTable = lookupTableManager.getLookupTable(LUT_NAME); | ||
this.lutChangeSubscription = this.lookupTable.onChange( (data, envelope) => { | ||
this.colors = []; | ||
for(var idx = 0; idx < this.metadata.layers; idx++) { | ||
var color = this.lookupTable.getColor(this.metadata.scalars[idx]); | ||
this.lutTextureData[idx*3] = color[0] * 255; | ||
this.lutTextureData[idx*3+1] = color[1] * 255; | ||
this.lutTextureData[idx*3+2] = color[2] * 255; | ||
this.lutTextureData[idx*4] = color[0] * 255; | ||
this.lutTextureData[idx*4+1] = color[1] * 255; | ||
this.lutTextureData[idx*4+2] = color[2] * 255; | ||
this.colors.push('rgb(' + [this.lutTextureData[idx*4], this.lutTextureData[idx*4 + 1], this.lutTextureData[idx*4 + 2]].join(',') + ')'); | ||
} | ||
this.emit(COLOR_CHANGE_TOPIC); | ||
this.render(); | ||
@@ -77,6 +85,5 @@ }); | ||
this.compositors.forEach((c) => { | ||
c.updateOpacities(this.opacities); | ||
}); | ||
for(var idx = 0; idx < this.metadata.layers; idx++) { | ||
this.lutTextureData[idx*4 + 3] = array[idx] * 255; | ||
} | ||
this.render(); | ||
@@ -120,3 +127,8 @@ } | ||
// -------------------------------------------------------------------------- | ||
WebGLSortedVolumeImageBuilder.prototype.onColorChange = function(callback) { | ||
return this.on(COLOR_CHANGE_TOPIC, callback); | ||
} | ||
// -------------------------------------------------------------------------- | ||
@@ -169,5 +181,3 @@ | ||
this.compositors.forEach((c) => { | ||
c.updateOpacities(this.opacities); | ||
}); | ||
this.updateOpacities(this.opacities); | ||
} | ||
@@ -218,1 +228,7 @@ | ||
// -------------------------------------------------------------------------- | ||
WebGLSortedVolumeImageBuilder.prototype.getColors = function() { | ||
return this.colors; | ||
} | ||
@@ -11,6 +11,2 @@ var CanvasOffscreenBuffer = require('../../util/CanvasOffscreenBuffer'); | ||
this.numLayers = this.metadata.layers; | ||
var opacityArrayBuffer = new ArrayBuffer(this.numLayers); | ||
this.alphaMultiplier = new Uint8ClampedArray(opacityArrayBuffer); | ||
this.colorTable = colorTable; | ||
@@ -38,11 +34,2 @@ | ||
SortedVolumeCompositor.prototype.updateOpacities = function(updatedOpacities) { | ||
var layerIdx = this.numLayers; | ||
while (layerIdx--) { | ||
this.alphaMultiplier[layerIdx] = updatedOpacities[layerIdx] * 255.0; | ||
} | ||
}; | ||
// -------------------------------------------------------------------------- | ||
SortedVolumeCompositor.prototype.setLayerColors = function(colorTable) { | ||
@@ -63,3 +50,2 @@ this.colorTable = colorTable; | ||
width = this.width, | ||
drawIdx = this.numLayers, | ||
ctx = this.bgCanvas.get2DContext(); | ||
@@ -78,3 +64,3 @@ | ||
// Just iterate through all the layers in the data for now | ||
while (drawIdx--) { | ||
for (var drawIdx = 0; drawIdx < this.numLayers; drawIdx++) { | ||
for(var y = 0; y < this.height; y++) { | ||
@@ -85,8 +71,8 @@ for(var x = 0; x < this.width; x++) { | ||
layerIdx = this.orderData[drawIdx * imageSize + idx], | ||
multiplier = this.alphaMultiplier[layerIdx] / 255.0, | ||
alphB = this.alphaData[layerIdx * imageSize + idx] / 255.0, | ||
multiplier = this.colorTable[layerIdx*4 + 3] / 255.0, | ||
alphB = this.alphaData[drawIdx * imageSize + idx] / 255.0, | ||
intensity = 1.0; | ||
if(this.intensityData) { | ||
intensity = this.intensityData[layerIdx * imageSize + idx] / 255.0; | ||
intensity = this.intensityData[drawIdx * imageSize + idx] / 255.0; | ||
} | ||
@@ -99,5 +85,5 @@ | ||
rgbB = [ | ||
this.colorTable[layerIdx*3 ] * intensity * alphB * multiplier * alphANeg, | ||
this.colorTable[layerIdx*3+1] * intensity * alphB * multiplier * alphANeg, | ||
this.colorTable[layerIdx*3+2] * intensity * alphB * multiplier * alphANeg | ||
this.colorTable[layerIdx*4 ] * intensity * alphB * multiplier * alphANeg, | ||
this.colorTable[layerIdx*4+1] * intensity * alphB * multiplier * alphANeg, | ||
this.colorTable[layerIdx*4+2] * intensity * alphB * multiplier * alphANeg | ||
], | ||
@@ -131,2 +117,7 @@ alphOut = alphA + (alphB * multiplier * (1.0 - alphA)); | ||
SortedVolumeCompositor.prototype.destroy = function() { | ||
this.bgCanvas.destroy(); | ||
this.bgCanvas = null; | ||
this.queryDataModel = null; | ||
this.imageBuilder = null; | ||
} |
@@ -20,9 +20,5 @@ var CanvasOffscreenBuffer = require('../../util/CanvasOffscreenBuffer'), | ||
this.defaultIntensityData = new Uint8Array([255]); | ||
this.spriteSize = this.infoJson.SortedComposite.textures.alpha.size; | ||
this.numLayers = this.infoJson.SortedComposite.layers; | ||
var opacityArrayBuffer = new ArrayBuffer(this.numLayers); | ||
this.clampledOpacityView = new Uint8ClampedArray(opacityArrayBuffer); | ||
this.opacityView = new Uint8Array(opacityArrayBuffer); | ||
this.colorTable = colorTable; | ||
this.lutView = colorTable; | ||
@@ -87,7 +83,6 @@ this.width = this.infoJson.SortedComposite.dimensions[0]; | ||
textures: [ | ||
{ id: 'orderTexture', pixelStore: align1PixelStore, texParameter }, | ||
{ id: 'alphaTexture', pixelStore: align1PixelStore, texParameter }, | ||
{ id: 'intensityTexture', pixelStore: align1PixelStore, texParameter }, | ||
{ id: 'alphaMultiplierTexture', pixelStore: align1PixelStore, texParameter }, | ||
{ id: 'lutTexture', pixelStore: align1PixelStore, texParameter }, | ||
{ id: 'orderTexture', pixelStore: align1PixelStore, texParameter }, | ||
{ id: 'alphaTexture', pixelStore: align1PixelStore, texParameter }, | ||
{ id: 'intensityTexture', pixelStore: align1PixelStore, texParameter }, | ||
{ id: 'lutTexture', pixelStore, texParameter }, | ||
{ id: 'ping', pixelStore, texParameter }, | ||
@@ -135,9 +130,9 @@ { id: 'pong', pixelStore, texParameter }, | ||
this.orderData = data.order.data; | ||
this.alphaData = new Uint8Array(data.alpha.data); | ||
this.alphaData = data.alpha.data; | ||
if (data.intensity) { | ||
this.intensityData = new Uint8Array(data.intensity.data); | ||
this.intensitySize = this.spriteSize; | ||
this.intensityData = data.intensity.data; | ||
this.intensitySize = [ this.width, this.height ]; | ||
} else { | ||
this.intensityData = this.defaultIntensityData; | ||
this.intensitySize = 1; | ||
this.intensitySize = [ 1, 1 ]; | ||
} | ||
@@ -148,17 +143,2 @@ }; | ||
WebGLSortedVolumeCompositor.prototype.updateOpacities = function(updatedOpacities) { | ||
var layerIdx = this.numLayers; | ||
while (layerIdx--) { | ||
this.clampledOpacityView[layerIdx] = updatedOpacities[layerIdx] * 255.0; | ||
} | ||
}; | ||
// -------------------------------------------------------------------------- | ||
WebGLSortedVolumeCompositor.prototype.setLayerColors = function(colorTable) { | ||
this.colorTable = colorTable; | ||
}; | ||
// -------------------------------------------------------------------------- | ||
WebGLSortedVolumeCompositor.prototype.extractLayerData = function(buffer, layerIndex) { | ||
@@ -174,3 +154,3 @@ var offset = layerIndex * this.width * this.height, | ||
WebGLSortedVolumeCompositor.prototype.render = function() { | ||
if (!this.alphaData || !this.orderData || !this.colorTable) { | ||
if (!this.alphaData || !this.orderData || !this.lutView) { | ||
return null; | ||
@@ -185,3 +165,5 @@ } | ||
while (layerIdx--) { | ||
this.drawColorPass(this.extractLayerData(this.orderData, layerIdx)); | ||
this.drawColorPass(this.extractLayerData(this.orderData, layerIdx), | ||
this.extractLayerData(this.alphaData, layerIdx), | ||
this.extractLayerData(this.intensityData, layerIdx)); | ||
this.drawBlendPass(); | ||
@@ -280,3 +262,3 @@ } | ||
WebGLSortedVolumeCompositor.prototype.drawColorPass = function(layerOrderData) { | ||
WebGLSortedVolumeCompositor.prototype.drawColorPass = function(layerOrderData, layerAlphaData, layerIntensityData) { | ||
// Draw to the color fbo on this pass | ||
@@ -291,11 +273,5 @@ this.gl.bindFramebuffer(this.gl.FRAMEBUFFER, this.glResources.framebuffers.colorFbo); | ||
// Send uniforms specifying sprite and image dimensions: "spriteDim", "imageDim" | ||
var spriteDimLoc = this.gl.getUniformLocation(this.glResources.programs.colorProgram, "spriteDim"); | ||
this.gl.uniform2fv(spriteDimLoc, [this.spriteSize, this.spriteSize]); | ||
var imgDimLoc = this.gl.getUniformLocation(this.glResources.programs.colorProgram, "imageDim"); | ||
this.gl.uniform2fv(imgDimLoc, [this.width, this.height]); | ||
// Send uniform specifying the number of total layers | ||
var numLayersLoc = this.gl.getUniformLocation(this.glResources.programs.colorProgram, "numberOfLayers"); | ||
this.gl.uniform1i(numLayersLoc, this.numLayers); | ||
this.gl.uniform1f(numLayersLoc, this.numLayers); | ||
@@ -318,3 +294,3 @@ var texCount = 0; | ||
this.gl.bindTexture(this.gl.TEXTURE_2D, this.glResources.textures.alphaTexture); | ||
this.gl.texImage2D(this.gl.TEXTURE_2D, 0, this.gl.LUMINANCE, this.spriteSize, this.spriteSize, 0, this.gl.LUMINANCE, this.gl.UNSIGNED_BYTE, this.alphaData); | ||
this.gl.texImage2D(this.gl.TEXTURE_2D, 0, this.gl.LUMINANCE, this.width, this.height, 0, this.gl.LUMINANCE, this.gl.UNSIGNED_BYTE, layerAlphaData); | ||
@@ -327,12 +303,4 @@ // Set up the intensity sprite texture | ||
this.gl.bindTexture(this.gl.TEXTURE_2D, this.glResources.textures.intensityTexture); | ||
this.gl.texImage2D(this.gl.TEXTURE_2D, 0, this.gl.LUMINANCE, this.intensitySize, this.intensitySize, 0, this.gl.LUMINANCE, this.gl.UNSIGNED_BYTE, this.intensityData); | ||
this.gl.texImage2D(this.gl.TEXTURE_2D, 0, this.gl.LUMINANCE, this.intensitySize[0], this.intensitySize[1], 0, this.gl.LUMINANCE, this.gl.UNSIGNED_BYTE, layerIntensityData); | ||
// Set up the "graphic equalizer" texture (contains alphas to multiply each layer color) | ||
var alphaMultLoc = this.gl.getUniformLocation(this.glResources.programs.colorProgram, "alphaMultiplierSampler"); | ||
this.gl.uniform1i(alphaMultLoc, texCount); | ||
this.gl.activeTexture(this.gl.TEXTURE0 + texCount); | ||
texCount += 1; | ||
this.gl.bindTexture(this.gl.TEXTURE_2D, this.glResources.textures.alphaMultiplierTexture); | ||
this.gl.texImage2D(this.gl.TEXTURE_2D, 0, this.gl.LUMINANCE, this.numLayers, 1, 0, this.gl.LUMINANCE, this.gl.UNSIGNED_BYTE, this.opacityView); | ||
// Set up the lookup texture (contains alphas to multiply each layer color) | ||
@@ -344,3 +312,3 @@ var lutLoc = this.gl.getUniformLocation(this.glResources.programs.colorProgram, "lutSampler"); | ||
this.gl.bindTexture(this.gl.TEXTURE_2D, this.glResources.textures.lutTexture); | ||
this.gl.texImage2D(this.gl.TEXTURE_2D, 0, this.gl.RGB, this.numLayers, 1, 0, this.gl.RGB, this.gl.UNSIGNED_BYTE, this.colorTable); | ||
this.gl.texImage2D(this.gl.TEXTURE_2D, 0, this.gl.RGBA, this.numLayers, 1, 0, this.gl.RGBA, this.gl.UNSIGNED_BYTE, this.lutView); | ||
@@ -347,0 +315,0 @@ // Draw the rectangle. |
@@ -35,3 +35,3 @@ var Presets = require('./Presets.js'), | ||
export default function LookupTable(name) { | ||
export default function LookupTable(name, discrete=false) { | ||
this.name = name; | ||
@@ -45,2 +45,4 @@ this.scalarRange = [0, 1]; | ||
this.setPreset('spectralflip'); | ||
this.discrete = discrete; | ||
this.scale = 1; | ||
@@ -84,2 +86,23 @@ // Auto rebuild | ||
LookupTable.prototype.updateControlPoints = function(controlPoints) { | ||
this.colorTable = null; | ||
this.controlPoints = []; | ||
var count = controlPoints.length; | ||
for(var i = 0; i < count; i++) { | ||
this.controlPoints.push({ | ||
x: controlPoints[i].x, | ||
r: controlPoints[i].r, | ||
g: controlPoints[i].g, | ||
b: controlPoints[i].b | ||
}); | ||
} | ||
// Auto rebuild | ||
this.build(); | ||
this.emit(CHANGE_TOPIC, { change: 'controlPoints', lut: this }); | ||
}; | ||
LookupTable.prototype.setColorForNaN = function(r = 0, g = 0, b = 0, a = 0) { | ||
@@ -94,3 +117,3 @@ this.colorNaN = [r,g,b,a]; | ||
LookupTable.prototype.getScalarRange = function() { | ||
return this.scalarRange; | ||
return [Number(this.scalarRange[0]), Number(this.scalarRange[1])]; | ||
}; | ||
@@ -113,14 +136,24 @@ | ||
var currentControlIdx = 0; | ||
for (var idx = 0; idx < this.colorTableSize; idx++) { | ||
var value = idx / (this.colorTableSize - 1), | ||
pointA = extractPoint(this.controlPoints, currentControlIdx), | ||
pointB = extractPoint(this.controlPoints, currentControlIdx + 1); | ||
if(this.discrete) { | ||
this.colorTableSize = this.controlPoints.length; | ||
this.scale = 50; | ||
for (var idx = 0; idx < this.colorTableSize; idx++) { | ||
var color = this.controlPoints[idx]; | ||
this.colorTable.push([color.r, color.g, color.b, 255]); | ||
} | ||
} else { | ||
this.scale = 1; | ||
for (var idx = 0; idx < this.colorTableSize; idx++) { | ||
var value = idx / (this.colorTableSize - 1), | ||
pointA = extractPoint(this.controlPoints, currentControlIdx), | ||
pointB = extractPoint(this.controlPoints, currentControlIdx + 1); | ||
if (value > pointB[0]) { | ||
currentControlIdx += 1; | ||
pointA = extractPoint(this.controlPoints, currentControlIdx); | ||
pointB = extractPoint(this.controlPoints, currentControlIdx + 1); | ||
if (value > pointB[0]) { | ||
currentControlIdx += 1; | ||
pointA = extractPoint(this.controlPoints, currentControlIdx); | ||
pointB = extractPoint(this.controlPoints, currentControlIdx + 1); | ||
} | ||
this.colorTable.push(interpolateColor(pointA, pointB, value)); | ||
} | ||
this.colorTable.push(interpolateColor(pointA, pointB, value)); | ||
} | ||
@@ -206,3 +239,3 @@ | ||
var colors = this.colorTable, | ||
length = colors.length, | ||
length = this.scale * colors.length, | ||
ctx = canvas.getContext("2d"), | ||
@@ -212,5 +245,6 @@ canvasData = ctx.getImageData(0, 0, length, 1); | ||
for(var i = 0; i < length; i++) { | ||
canvasData.data[i*4 + 0] = Math.floor(255 * colors[i][0]); | ||
canvasData.data[i*4 + 1] = Math.floor(255 * colors[i][1]); | ||
canvasData.data[i*4 + 2] = Math.floor(255 * colors[i][2]); | ||
var colorIdx = Math.floor(i/this.scale); | ||
canvasData.data[i*4 + 0] = Math.floor(255 * colors[colorIdx][0]); | ||
canvasData.data[i*4 + 1] = Math.floor(255 * colors[colorIdx][1]); | ||
canvasData.data[i*4 + 2] = Math.floor(255 * colors[colorIdx][2]); | ||
canvasData.data[i*4 + 3] = 255; | ||
@@ -217,0 +251,0 @@ } |
@@ -53,4 +53,3 @@ var LookupTable = require('./LookupTable.js'), | ||
} | ||
LookupTableManager.prototype.getLookupTable = function(name) { | ||
LookupTableManager.prototype.updateActiveLookupTable = function(name) { | ||
setImmediate(()=>{ | ||
@@ -60,8 +59,21 @@ this.emit(ACTIVE_LOOKUP_TABLE_CHANGE_TOPIC, name); | ||
this.activeField = name; | ||
} | ||
LookupTableManager.prototype.getLookupTable = function(name) { | ||
return this.luts[name]; | ||
} | ||
LookupTableManager.prototype.addFields = function(fieldsRange) { | ||
LookupTableManager.prototype.addFields = function(fieldsRange, lutConfigs) { | ||
for(var field in fieldsRange) { | ||
this.addLookupTable(field, fieldsRange[field]); | ||
var lut = this.addLookupTable(field, fieldsRange[field]); | ||
if(lutConfigs && lutConfigs[field]) { | ||
if(lutConfigs[field].discrete !== undefined) { | ||
lut.discrete = lutConfigs[field].discrete; | ||
} | ||
if(lutConfigs[field].preset) { | ||
lut.setPreset(lutConfigs[field].preset); | ||
} else if(lutConfigs[field].controlpoints) { | ||
lut.updateControlPoints(lutConfigs[field].controlpoints); | ||
} | ||
} | ||
} | ||
@@ -68,0 +80,0 @@ } |
@@ -24,3 +24,3 @@ var Monologue = require('monologue.js'), | ||
// fields: [ | ||
// { name: 'Temperature', data: [y0, y1, ..., yn]}, | ||
// { name: 'Temperature', data: [y0, y1, ..., yn], range: [0, 1]}, | ||
// ... | ||
@@ -149,3 +149,5 @@ // ] | ||
function getY(idx) { | ||
return yOffset + height - Math.floor((values[idx] - min) * scaleY); | ||
var value = values[idx]; | ||
value = (value > min) ? ( (value < max) ? value : max) : min; | ||
return yOffset + height - Math.floor((value - min) * scaleY); | ||
} | ||
@@ -159,3 +161,9 @@ | ||
for(var idx = 1; idx < size; idx++) { | ||
ctx.lineTo(xValues[idx], getY(idx)); | ||
if(isNaN(values[idx])) { | ||
if(idx + 1 < size && !isNaN(values[idx+1])) { | ||
ctx.moveTo(xValues[idx+1], getY(idx+1)); | ||
} | ||
} else { | ||
ctx.lineTo(xValues[idx], getY(idx)); | ||
} | ||
} | ||
@@ -162,0 +170,0 @@ ctx.stroke(); |
{ | ||
"name": "tonic-image-builder", | ||
"description": "JavaScript library used to gather image generation based on various input data.", | ||
"version": "0.0.8", | ||
"version": "0.1.0", | ||
"license": "BSD-3-Clause", | ||
@@ -12,32 +12,8 @@ "main": "./src/index.js", | ||
"dependencies": { | ||
"gl-matrix": "2.3.1", | ||
"monologue.js": "0.3.3", | ||
"mout": "0.11.0" | ||
}, | ||
"devDependencies": { | ||
"monologue.js": "0.3.3", | ||
"mout": "0.11.0", | ||
"gl-matrix": "2.3.1", | ||
"tonic-site-generator": "0.1.0", | ||
"jshint": "2.8.0", | ||
"jshint-loader": "0.8.3", | ||
"file-loader": "0.8.4", | ||
"url-loader": "0.5.6", | ||
"babel-core": "5.8.22", | ||
"babel-loader": "5.3.2", | ||
"expose-loader": "0.7.0", | ||
"node-libs-browser": "0.5.2", | ||
"shader-loader": "1.1.3", | ||
"webpack": "1.11.0", | ||
"jasmine-core": "2.3.4", | ||
"karma": "0.13.9", | ||
"karma-cli": "0.1.0", | ||
"karma-jasmine": "0.3.6", | ||
"phantomjs": "1.9.18", | ||
"karma-phantomjs-launcher": "0.2.1", | ||
"karma-chrome-launcher": "0.2.0", | ||
"karma-firefox-launcher": "0.1.6", | ||
"karma-safari-launcher": "0.1.1", | ||
"serve": "1.4.0" | ||
"cz-conventional-changelog": "1.1.0" | ||
}, | ||
@@ -48,10 +24,21 @@ "scripts": { | ||
"build:release": "webpack -p", | ||
"www": "tonic-site-generator site.config.js", | ||
"www:http" : "tonic-site-generator site.config.js --local-test && serve docs/www", | ||
"www:http": "tonic-site-generator site.config.js --local-test && serve docs/www", | ||
"test": "karma start ./tests/karma.conf.js", | ||
"test:chrome": "karma start ./tests/karma.conf.js --browsers Chrome --single-run", | ||
"test:firefox": "karma start ./tests/karma.conf.js --browsers Firefox" | ||
"test:firefox": "karma start ./tests/karma.conf.js --browsers Firefox", | ||
"commit": "git-cz", | ||
"semantic-release": "semantic-release pre && npm publish && semantic-release post", | ||
"dep:webpack": "npm install node-libs-browser@0.5.2 autoprefixer-loader@2.0.0 babel-core@5.8.22 babel-loader@5.3.2 css-loader@0.16.0 expose-loader@0.7.0 file-loader@0.8.4 html-webpack-plugin@1.6.1 jshint@2.8.0 jshint-loader@0.8.3 json-loader@^0.5.2 shader-loader@1.1.3 style-loader@0.12.3 url-loader@0.5.6 webpack@1.11.0", | ||
"dep:test": "npm install jasmine-core@2.3.4 karma-jasmine@0.3.6 karma-phantomjs-launcher@0.2.1", | ||
"dep:test-g": "npm install karma@0.13.10 phantomjs@1.9.18", | ||
"dep:www": "npm install tonic-site-generator@0.1.0 serve@1.4.0", | ||
"dep:release": "npm install commitizen@1.0.4 semantic-release@4.3.5", | ||
"dep:travis": "npm run dep:test-g -- -g && npm run dep:webpack && npm run dep:release && npm run dep:test", | ||
"dep:local": "npm run dep:webpack && npm run dep:test-g && npm run dep:test && npm run dep:www && npm run dep:release", | ||
"dep:global": "npm run dep:webpack -- -g && npm run dep:test-g -- -g && npm run dep:test -- -g && npm run dep:www -- -g && npm run dep:release -- -g" | ||
}, | ||
"czConfig": { | ||
"path": "node_modules/cz-conventional-changelog" | ||
} | ||
} | ||
} |
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is not supported yet
2135205
1.91%1
-95.83%159
3.92%12527
7.1%3
Infinity%+ Added
+ Added
+ Added
+ Added
+ Added
+ Added
+ Added