Commit 81aa3d06 authored by Dmitry.Shahtanov's avatar Dmitry.Shahtanov Committed by Alexander.Trofimov

fixed: Bug 24650 - Некорректная ошибка выводится для функции ISPMT если аргумент nper равен нулю,

Bug 24627 - Не фильтруются текстовые данные при вычислении формулы IRR

git-svn-id: svn://192.168.3.15/activex/AVS/Sources/TeamlabOffice/trunk/OfficeWeb@56557 954022d7-b5bf-4e40-9824-e11837661b57
parent ff9f7772
...@@ -2759,67 +2759,97 @@ cIRR.prototype = Object.create( cBaseFunction.prototype ); ...@@ -2759,67 +2759,97 @@ cIRR.prototype = Object.create( cBaseFunction.prototype );
cIRR.prototype.Calculate = function ( arg ) { cIRR.prototype.Calculate = function ( arg ) {
var arg0 = arg[0], arg1 = arg[1] ? arg[1] : new cNumber( 0.1 ); var arg0 = arg[0], arg1 = arg[1] ? arg[1] : new cNumber( 0.1 );
function irr( arr, x ) { function npv(r,cf){
x = x.getValue(); var res = 0;
for(var i = 1; i<=cf.length;i++){
res += cf[i-1].getValue() / Math.pow( 1 + r, i );
}
return res;
}
var count = 0, g_Eps = 1e-7, eps = 1, funcVal = 0, derivVal = 0, xN = 0, nIM = 100, nMC = 0, arr0 = arr[0], arrI, wasNegative = false, wasPositive = false; function irr2(x, arr){
var g_Eps = 1e-7, nIM = 500, eps = 1, nMC= 0, xN, guess = x;
if ( arr0 instanceof cError ) { while ( eps > g_Eps && nMC < nIM ) {
return new cError( cErrorType.not_available ); xN = x - npv(x,arr) / ( (npv( x + g_Eps, arr ) - npv( x - g_Eps, arr )) / (2 * g_Eps) );
nMC++;
eps = Math.abs( xN - x );
x = xN;
} }
if ( isNaN( x ) || Infinity == Math.abs(x) ) {
var max = Number.MAX_VALUE, min = -Number.MAX_VALUE,
step = 1.6,
low = guess - 0.01 <= min ? min + g_Eps : guess - 0.01,
high = guess + 0.01 >= max ? max - g_Eps : guess + 0.01,
i, xBegin, xEnd, x, y, currentIter = 0;
if ( arr0.getValue() < 0 ) if ( guess <= min || guess >= max ) {
wasNegative = true;
else if ( arr0.getValue() > 0 )
wasPositive = true;
if ( arr.length < 2 )
return new cError( cErrorType.not_numeric ); return new cError( cErrorType.not_numeric );
}
while ( eps > g_Eps && nMC < nIM ) { for ( i = 0; i < nIM; i++ ) {
count = 0; xBegin = low <= min ? min + g_Eps : low;
funcVal = 0; xEnd = high >= max ? max - g_Eps : high;
derivVal = 0; x = npv( xBegin, arr );
funcVal += arr0.getValue() / Math.pow( 1 + x, count ); y = npv( xEnd, arr );
derivVal += -count * arr0.getValue() / Math.pow( 1 + x, count + 1 ); if ( x * y <= 0 ) {
count++; break;
for ( var i = 1; i < arr.length; i++ ) {
if ( arr[i] instanceof cError ) {
return new cError( cErrorType.not_available );
} }
arrI = arr[i].getValue(); else if ( x * y > 0 ) {
funcVal += arrI / Math.pow( 1 + x, count ); low = (xBegin + step * (xBegin - xEnd));
derivVal += -count * arrI / Math.pow( 1 + x, count + 1 ); high = (xEnd + step * (xEnd - xBegin));
if ( arrI < 0 ) }
wasNegative = true; else {
else if ( arrI > 0 ) return new cError( cErrorType.not_numeric );
wasPositive = true;
count++
} }
xN = x - funcVal / derivVal;
nMC++;
eps = Math.abs( xN - x );
x = xN;
} }
if ( !(wasNegative && wasPositive) ) if ( i == nIM ) {
return new cError( cErrorType.not_numeric ); return new cError( cErrorType.not_numeric );
}
if ( eps < g_Eps ) var fXbegin = npv( xBegin, arr ), fXend = npv( xEnd, arr ), fXi, xI;
return new cNumber( x );
if ( Math.abs( fXbegin ) < g_Eps ) {
return new cNumber( fXbegin );
}
if ( Math.abs( fXend ) < g_Eps ) {
return new cNumber( fXend );
}
do {
xI = xBegin + (xEnd - xBegin) / 2;
fXi = npv( xI, arr );
if ( fXbegin * fXi < 0 ) {
xEnd = xI;
}
else {
xBegin = xI;
}
fXbegin = npv( xBegin, arr );
currentIter++;
} while ( Math.abs( fXi ) > g_Eps && currentIter < nIM )
return new cNumber( xI );
}
else else
return new cError( cErrorType.not_numeric ); return new cNumber( x );
} }
var arr = []; var arr = [];
if ( arg0 instanceof cArray ) { if ( arg0 instanceof cArray ) {
arg0.foreach( function ( v ) { arg0.foreach( function ( v ) {
arr.push( v.tocNumber() ) if( v instanceof cNumber ){
} ) arr.push( v );
}
} );
} }
else if ( arg0 instanceof cArea ) { else if ( arg0 instanceof cArea ) {
arg0.foreach2( function ( v ) { arg0.foreach2( function ( v ) {
arr.push( v.tocNumber() ) if( v instanceof cNumber ){
} ) arr.push( v );
}
} );
} }
arg1 = arg1.tocNumber(); arg1 = arg1.tocNumber();
...@@ -2827,7 +2857,20 @@ cIRR.prototype.Calculate = function ( arg ) { ...@@ -2827,7 +2857,20 @@ cIRR.prototype.Calculate = function ( arg ) {
if ( arg1 instanceof cError ) { if ( arg1 instanceof cError ) {
return this.value = new cError( cErrorType.not_numeric ); return this.value = new cError( cErrorType.not_numeric );
} }
this.value = irr( arr, arg1 );
var wasNeg = false, wasPos = false;
for(var i = 0; i<arr.length;i++){
if( arr[i].getValue() > 0 )
wasNeg = true;
if( arr[i].getValue() < 0 )
wasPos = true;
}
if( !(wasNeg && wasPos) ){
return this.value = new cError( cErrorType.not_numeric );
}
this.value = irr2( arg1.getValue(), arr );
this.value.numFormat = 9; this.value.numFormat = 9;
return this.value; return this.value;
...@@ -2898,6 +2941,10 @@ cISPMT.prototype.Calculate = function ( arg ) { ...@@ -2898,6 +2941,10 @@ cISPMT.prototype.Calculate = function ( arg ) {
if ( nper instanceof cError ) return this.value = nper; if ( nper instanceof cError ) return this.value = nper;
if ( pv instanceof cError ) return this.value = pv; if ( pv instanceof cError ) return this.value = pv;
if( nper.getValue() < 0 ){
return this.value = new cError( cErrorType.division_by_zero );
}
return this.value = new cNumber( pv.getValue() * rate.getValue() * (per.getValue() / nper.getValue() - 1) ); return this.value = new cNumber( pv.getValue() * rate.getValue() * (per.getValue() / nper.getValue() - 1) );
}; };
cISPMT.prototype.getInfo = function () { cISPMT.prototype.getInfo = function () {
...@@ -3682,7 +3729,7 @@ cODDFYIELD.prototype.Calculate = function ( arg ) { ...@@ -3682,7 +3729,7 @@ cODDFYIELD.prototype.Calculate = function ( arg ) {
xEnd = high >= max ? max - g_Eps : high; xEnd = high >= max ? max - g_Eps : high;
x = iterF( xBegin ); x = iterF( xBegin );
y = iterF( xEnd ); y = iterF( xEnd );
if ( x * y == 0 || x * y < 0 ) { if ( x * y <= 0 ) {
break; break;
} }
else if ( x * y > 0 ) { else if ( x * y > 0 ) {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment