the function given at simple financial rate function in javascript is not giving me same answers as excel rate function some time. It works perfectly for the problem given at http://allfinancialmatters.com/2009/11/03/how-to-use-the-rate-function-in-excel/ but for my test cases. its results are different from excel rate. this is strange behaviour. i am unable to sort this out. my test cases (with excel output) are
JavaScript
x
6
1
RATE(360,-665.3, 99000) = 0.0059
2
RATE(360,-958.63, 192000) =0.0036
3
RATE(180,-1302.96,192000) = 0.0023
4
RATE(360, -889.19, 192000) =0.00312
5
RATE(360, -1145.8, 240000) = 0.0033
6
my code.js is
JavaScript
1
60
60
1
function rate(paymentsPerYear, paymentAmount, presentValue, futureValue, dueEndOrBeginning, interest)
2
{
3
//If interest, futureValue, dueEndorBeginning was not set, set now
4
//if (interest == null) // not working here :D
5
if (isNaN(interest))
6
interest = 0.1;
7
//interest = 0.1;
8
9
if (isNaN(futureValue))
10
futureValue = 0;
11
12
if (isNaN(dueEndOrBeginning))
13
dueEndOrBeginning = 0;
14
15
var FINANCIAL_MAX_ITERATIONS = 128;//Bet accuracy with 128
16
var FINANCIAL_PRECISION = 0.0000001;//1.0e-8
17
18
var y, y0, y1, x0, x1 = 0, f = 0, i = 0;
19
var rate = interest; // initiallizing rate to our guess interest
20
if (Math.abs(rate) < FINANCIAL_PRECISION)
21
{
22
y = presentValue * (1 + paymentsPerYear * rate) + paymentAmount * (1 + rate * dueEndOrBeginning) * paymentsPerYear + futureValue;
23
}
24
else
25
{
26
f = Math.exp(paymentsPerYear * Math.log(1 + rate));
27
y = presentValue * f + paymentAmount * (1 / rate + dueEndOrBeginning) * (f - 1) + futureValue;
28
}
29
y0 = presentValue + paymentAmount * paymentsPerYear + futureValue;
30
y1 = presentValue * f + paymentAmount * (1 / rate + dueEndOrBeginning) * (f - 1) + futureValue;
31
32
// find root by Newton secant method
33
i = x0 = 0.0;
34
x1 = rate;
35
while ((Math.abs(y0 - y1) > FINANCIAL_PRECISION)
36
&& (i < FINANCIAL_MAX_ITERATIONS))
37
{
38
rate = (y1 * x0 - y0 * x1) / (y1 - y0);
39
x0 = x1;
40
x1 = rate;
41
42
if (Math.abs(rate) < FINANCIAL_PRECISION)
43
{
44
y = presentValue * (1 + paymentsPerYear * rate) + paymentAmount * (1 + rate * dueEndOrBeginning) * paymentsPerYear + futureValue;
45
}
46
else
47
{
48
f = Math.exp(paymentsPerYear * Math.log(1 + rate));
49
y = presentValue * f + paymentAmount * (1 / rate + dueEndOrBeginning) * (f - 1) + futureValue;
50
}
51
52
y0 = y1;
53
y1 = y;
54
++i;
55
}
56
return rate;
57
//return String(parseFloat(rate).toFixed(3)); // rounding it to 3 decimal places
58
//return parseFloat(rate).toFixed(3);
59
}
60
and my HTML file is
JavaScript
1
74
74
1
<head><title>JavaScript Loan Calculator</title>
2
<script src="code.js"></script>
3
</head>
4
<body bgcolor="white">
5
6
<form name="loandata">
7
<table>
8
9
<tr>
10
<td>1)</td>
11
<td>paymentsPerYear:</td>
12
<td><input type="text" name="paymentsPerYear" size="12"
13
onchange="calculate();"></td>
14
</tr>
15
<tr>
16
<td>2)</td>
17
<td>paymentAmount:</td>
18
<td><input type="text" name="paymentAmount" size="12"
19
onchange="calculate();"></td>
20
</tr>
21
<tr>
22
<td>3)</td>
23
<td>presentValue:</td>
24
<td><input type="text" name="presentValue" size="12"
25
onchange="calculate();"></td>
26
</tr>
27
28
29
<tr>
30
<td>4)</td>
31
<td>futureValue:</td>
32
<td><input type="text" name="futureValue" size="12"></td>
33
</tr>
34
<tr>
35
<td>5)</td>
36
<td>dueEndOrBeginning:</td>
37
<td><input type="text" name="dueEndOrBeginning" size="12"></td>
38
</tr>
39
<tr>
40
<td>6)</td>
41
<td>interest:</td>
42
<td><input type="text" name="interest" size="12"></td>
43
</tr>
44
<tr><td colspan="3">
45
<input type="button" value="Compute" onClick="calculate();">
46
</td></tr>
47
<tr>
48
49
<td>APR:</td>
50
<td><input type="text" name="APR" id="APR" size="12"></td>
51
</tr>
52
</table>
53
</form>
54
55
56
<script language="JavaScript">
57
function calculate() {
58
var paymentsPerYear = document.loandata.paymentsPerYear.value;
59
var paymentAmount = document.loandata.paymentAmount.value;
60
var presentValue = document.loandata.presentValue.value;
61
var futureValue = document.loandata.futureValue.value;
62
var dueEndOrBeginning = document.loandata.dueEndOrBeginning.value ;
63
var interest = document.loandata.interest.value ;
64
var ans = rate(parseFloat(paymentsPerYear), parseFloat(paymentAmount), parseFloat(presentValue), parseFloat(futureValue), parseFloat(dueEndOrBeginning), parseFloat(interest));
65
document.loandata.APR.value=ans;
66
//alert(futureValue);
67
68
69
}
70
71
</script>
72
</body>
73
</html>
74
Advertisement
Answer
In case anyone is still looking for a javascript implementation of the Excel’s Rate function, this is what I came up with:
JavaScript
1
54
54
1
var rate = function(nper, pmt, pv, fv, type, guess) {
2
// Sets default values for missing parameters
3
fv = typeof fv !== 'undefined' ? fv : 0;
4
type = typeof type !== 'undefined' ? type : 0;
5
guess = typeof guess !== 'undefined' ? guess : 0.1;
6
7
// Sets the limits for possible guesses to any
8
// number between 0% and 100%
9
var lowLimit = 0;
10
var highLimit = 1;
11
12
// Defines a tolerance of up to +/- 0.00005% of pmt, to accept
13
// the solution as valid.
14
var tolerance = Math.abs(0.00000005 * pmt);
15
16
// Tries at most 40 times to find a solution within the tolerance.
17
for (var i = 0; i < 40; i++) {
18
// Resets the balance to the original pv.
19
var balance = pv;
20
21
// Calculates the balance at the end of the loan, based
22
// on loan conditions.
23
for (var j = 0; j < nper; j++ ) {
24
if (type == 0) {
25
// Interests applied before payment
26
balance = balance * (1 + guess) + pmt;
27
} else {
28
// Payments applied before insterests
29
balance = (balance + pmt) * (1 + guess);
30
}
31
}
32
33
// Returns the guess if balance is within tolerance. If not, adjusts
34
// the limits and starts with a new guess.
35
if (Math.abs(balance + fv) < tolerance) {
36
return guess;
37
} else if (balance + fv > 0) {
38
// Sets a new highLimit knowing that
39
// the current guess was too big.
40
highLimit = guess;
41
} else {
42
// Sets a new lowLimit knowing that
43
// the current guess was too small.
44
lowLimit = guess;
45
}
46
47
// Calculates the new guess.
48
guess = (highLimit + lowLimit) / 2;
49
}
50
51
// Returns null if no acceptable result was found after 40 tries.
52
return null;
53
};
54
Testing the function with Abdul’s test cases give the following results:
JavaScript
1
15
15
1
rate(360,-665.3, 99000);
2
0.005916521358085446
3
4
rate(360,-958.63, 192000);
5
0.0036458502960158515
6
7
rate(180,-1302.96,192000);
8
0.0022917255526408564
9
10
rate(360, -889.19, 192000);
11
0.0031250616819306744
12
13
rate(360, -1145.8, 240000);
14
0.003333353153720964
15