@ -1,30 +1,19 @@
function TurkeyLayer ( name , percentRadius , turkeyModel , ovenModel ) {
function TurkeyLayer ( name , layerNumber1 , turkeyModel , ovenModel ) {
var that = this ;
var that = this ;
this . name = name ;
this . name = name ;
this . percentRadius = percentRadius ;
this . layerNumber = layerNumber1 ;
this . initialTemp = 20 ;
this . waterLost = 0 ;
this . waterLost = 0 ;
this . finalTemperature = 20 ;
this . finalTemperature = 20 ;
this . cookCondition = "Raw" ;
this . cookCondition = "Raw" ;
return {
return {
updateTemperatureTick : function ( ) {
updateTemperatureTick : function ( ) {
that . finalTemperature = UtilityFunctions . transientSphereSeries ( turkeyModel . density ,
that . finalTemperature = turkeyModel . globTemp [ that . layerNumber ]
turkeyModel . thermalConduct ,
console . log ( turkeyModel . globTemp )
turkeyModel . heatConvection ,
turkeyModel . cp ,
percentRadius * turkeyModel . totalRadius ,
turkeyModel . totalRadius ,
that . initialTemp ,
ovenModel . steadyTemp ,
ovenModel . globalTime ) ;
that . waterLost = that . waterLost + UtilityFunctions . waterLoss ( that . finalTemperature ) ;
that . waterLost = that . waterLost + UtilityFunctions . waterLoss ( that . finalTemperature ) ;
that . cookCondition = UtilityFunctions . cookCondition ( that . waterLost ) ;
that . cookCondition = UtilityFunctions . cookCondition ( that . waterLost , that . name ) ;
if ( DEBUG ) console . log ( that . name + ": " + that . waterLost + " " + that . cookCondition ) ;
if ( DEBUG ) console . log ( that . name + ": " + that . waterLost + " " + that . cookCondition + " " + that . finalTemperature + " C" ) ;
} ,
resetLayerTemps : function ( ) {
that . initialTemp = that . finalTemperature ;
} ,
} ,
getCondition : function ( ) {
getCondition : function ( ) {
return that . cookCondition ;
return that . cookCondition ;
@ -38,45 +27,71 @@ function TurkeyLayer( name, percentRadius, turkeyModel, ovenModel ){
function TurkeyModel ( weight , ovenModel ) {
function TurkeyModel ( weight , ovenModel ) {
this . density = 105 0; // kg/m3 Assuming Density of Water 1000 kg/m3
this . density = 70 0; // kg/m3 Assuming Density of Water 1000 kg/m3
this . cp = 200 0 ; // 2810 J/kg K for Turkey. Extra is to semi-account for water evaporation energy
this . cp = 281 0 ; // 2810 J/kg K for Turkey. Extra is to semi-account for water evaporation energy
this . heatConvection = 9 ; // W/m2 K Some Reasonable estimate for natural Convection. Change as needed. 5-25
this . heatConvection = 9 ; // W/m2 K Some Reasonable estimate for natural Convection. Change as needed. 5-25
this . thermalConduct = 0.412 ; // W/m K // Chicken
this . thermalConduct = 0.412 ; // W/m K // Chicken
this . skin = { } ;
this . skin = { } ;
this . body = { } ;
this . body = { } ;
this . core = { } ;
this . core = { } ;
this . splitsNum = 20 ;
this . totalRadius = UtilityFunctions . calculateRadius ( weight , this . density ) ;
console . log ( UtilityFunctions . lbs2kgs ( weight ) )
this . totalRadius = UtilityFunctions . calculateRadius ( UtilityFunctions . lbs2kgs ( weight ) , this . density ) ;
this . totalLayers = [ new TurkeyLayer ( "Skin" , 0.85 , this , ovenModel ) ,
this . totalLayers = [ new TurkeyLayer ( "Skin" , this . splitsNum - 1 , this , ovenModel ) ,
new TurkeyLayer ( "Body" , 0.45 , this , ovenModel ) ,
new TurkeyLayer ( "Body" , this . splitsNum - 4 , this , ovenModel ) ,
new TurkeyLayer ( "Core" , 0.01 , this , ovenModel ) ] ;
new TurkeyLayer ( "Core" , 0 , this , ovenModel ) ] ;
// Whenever temperature is changed
// Whenever temperature is changed
this . updateLayerTemps = function ( ) {
this . updateLayerTemps = function ( ) {
this . globTemp = UtilityFunctions . transientSphereSeries ( this . density ,
this . thermalConduct ,
this . heatConvection ,
this . cp ,
this . totalRadius ,
ovenModel . tempInfini ,
this . splitsNum ,
this . deltar ,
this . globTemp ,
this . pointRadius
) ;
for ( var i in this . totalLayers ) {
for ( var i in this . totalLayers ) {
this . totalLayers [ i ] . updateTemperatureTick ( ) ;
this . totalLayers [ i ] . updateTemperatureTick ( ) ;
}
}
} ;
} ;
this . resetLayerTemps = function ( ) {
this . resetLayerTemps = function ( ) {
for ( var i in this . totalLayers ) {
for ( var i in this . totalLayers ) {
this . totalLayers [ i ] . resetLayerTemps ( ) ;
this . totalLayers [ i ] . resetLayerTemps ( ) ;
}
}
} ;
} ;
//Sheen Model Stuff
this . globTemp = [ ] ;
this . pointRadius = [ ]
this . splitsNum = 20 ;
this . deltar = this . totalRadius / this . splitsNum ; //20 Data Points
this . initializePoints = function ( ) {
var step = ( this . totalRadius - this . deltar ) / ( this . splitsNum - 1 ) ;
for ( var i = 0 ; i < this . splitsNum ; i ++ ) {
this . pointRadius . push ( step * i + this . deltar ) ;
this . globTemp . push ( 20 + step * i ) ; //Starts at 20 C for initilizating
}
} ;
this . initializePoints ( )
}
}
function OvenModel ( turkeyWeight , gameState ) {
function OvenModel ( turkeyWeight , gameState ) {
var that = this ;
var that = this ;
this . tempInfini = 20 ; //C
this . tempInfini = 20 ; //C
this . setTemp = 20 ;
this . setTemp = 20 ;
this . steadyTemp = 20 ;
this . steadyTimer = 0 ;
this . globalTime = 0 ;
this . globalTime = 0 ;
var turkey = new TurkeyModel ( UtilityFunctions . lbs2kgs ( turkeyWeight ) , this ) ;
var turkey = new TurkeyModel ( turkeyWeight , this ) ;
var proportional = 0.004 ; // This value is arbitrary to how fast you want the temperatures to converge. (Or oscillate, which could be realistic as well)
var proportional = 0.004 ; // This value is arbitrary to how fast you want the temperatures to converge. (Or oscillate, which could be realistic as well)
var errorTolerance = 10 ; //Stove is accurate to 1 degree Celcius Should hopefully oscillate below that value.
var errorTolerance = 10 ; //Stove is accurate to 1 degree Celcius Should hopefully oscillate below that value.
// Equalize temp will need to be sent each time iteration
// Equalize temp will need to be sent each time iteration
@ -90,13 +105,6 @@ function OvenModel( turkeyWeight, gameState ) {
}
}
if ( error > errorTolerance ) {
if ( error > errorTolerance ) {
if ( this . steadyTimer >= 80 ) {
//Reset the model's time calculation if there are major changes in the tolerance of the temperature or the steady timer expires
this . steadyTimer = 0 ;
this . globalTime = 0 ;
this . steadyTemp = this . tempInfini
turkey . resetLayerTemps ( ) ;
}
return ( true ) ;
return ( true ) ;
}
}
}
}
@ -134,19 +142,15 @@ function OvenModel( turkeyWeight, gameState ) {
} ,
} ,
secondTick : function ( ) {
secondTick : function ( ) {
that . globalTime = that . globalTime + 1 ;
that . globalTime = that . globalTime + 1 ;
that . steadyTimer = that . steadyTimer + 1 ;
if ( that . equalizeTemp ( ) ) {
if ( that . equalizeTemp ( ) ) {
// Turn on oven light
// Turn on oven light
gameState . pubsub . publish ( "OvenLight" , "On" ) ;
gameState . pubsub . publish ( "OvenLight" , "On" ) ;
}
}
else {
else {
that . steadyTemp = that . tempInfini ;
// Turn off oven light
// Turn off oven light
gameState . pubsub . publish ( "OvenLight" , "Off" ) ;
gameState . pubsub . publish ( "OvenLight" , "Off" ) ;
}
}
if ( DEBUG ) console . log ( "Steady Temp " + that . steadyTemp )
if ( DEBUG ) console . log ( "Steady Timer " + that . steadyTimer )
if ( DEBUG ) console . log ( "Oven Temp " + that . tempInfini )
if ( DEBUG ) console . log ( "Oven Temp " + that . tempInfini )
turkey . updateLayerTemps ( ) ;
turkey . updateLayerTemps ( ) ;
}
}
@ -179,26 +183,6 @@ UtilityFunctions = {
return complexRadius ;
return complexRadius ;
} ,
} ,
findAllRoots : function ( min , max , splitsNum , Biot ) {
var step = ( max - min ) / ( splitsNum - 1 ) ;
var answer ;
var negativeTest ;
var storage = [ ] ;
for ( var i = step ; i < max ; i = i + step ) {
negativeTest = this . lambdaFormula ( i - step , Biot ) * this . lambdaFormula ( i , Biot ) ;
if ( negativeTest <= 0 ) {
answer = this . bisectionMethod ( i - step , i , Biot ) ;
if ( answer != undefined ) {
storage . push ( answer ) ;
}
}
else {
//if(DEBUG) console.log("No Bracketed Root " + negativeTest)
}
}
return storage ;
} ,
sphereVolume : function ( radius ) {
sphereVolume : function ( radius ) {
return ( ( 4 / 3 ) * Math . PI * Math . pow ( radius , 3 ) )
return ( ( 4 / 3 ) * Math . PI * Math . pow ( radius , 3 ) )
} ,
} ,
@ -207,74 +191,50 @@ UtilityFunctions = {
return ( Math . pow ( 10 , ( temperature - 20 ) / 80 ) - 1 )
return ( Math . pow ( 10 , ( temperature - 20 ) / 80 ) - 1 )
} ,
} ,
transientSphereSeries : function ( density , thermalConduct , heatConvection , cp , rTotal , tempInfinity , splitsNum , deltar , globTemp , pointRadius ) {
bisectionMethod : function ( min , max , Biot ) {
//Not Global Stuff
errorTolerance = ( 1 / Math . pow ( 10 , 8 ) )
var r0 = rTotal ;
result = Infinity // some large value to ensure the calculation goes through.
var deltat = 0.1
negativeTest = this . lambdaFormula ( min , Biot ) * this . lambdaFormula ( max , Biot )
if ( negativeTest <= 0 ) {
var antiFreeze = 0 ;
while ( Math . abs ( result ) > errorTolerance && antiFreeze <= 500 ) { //The greater the antiFreeze, the more wasted cycles around a singularity
lambdaN = ( min + max ) / 2
result = this . lambdaFormula ( lambdaN , Biot )
if ( Math . abs ( result ) <= errorTolerance && result <= errorTolerance ) {
return ( lambdaN ) ; //At Root
}
else if ( ( this . lambdaFormula ( min , Biot ) * this . lambdaFormula ( lambdaN , Biot ) ) >= 0 ) {
min = lambdaN ;
}
else if ( ( this . lambdaFormula ( max , Biot ) * this . lambdaFormula ( lambdaN , Biot ) ) >= 0 ) {
max = lambdaN ;
}
antiFreeze ++
}
}
} ,
lambdaFormula : function ( lambdaN , Biot ) {
var result = 1 - lambdaN * ( 1 / Math . tan ( lambdaN ) ) - Biot ;
return ( result )
} ,
transientSphereSeries : function ( density , thermalConduct , heatConvection , cp , rPosition , rTotal , tempInitial , tempInfini , t ) {
var min = 0 ;
var max = 10000 ; // This are for setting Lambda boundaries and nothing else
var sum = 0 ;
var alpha = thermalConduct / ( density * cp ) ;
var lambdaN ;
var sinPortion ;
var exponentialPortion ;
var frontCoefficientPortion ;
var alpha = thermalConduct / ( density * cp )
var h = heatConvection ;
//if(DEBUG) console.log("Alpha is " + alpha)
for ( var j = 0 ; j < ( 1 / deltat ) ; j ++ ) {
var dTdr = [ ]
var Fourier = ( alpha * t ) / Math . pow ( rTotal , 2 )
// globTemp[splitsNum-1] should be last entry in globtemp
//if(DEBUG) console.log("Fourier is " + Fourier)
for ( var k = 0 ; k < splitsNum ; k ++ ) {
if ( k == 0 ) {
var biotNum = heatConvection * rTotal / thermalConduct
dTdr . push ( ( globTemp [ 1 ] - globTemp [ 0 ] ) / deltar ) }
else if ( k == splitsNum - 1 ) {
if ( biotNum != this . cachedBiot ) {
dTdr . push ( ( globTemp [ splitsNum - 1 ] - globTemp [ splitsNum - 2 ] ) / deltar ) }
if ( DEBUG ) console . log ( "Recalculating Lambda Terms" )
else {
this . cachedLambda = this . findAllRoots ( min , max , max * Math . PI * 10 , biotNum )
dTdr . push ( ( globTemp [ k + 1 ] - globTemp [ k - 1 ] ) / ( 2 * deltar ) ) }
this . cachedBiot = biotNum ;
}
}
dTdr [ splitsNum - 1 ] = heatConvection * ( tempInfinity - globTemp [ splitsNum - 1 ] ) / thermalConduct
//if(DEBUG) console.log("The Biot Value is " + biotNum)
var parenthesis = [ ]
for ( var k = 0 ; k < splitsNum ; k ++ ) {
parenthesis . push ( dTdr [ k ] * Math . pow ( pointRadius [ k ] , 2 ) )
}
for ( var i = 0 ; i < this . cachedLambda . length ; i ++ ) {
dPdr = [ ]
var lambdaN = this . cachedLambda [ i ] ;
for ( var k = 0 ; k < splitsNum ; k ++ ) {
var sinPortion = Math . sin ( lambdaN * rPosition / rTotal ) / ( lambdaN * rPosition / rTotal ) ;
if ( k == 0 ) {
var exponentialPortion = ( 1 / Math . exp ( Math . pow ( lambdaN , 2 ) * Fourier ) ) ;
dPdr . push ( ( parenthesis [ 1 ] - parenthesis [ 0 ] ) / deltar ) }
var frontCoefficientPortion = 4 * ( Math . sin ( lambdaN ) - ( lambdaN * Math . cos ( lambdaN ) ) ) / ( 2 * lambdaN - Math . sin ( 2 * lambdaN ) ) ;
else if ( k == splitsNum - 1 ) {
sum = frontCoefficientPortion * exponentialPortion * sinPortion + sum ;
dPdr . push ( ( parenthesis [ splitsNum - 1 ] - parenthesis [ splitsNum - 2 ] ) / deltar ) }
else {
dPdr . push ( ( parenthesis [ k + 1 ] - parenthesis [ k - 1 ] ) / ( 2 * deltar ) ) }
}
}
tempAtTimeAndRadius = ( sum * ( tempInitial - tempInfini ) ) + tempInfini
for ( var k = 0 ; k < splitsNum ; k ++ ) {
globTemp [ k ] = alpha * dPdr [ k ] / Math . pow ( pointRadius [ k ] , 2 ) * deltat + globTemp [ k ] //dTdr * deltaT in one loop
}
//dTdt(1)=dTdt(1)/2;
}
if ( DEBUG ) console . log ( "The Temperature at radius " + rPosition + " m and time " + t / 60 / 60 + " hours is " + tempAtTimeAndRadius + " C or " + this . C2F ( tempAtTimeAndRadius ) + " F" ) ;
return ( globTemp )
return ( tempAtTimeAndRadius )
} ,
} ,
/* Utility Functions */
/* Utility Functions */
@ -290,7 +250,9 @@ UtilityFunctions = {
randRange : function ( min , max ) {
randRange : function ( min , max ) {
return Math . floor ( Math . random ( ) * ( max - min + 1 ) ) + min ;
return Math . floor ( Math . random ( ) * ( max - min + 1 ) ) + min ;
} ,
} ,
cookCondition : function ( cookValue , volume ) {
cookCondition : function ( cookValue , layerName ) {
if ( layerName == "skin" ) {
var multiplier = 1 ;
var multiplier = 1 ;
if ( cookValue >= multiplier * 600000 ) {
if ( cookValue >= multiplier * 600000 ) {
return [ "Fire" , ( cookValue - 600000 ) / ( multiplier * 600000 ) , "fire" ] ;
return [ "Fire" , ( cookValue - 600000 ) / ( multiplier * 600000 ) , "fire" ] ;
@ -317,13 +279,42 @@ UtilityFunctions = {
return [ "Raw" , 1 , "raw" ] ;
return [ "Raw" , 1 , "raw" ] ;
}
}
}
}
else {
var multiplier = 1 ;
if ( cookValue >= multiplier * 600000 ) {
return [ "Fire" , ( cookValue - 600000 ) / ( multiplier * 600000 ) , "fire" ] ;
}
else if ( cookValue >= multiplier * 250000 ) {
return [ "Burnt" , ( cookValue - 250000 ) / ( multiplier * 600000 ) , "burnt" ] ;
}
else if ( cookValue >= multiplier * 150000 ) {
return [ "Dry" , ( cookValue - 150000 ) / ( multiplier * 250000 ) , "dry" ] ;
}
else if ( cookValue >= multiplier * 85000 ) {
return [ "Cooked" , ( cookValue - 12000 ) / ( multiplier * 150000 ) , "overcooked" ] ;
}
else if ( cookValue >= multiplier * 12000 ) {
return [ "Cooked" , ( cookValue - 12000 ) / ( multiplier * 150000 ) , "cooked" ] ;
}
else if ( cookValue >= multiplier * 10000 ) {
return [ "Undercooked" , ( cookValue - 5000 ) / ( multiplier * 12000 ) , "slightly cooked" ] ;
}
else if ( cookValue >= multiplier * 5000 ) {
return [ "Undercooked" , ( cookValue - 5000 ) / ( multiplier * 12000 ) , "undercooked" ] ;
}
else {
return [ "Raw" , 1 , "raw" ] ;
}
}
}
}
}
//Running the Program Stuff
//Running the Program Stuff
/ *
/ *
var ovenObject = new OvenModel ( ) ;
var ovenObject = new OvenModel ( ) ;
var turkey = new TurkeyModel ( 9.07185 , ovenObject ) ;
var turkey = new TurkeyModel ( 9 , ovenObject ) ;
globalTime = 0 ;
globalTime = 0 ;
setInterval ( function ( ) { ovenObject . secondTick ( ) ; } , 10 ) ;
setInterval ( function ( ) { ovenObject . secondTick ( ) ; } , 1000 ) ;
* /
* /