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

Добавлены функции: PPMT, MIRR, IPMT, DB, DDB, SLN.

исправлена ошибка при переносе ячеек с формулами.

git-svn-id: svn://192.168.3.15/activex/AVS/Sources/TeamlabOffice/trunk/OfficeWeb@55377 954022d7-b5bf-4e40-9824-e11837661b57
parent fea5960a
...@@ -19,152 +19,154 @@ ...@@ -19,152 +19,154 @@
//]]> //]]>
</script> </script>
<script type="text/javascript" src="../../Common/browser.js"></script> <script type="text/javascript" src="../../Common/browser.js"></script>
<script src="../../Common/downloaderfiles.js"></script> <script src="../../Common/downloaderfiles.js"></script>
<script type="text/javascript" src="../../Common/3rdparty/Underscore/underscore-min.js"></script> <script type="text/javascript" src="../../Common/3rdparty/Underscore/underscore-min.js"></script>
<script type="text/javascript" src="../../Common/3rdparty/Sockjs/sockjs-0.3.min.js"></script> <script type="text/javascript" src="../../Common/3rdparty/Sockjs/sockjs-0.3.min.js"></script>
<script type="text/javascript" src="../../Common/docscoapicommon.js"></script> <script type="text/javascript" src="../../Common/docscoapicommon.js"></script>
<script type="text/javascript" src="../../Common/docscoapi.js"></script> <script type="text/javascript" src="../../Common/docscoapi.js"></script>
<script type="text/javascript" src="../../Common/Charts/DrawingArea.js"></script> <script type="text/javascript" src="../../Common/Charts/DrawingArea.js"></script>
<script type="text/javascript" src="../../Common/Charts/DrawingObjects.js"></script> <script type="text/javascript" src="../../Common/Charts/DrawingObjects.js"></script>
<script type="text/javascript" src="../../Common/Charts/charts.js"></script> <script type="text/javascript" src="../../Common/Charts/charts.js"></script>
<script type="text/javascript" src="../../Common/Charts/ChartsDrawer.js"></script> <script type="text/javascript" src="../../Common/Charts/ChartsDrawer.js"></script>
<script type="text/javascript" src="../../Common/commonDefines.js"></script> <script type="text/javascript" src="../../Common/commonDefines.js"></script>
<script type="text/javascript" src="../../Common/editorscommon.js"></script> <script type="text/javascript" src="../../Common/editorscommon.js"></script>
<script type="text/javascript" src="../../Common/apiCommon.js"></script> <script type="text/javascript" src="../../Common/apiCommon.js"></script>
<script type="text/javascript" src="../../Common/FontsFreeType/font_engine.js"></script> <script type="text/javascript" src="../../Common/FontsFreeType/font_engine.js"></script>
<script type="text/javascript" src="../../Common/FontsFreeType/FontFile.js"></script> <script type="text/javascript" src="../../Common/FontsFreeType/FontFile.js"></script>
<script type="text/javascript" src="../../Common/FontsFreeType/FontManager.js"></script> <script type="text/javascript" src="../../Common/FontsFreeType/FontManager.js"></script>
<script type="text/javascript" src="../../Word/Drawing/HatchPattern.js"></script> <script type="text/javascript" src="../../Word/Drawing/HatchPattern.js"></script>
<script type="text/javascript" src="../../Common/AllFonts.js"></script> <script type="text/javascript" src="../../Common/AllFonts.js"></script>
<script type="text/javascript" src="../../Word/Drawing/Externals.js"></script> <script type="text/javascript" src="../../Word/Drawing/Externals.js"></script>
<script type="text/javascript" src="../../Word/Drawing/Metafile.js"></script> <script type="text/javascript" src="../../Word/Drawing/Metafile.js"></script>
<script type="text/javascript" src="../../Common/trackFile.js"></script> <script type="text/javascript" src="../../Common/trackFile.js"></script>
<script type="text/javascript" src="../apiDefines.js"></script> <script type="text/javascript" src="../apiDefines.js"></script>
<script type="text/javascript" src="../utils/utils.js"></script> <script type="text/javascript" src="../utils/utils.js"></script>
<script type="text/javascript" src="../model/clipboard.js"></script> <script type="text/javascript" src="../model/clipboard.js"></script>
<script type="text/javascript" src="../model/autofilters.js"></script> <script type="text/javascript" src="../model/autofilters.js"></script>
<script type="text/javascript" src="../graphics/DrawingContext.js"></script> <script type="text/javascript" src="../graphics/DrawingContext.js"></script>
<script type="text/javascript" src="../graphics/pdfprinter.js"></script> <script type="text/javascript" src="../graphics/pdfprinter.js"></script>
<script type="text/javascript" src="../../OfflineDocuments/Excel/test-workbook2/editor.js"></script> <script type="text/javascript" src="../../OfflineDocuments/Excel/test-workbook2/editor.js"></script>
<script type="text/javascript" src="../model/CollaborativeEditing.js"></script> <script type="text/javascript" src="../model/CollaborativeEditing.js"></script>
<script type="text/javascript" src="../model/ConditionalFormatting.js"></script> <script type="text/javascript" src="../model/ConditionalFormatting.js"></script>
<script type="text/javascript" src="../model/FormulaObjects/parserFormula.js"></script> <script type="text/javascript" src="../model/FormulaObjects/parserFormula.js"></script>
<script type="text/javascript" src="../model/FormulaObjects/dateandtimeFunctions.js"></script> <script type="text/javascript" src="../model/FormulaObjects/dateandtimeFunctions.js"></script>
<script type="text/javascript" src="../model/FormulaObjects/engineeringFunctions.js"></script> <script type="text/javascript" src="../model/FormulaObjects/engineeringFunctions.js"></script>
<script type="text/javascript" src="../model/FormulaObjects/cubeFunctions.js"></script> <script type="text/javascript" src="../model/FormulaObjects/cubeFunctions.js"></script>
<script type="text/javascript" src="../model/FormulaObjects/databaseFunctions.js"></script> <script type="text/javascript" src="../model/FormulaObjects/databaseFunctions.js"></script>
<script type="text/javascript" src="../model/FormulaObjects/textanddataFunctions.js"></script> <script type="text/javascript" src="../model/FormulaObjects/textanddataFunctions.js"></script>
<script type="text/javascript" src="../model/FormulaObjects/statisticalFunctions.js"></script> <script type="text/javascript" src="../model/FormulaObjects/statisticalFunctions.js"></script>
<script type="text/javascript" src="../model/FormulaObjects/financialFunctions.js"></script> <script type="text/javascript" src="../model/FormulaObjects/financialFunctions.js"></script>
<script type="text/javascript" src="../model/FormulaObjects/mathematicFunctions.js"></script> <script type="text/javascript" src="../model/FormulaObjects/mathematicFunctions.js"></script>
<script type="text/javascript" src="../model/FormulaObjects/lookupandreferenceFunctions.js"></script> <script type="text/javascript" src="../model/FormulaObjects/lookupandreferenceFunctions.js"></script>
<script type="text/javascript" src="../model/FormulaObjects/informationFunctions.js"></script> <script type="text/javascript" src="../model/FormulaObjects/informationFunctions.js"></script>
<script type="text/javascript" src="../model/FormulaObjects/logicalFunctions.js"></script> <script type="text/javascript" src="../model/FormulaObjects/logicalFunctions.js"></script>
<script type="text/javascript" src="../model/CellComment.js"></script> <script type="text/javascript" src="../model/CellComment.js"></script>
<script type="text/javascript" src="../../Common/NumFormat.js"></script> <script type="text/javascript" src="../../Common/NumFormat.js"></script>
<script type="text/javascript" src="../model/Serialize.js"></script> <script type="text/javascript" src="../model/Serialize.js"></script>
<script type="text/javascript" src="../model/WorkbookElems.js"></script> <script type="text/javascript" src="../model/WorkbookElems.js"></script>
<script type="text/javascript" src="../model/Workbook.js"></script> <script type="text/javascript" src="../model/Workbook.js"></script>
<script type="text/javascript" src="../model/CellInfo.js"></script> <script type="text/javascript" src="../model/CellInfo.js"></script>
<script type="text/javascript" src="../model/AdvancedOptions.js"></script> <script type="text/javascript" src="../model/AdvancedOptions.js"></script>
<script type="text/javascript" src="../model/History.js"></script> <script type="text/javascript" src="../model/History.js"></script>
<script type="text/javascript" src="../model/UndoRedo.js"></script> <script type="text/javascript" src="../model/UndoRedo.js"></script>
<script type="text/javascript" src="../../Common/scroll.js"></script> <script type="text/javascript" src="../../Common/scroll.js"></script>
<script type="text/javascript" src="../view/mobileTouch.js"></script> <script type="text/javascript" src="../view/PopUpSelector.js"></script>
<script type="text/javascript" src="../view/iscroll.js"></script> <script type="text/javascript" src="../view/StringRender.js"></script>
<script type="text/javascript" src="../view/StringRender.js"></script> <script type="text/javascript" src="../view/CellTextRender.js"></script>
<script type="text/javascript" src="../view/CellTextRender.js"></script> <script type="text/javascript" src="../view/CellEditorView.js"></script>
<script type="text/javascript" src="../view/CellEditorView.js"></script> <script type="text/javascript" src="../view/WorksheetView.js"></script>
<script type="text/javascript" src="../view/WorksheetView.js"></script> <script type="text/javascript" src="../view/HandlerList.js"></script>
<script type="text/javascript" src="../view/HandlerList.js"></script> <script type="text/javascript" src="../view/EventsController.js"></script>
<script type="text/javascript" src="../view/EventsController.js"></script> <script type="text/javascript" src="../view/WorkbookView.js"></script>
<script type="text/javascript" src="../view/WorkbookView.js"></script>
<script type="text/javascript" src="../api.js"></script>
<script type="text/javascript" src="../api.js"></script>
<!--for theme-->
<!--for theme--> <script type="text/javascript" src="../../Common/Shapes/EditorSettings.js"></script>
<script type="text/javascript" src="../../Common/Shapes/EditorSettings.js"></script>
<script type="text/javascript" src="../../Common/Shapes/Serialize.js"></script> <script type="text/javascript" src="../../Common/Shapes/Serialize.js"></script>
<script type="text/javascript" src="../../Common/Shapes/SerializeWriter.js"></script> <script type="text/javascript" src="../../Common/Shapes/SerializeWriter.js"></script>
<script type="text/javascript" src="../../Word/Editor/SerializeCommon.js"></script> <script type="text/javascript" src="../../Word/Editor/SerializeCommon.js"></script>
<script type="text/javascript" src="../../Common/Drawings/Math.js"></script> <script type="text/javascript" src="../../Common/Drawings/Math.js"></script>
<script type="text/javascript" src="../../Common/Drawings/ArcTo.js"></script> <script type="text/javascript" src="../../Common/Drawings/ArcTo.js"></script>
<script type="text/javascript" src="../../Word/Drawing/ColorArray.js"></script> <script type="text/javascript" src="../../Word/Drawing/ColorArray.js"></script>
<script type="text/javascript" src="../../Word/apiCommon.js"></script> <script type="text/javascript" src="../../Word/apiCommon.js"></script>
<!--for chart--> <!--for chart-->
<script type="text/javascript" src="../../Common/SerializeCommonWordExcel.js"></script> <script type="text/javascript" src="../../Common/SerializeCommonWordExcel.js"></script>
<script src="../../Common/Charts/libraries/OfficeExcel.common.core.js"></script> <script type="text/javascript" src="../../Common/SerializeChart.js"></script>
<script src="../../Common/Charts/libraries/OfficeExcel.common.core.js"></script>
<script src="../../Common/Charts/libraries/OfficeExcel.common.key.js"></script>
<script src="../../Common/Charts/libraries/OfficeExcel.common.key.js"></script>
<script src="../../Common/Charts/libraries/OfficeExcel.bar.js"></script>
<script src="../../Common/Charts/libraries/OfficeExcel.hbar.js"></script> <script src="../../Common/Charts/libraries/OfficeExcel.bar.js"></script>
<script src="../../Common/Charts/libraries/OfficeExcel.line.js"></script> <script src="../../Common/Charts/libraries/OfficeExcel.hbar.js"></script>
<script src="../../Common/Charts/libraries/OfficeExcel.pie.js"></script> <script src="../../Common/Charts/libraries/OfficeExcel.line.js"></script>
<script src="../../Common/Charts/libraries/OfficeExcel.scatter.js"></script> <script src="../../Common/Charts/libraries/OfficeExcel.pie.js"></script>
<script src="../../Common/Charts/libraries/OfficeExcel.chartProperties.js"></script> <script src="../../Common/Charts/libraries/OfficeExcel.scatter.js"></script>
<script src="../../Common/Charts/libraries/OfficeExcel.chartProperties.js"></script>
<!--for shapes-->
<script src="../../Common/Drawings/Hit.js"></script> <!--for shapes-->
<script src="../../Common/Drawings/States.js"></script> <script src="../../Common/Drawings/Hit.js"></script>
<script src="../../Common/Drawings/TrackObjects/AdjustmentTracks.js"></script> <script src="../../Common/Drawings/States.js"></script>
<script src="../../Common/Drawings/TrackObjects/ResizeTracks.js"></script> <script src="../../Common/Drawings/DrawingObjectsHandlers.js"></script>
<script src="../../Common/Drawings/TrackObjects/RotateTracks.js"></script> <script src="../../Common/Drawings/TrackObjects/AdjustmentTracks.js"></script>
<script src="../../Common/Drawings/TrackObjects/NewShapeTracks.js"></script> <script src="../../Common/Drawings/TrackObjects/ResizeTracks.js"></script>
<script src="../../Common/Drawings/TrackObjects/PolyLine.js"></script> <script src="../../Common/Drawings/TrackObjects/RotateTracks.js"></script>
<script src="../../Common/Drawings/TrackObjects/Spline.js"></script> <script src="../../Common/Drawings/TrackObjects/NewShapeTracks.js"></script>
<script src="../../Common/Drawings/TrackObjects/MoveTracks.js"></script> <script src="../../Common/Drawings/TrackObjects/PolyLine.js"></script>
<script src="../../Common/Drawings/TrackObjects/Spline.js"></script>
<script src="../../Common/Drawings/TrackObjects/MoveTracks.js"></script>
<script src="../../Common/Drawings/CommonController.js"></script>
<script src="../../Common/Drawings/Format/Constants.js"></script> <script src="../../Common/Drawings/Format/Constants.js"></script>
<script src="../../Common/Drawings/Format/Format.js"></script> <script src="../../Common/Drawings/Format/Format.js"></script>
<script src="../../Common/Drawings/Format/CreateGeometry.js"></script> <script src="../../Common/Drawings/Format/CreateGeometry.js"></script>
<script src="../../Common/Drawings/Format/Geometry.js"></script> <script src="../../Common/Drawings/Format/Geometry.js"></script>
<script src="../../Common/Drawings/Format/Path.js"></script> <script src="../../Common/Drawings/Format/Path.js"></script>
<script src="../../Common/Drawings/Format/GroupShape.js"></script> <script src="../../Common/Drawings/Format/Shape.js"></script>
<script src="../../Common/Drawings/Format/Image.js"></script> <script src="../../Common/Drawings/Format/Image.js"></script>
<script src="../../Common/Drawings/Format/Shape.js"></script> <script src="../../Common/Drawings/Format/GroupShape.js"></script>
<script src="../../Common/Drawings/Format/ChartSpace.js"></script> <script src="../../Common/Drawings/Format/ChartSpace.js"></script>
<script src="../../Common/Drawings/Format/ChartFormat.js"></script> <script src="../../Common/Drawings/Format/ChartFormat.js"></script>
<script src="../../Common/Drawings/Format/TextBody.js"></script> <script src="../../Common/Drawings/Format/TextBody.js"></script>
<script src="../../Common/Drawings/CommonController.js"></script> <script src="../view/DrawingObjectsController.js"></script>
<script src="../view/DrawingObjectsController.js"></script>
<script src="../model/DrawingObjects/Graphics.js"></script> <script src="../model/DrawingObjects/Graphics.js"></script>
<script src="../model/DrawingObjects/Overlay.js"></script> <script src="../model/DrawingObjects/Overlay.js"></script>
<script src="../../Word/Drawing/Controls.js"></script> <script src="../../Word/Drawing/Controls.js"></script>
<script src="../model/DrawingObjects/ShapeDrawer.js"></script> <script src="../model/DrawingObjects/ShapeDrawer.js"></script>
<script src="../../Word/Editor/CollaborativeEditing.js"></script> <script src="../../Word/Editor/CollaborativeEditing.js"></script>
<script src="../model/DrawingObjects/DrawingDocument.js"></script>
<script src="../model/DrawingObjects/GlobalLoaders.js"></script>
<script src="../model/DrawingObjects/DrawingDocument.js"></script>
<script src="../model/DrawingObjects/GlobalLoaders.js"></script>
<script src="../model/DrawingObjects/Format/ShapePrototype.js"></script> <script src="../model/DrawingObjects/Format/ShapePrototype.js"></script>
<script src="../model/DrawingObjects/Format/ImagePrototype.js"></script> <script src="../model/DrawingObjects/Format/ImagePrototype.js"></script>
<script src="../model/DrawingObjects/Format/GroupPrototype.js"></script> <script src="../model/DrawingObjects/Format/GroupPrototype.js"></script>
<script src="../model/DrawingObjects/Format/ChartSpacePrototype.js"></script> <script src="../model/DrawingObjects/Format/ChartSpacePrototype.js"></script>
<script src="../../Word/Editor/Styles.js"></script> <script src="../../Word/Editor/Styles.js"></script>
<script src="../../Word/Editor/Numbering.js"></script> <script src="../../Word/Editor/Numbering.js"></script>
<script src="../../Word/Editor/ParagraphContent.js"></script> <script src="../../Word/Editor/ParagraphContent.js"></script>
<script src="../../Word/Editor/Run.js"></script> <script src="../../Word/Editor/Run.js"></script>
<script src="../../Word/Editor/Hyperlink.js"></script> <script src="../../Word/Editor/Hyperlink.js"></script>
<script src="../../Word/Editor/Comments.js"></script> <script src="../../Word/Editor/Comments.js"></script>
<script src="../../Word/Editor/FlowObjects.js"></script> <script src="../../Word/Editor/FlowObjects.js"></script>
...@@ -173,7 +175,7 @@ ...@@ -173,7 +175,7 @@
<script src="../../Word/Editor/DocumentContent.js"></script> <script src="../../Word/Editor/DocumentContent.js"></script>
<script src="../../Word/Editor/HeaderFooter.js"></script> <script src="../../Word/Editor/HeaderFooter.js"></script>
<script src="../../Word/Editor/Table.js"></script> <script src="../../Word/Editor/Table.js"></script>
<!-- <script src="../../Word/Editor/Spelling.js"></script> --> <script src="../../Word/Editor/Spelling.js"></script>
<script src="../../Word/Editor/FontClassification.js"></script> <script src="../../Word/Editor/FontClassification.js"></script>
<script type="text/javascript" src="FormulaTests.js"></script> <script type="text/javascript" src="FormulaTests.js"></script>
......
...@@ -279,6 +279,43 @@ ...@@ -279,6 +279,43 @@
} }
function _coupnum( settlement, maturity, frequency, basis ) {
basis = ( basis !== undefined ? basis : 0 );
var n = new Date(maturity);
_lcl_GetCouppcd( settlement, n, frequency );
var nMonths = (maturity.getFullYear() - n.getFullYear()) * 12 + maturity.getMonth() - n.getMonth();
return nMonths * frequency / 12 ;
}
function _duration( settlement, maturity, coupon, yld, frequency, basis ){
var yearfrac = _yearFrac( new Date( settlement ), new Date( maturity ), basis );
var numOfCoups = _coupnum( new Date( settlement ), new Date( maturity ), frequency );
var duration = 0, f100 = 100;
coupon *= f100 / frequency;
yld /= frequency;
yld += 1;
var nDiff = yearfrac * frequency - numOfCoups;
var p = 0;
for ( var i = 1; i < numOfCoups; i++ ) {
duration += ( i + nDiff ) * ( coupon ) / Math.pow( yld, i + nDiff );
p += coupon / Math.pow( yld, i + nDiff );
}
duration += ( numOfCoups + nDiff ) * ( coupon + f100 ) / Math.pow( yld, numOfCoups + nDiff );
p += ( coupon + f100 ) / Math.pow( yld, numOfCoups + nDiff );
duration = duration / p / frequency;
return duration;
}
var ver = 2; var ver = 2;
var oParser, wb, ws, date1, date2, dif = 1e-9, var oParser, wb, ws, date1, date2, dif = 1e-9,
...@@ -4032,20 +4069,9 @@ ...@@ -4032,20 +4069,9 @@
test( "Test: \"COUPNUM\"", function () { test( "Test: \"COUPNUM\"", function () {
function coupnum( settlement, maturity, frequency, basis ) {
basis = ( basis !== undefined ? basis : 0 );
var n = new Date(maturity);
_lcl_GetCouppcd( settlement, n, frequency );
var nMonths = (maturity.getFullYear() - n.getFullYear()) * 12 + maturity.getMonth() - n.getMonth();
return nMonths * frequency / 12 ;
}
oParser = new parserFormula( "COUPNUM(DATE(2007,1,25),DATE(2008,11,15),2,1)", "A2", ws ); oParser = new parserFormula( "COUPNUM(DATE(2007,1,25),DATE(2008,11,15),2,1)", "A2", ws );
ok( oParser.parse() ); ok( oParser.parse() );
strictEqual( oParser.calculate().getValue(), coupnum( new Date(2007,0,25), new Date(2008,10,15), 2, 1 ) ); strictEqual( oParser.calculate().getValue(), _coupnum( new Date(2007,0,25), new Date(2008,10,15), 2, 1 ) );
} ) } )
...@@ -4090,9 +4116,9 @@ ...@@ -4090,9 +4116,9 @@
function pricemat( settl, matur, iss, rate, yld, basis ) { function pricemat( settl, matur, iss, rate, yld, basis ) {
var fIssMat = yearFrac( new Date(iss), new Date(matur), basis ); var fIssMat = _yearFrac( new Date(iss), new Date(matur), basis );
var fIssSet = yearFrac( new Date(iss), new Date(settl), basis ); var fIssSet = _yearFrac( new Date(iss), new Date(settl), basis );
var fSetMat = yearFrac( new Date(settl), new Date(matur), basis ); var fSetMat = _yearFrac( new Date(settl), new Date(matur), basis );
var res = 1.0 + fIssMat * rate; var res = 1.0 + fIssMat * rate;
res /= 1.0 + fSetMat * yld; res /= 1.0 + fSetMat * yld;
...@@ -4132,7 +4158,6 @@ ...@@ -4132,7 +4158,6 @@
} ) } )
test( "Test: \"YIELDMAT\"", function () { test( "Test: \"YIELDMAT\"", function () {
oParser = new parserFormula( "YIELDMAT(DATE(2008,3,15),DATE(2008,11,3),DATE(2007,11,8),0.0625,100.0123,0)", "A2", ws ); oParser = new parserFormula( "YIELDMAT(DATE(2008,3,15),DATE(2008,11,3),DATE(2007,11,8),0.0625,100.0123,0)", "A2", ws );
...@@ -4166,9 +4191,9 @@ ...@@ -4166,9 +4191,9 @@
function oddlyield( settlement, maturity, last_interest, rate, pr, redemption, frequency, basis ){ function oddlyield( settlement, maturity, last_interest, rate, pr, redemption, frequency, basis ){
var fDCi = yearFrac( last_interest, maturity, basis ) * frequency; var fDCi = _yearFrac( last_interest, maturity, basis ) * frequency;
var fDSCi = yearFrac( settlement, maturity, basis ) * frequency; var fDSCi = _yearFrac( settlement, maturity, basis ) * frequency;
var fAi = yearFrac( last_interest, settlement, basis ) * frequency; var fAi = _yearFrac( last_interest, settlement, basis ) * frequency;
var res = redemption + fDCi * 100.0 * rate / frequency; var res = redemption + fDCi * 100.0 * rate / frequency;
res /= pr + fAi * 100.0 * rate / frequency; res /= pr + fAi * 100.0 * rate / frequency;
...@@ -4184,6 +4209,28 @@ ...@@ -4184,6 +4209,28 @@
} ) } )
test( "Test: \"DURATION\"", function () {
oParser = new parserFormula( "DURATION(DATE(2008,1,1),DATE(2016,1,1),0.08,0.09,2,1)", "A2", ws );
ok( oParser.parse() );
strictEqual( oParser.calculate().getValue(), _duration( new Date(2008,0,1), new Date(2016,0,1), 0.08, 0.09, 2, 1 ) );
} )
test( "Test: \"MDURATION\"", function () {
function mduration(settl, matur, coupon, yld, frequency, basis){
return _duration( settl, matur, coupon, yld, frequency, basis ) / (1 + yld/frequency);
}
oParser = new parserFormula( "MDURATION(DATE(2008,1,1),DATE(2016,1,1),0.08,0.09,2,1)", "A2", ws );
ok( oParser.parse() );
strictEqual( oParser.calculate().getValue(), mduration( new Date(2008,0,1), new Date(2016,0,1), 0.08, 0.09, 2, 1 ) );
} )
test( "Test: \"SYD\"", function () { test( "Test: \"SYD\"", function () {
function syd( cost, salvage, life, per ){ function syd( cost, salvage, life, per ){
...@@ -4213,5 +4260,4 @@ ...@@ -4213,5 +4260,4 @@
} ) } )
} ); } );
"use strict"; "use strict";
function ConvertToDec( aStrSource, nBase, nCharLim ){
if ( nBase < 2 || nBase > 36 )
return "Error #1";
var nStrLen = aStrSource.length;
if( nStrLen > nCharLim )
return "Error #2";
else if( !nStrLen )
return 0;
var fVal = 0;
var p = aStrSource;
var nFirstDig = 0;
var bFirstDig = true;
var fBase = nBase;
for(var i=0; i < p.length; i++)
{
var n;
if( '0' <= p[i] && p[i] <= '9' )
n = p[i].charCodeAt() - '0'.charCodeAt();
else if( 'A' <= p[i] && p[i] <= 'Z' )
n = 10 + ( p[i].charCodeAt() - 'A'.charCodeAt() );
else if ( 'a' <= p[i] && p[i] <= 'z' )
n = 10 + ( p[i].charCodeAt() - 'a'.charCodeAt() );
else
n = nBase;
if( n < nBase )
{
if( bFirstDig )
{
bFirstDig = false;
nFirstDig = n;
}
fVal = fVal * fBase + n;
}
else
return "Error #3"
}
if( nStrLen == nCharLim && !bFirstDig && (nFirstDig >= nBase / 2) )
{ // handling negativ values
fVal = ( Math.pow( nBase, nCharLim ) - fVal ); // complement
fVal *= -1.0;
}
return fVal;
}
/** /**
* Created with JetBrains WebStorm. * Created with JetBrains WebStorm.
* User: Dmitry.Shahtanov * User: Dmitry.Shahtanov
......
"use strict"; "use strict";
function GetRmz( fZins, fZzr, fBw, fZw, nF ) { function getPMT( rate, nper, pv, fv, flag ) {
var fRmz; var res;
if ( fZins == 0.0 ) if ( rate == 0 )
fRmz = ( fBw + fZw ) / fZzr; res = ( pv + fv ) / nper;
else { else {
var fTerm = Math.pow( 1.0 + fZins, fZzr ); var part = Math.pow( 1 + rate, nper );
if ( nF > 0 ) if ( flag > 0 )
fRmz = ( fZw * fZins / ( fTerm - 1.0 ) + fBw * fZins / ( 1.0 - 1.0 / fTerm ) ) / ( 1.0 + fZins ); res = ( fv * rate / ( part - 1 ) + pv * rate / ( 1 - 1 / part ) ) / ( 1 + rate );
else else
fRmz = fZw * fZins / ( fTerm - 1.0 ) + fBw * fZins / ( 1.0 - 1.0 / fTerm ); res = fv * rate / ( part - 1 ) + pv * rate / ( 1 - 1 / part );
} }
return -fRmz; return -res;
} }
function GetZw( fZins, fZzr, fRmz, fBw, nF ) { function getFV( rate, nper, pmt, pv, type ) {
var fZw; var res;
if ( fZins == 0.0 ) if ( rate == 0 )
fZw = fBw + fRmz * fZzr; res = pv + pmt * nper;
else { else {
var fTerm = Math.pow( 1.0 + fZins, fZzr ); var part = Math.pow( 1 + rate, nper );
if ( nF > 0 ) if ( type > 0 )
fZw = fBw * fTerm + fRmz * ( 1.0 + fZins ) * ( fTerm - 1.0 ) / fZins; res = pv * part + pmt * ( 1 + rate ) * ( part - 1 ) / rate;
else else
fZw = fBw * fTerm + fRmz * ( fTerm - 1.0 ) / fZins; res = pv * part + pmt * ( part - 1 ) / rate;
} }
return -fZw; return -res;
} }
function RateIteration( fNper, fPayment, fPv, fFv, fPayType, fGuess ) { function getDDB( cost, salvage, life, period, factor ) {
var ddb, ipmt, oldCost, newCost;
ipmt = factor / life;
if ( ipmt >= 1 ) {
ipmt = 1;
if ( period == 1 )
oldCost = cost;
else
oldCost = 0;
}
else
oldCost = cost * Math.pow( 1 - ipmt, period - 1 );
newCost = cost * Math.pow( 1 - ipmt, period );
if ( newCost < salvage )
ddb = oldCost - salvage;
else
ddb = oldCost - newCost;
if ( ddb < 0 )
ddb = 0;
return ddb;
}
function getIPMT(rate, per, pv, type, pmt) {
var ipmt;
if ( per == 1 ) {
if ( type > 0 )
ipmt = 0;
else
ipmt = -pv;
}
else {
if ( type > 0 )
ipmt = getFV( rate, per - 2, pmt, pv, 1 ) - pmt;
else
ipmt = getFV( rate, per - 1, pmt, pv, 0 );
}
return ipmt * rate
}
function RateIteration( nper, payment, pv, fv, payType, guess ) {
function approxEqual( a, b ) { function approxEqual( a, b ) {
if ( a == b ) if ( a == b )
return true; return true;
var x = a - b; var x = a - b;
return (x < 0.0 ? -x : x) // return (x < 0 ? -x : x) < ( (a < 0 ? -a : a) * (1.0 / (16777216.0 * 16777216.0)));
< ((a < 0.0 ? -a : a) * (1.0 / (16777216.0 * 16777216.0))); return Math.abs(x) < ( Math.abs(a) * (1 / (16777216 * 16777216) ) ) ;
} }
var bValid = true, bFound = false, fX, fXnew, fTerm, fTermDerivation, fGeoSeries, fGeoSeriesDerivation; var bValid = true, bFound = false, fX, fXnew, fTerm, fTermDerivation, fGeoSeries, fGeoSeriesDerivation;
var nIterationsMax = 150, nCount = 0, fEpsilonSmall = 1.0E-14, SCdEpsilon = 1.0E-7; var iterationMax = 150, nCount = 0, minEps = 1.0E-14, eps = 1.0E-7;
fFv = fFv - fPayment * fPayType; fv = fv - payment * payType;
fPv = fPv + fPayment * fPayType; pv = pv + payment * payType;
if ( fNper == Math.round( fNper ) ) { if ( nper == Math.round( nper ) ) {
fX = fGuess.fGuess; fX = guess.fGuess;
var fPowN, fPowNminus1; var fPowN, fPowNminus1;
while ( !bFound && nCount < nIterationsMax ) { while ( !bFound && nCount < iterationMax ) {
fPowNminus1 = Math.pow( 1.0 + fX, fNper - 1.0 ); fPowNminus1 = Math.pow( 1 + fX, nper - 1 );
fPowN = fPowNminus1 * (1.0 + fX); fPowN = fPowNminus1 * (1 + fX);
if ( approxEqual( Math.abs( fX ), 0.0 ) ) { if ( approxEqual( Math.abs( fX ), 0 ) ) {
fGeoSeries = fNper; fGeoSeries = nper;
fGeoSeriesDerivation = fNper * (fNper - 1.0) / 2.0; fGeoSeriesDerivation = nper * (nper - 1) / 2;
} }
else { else {
fGeoSeries = (fPowN - 1.0) / fX; fGeoSeries = (fPowN - 1) / fX;
fGeoSeriesDerivation = fNper * fPowNminus1 / fX - fGeoSeries / fX; fGeoSeriesDerivation = nper * fPowNminus1 / fX - fGeoSeries / fX;
} }
fTerm = fFv + fPv * fPowN + fPayment * fGeoSeries; fTerm = fv + pv * fPowN + payment * fGeoSeries;
fTermDerivation = fPv * fNper * fPowNminus1 + fPayment * fGeoSeriesDerivation; fTermDerivation = pv * nper * fPowNminus1 + payment * fGeoSeriesDerivation;
if ( Math.abs( fTerm ) < fEpsilonSmall ) if ( Math.abs( fTerm ) < minEps )
bFound = true; bFound = true;
else { else {
if ( approxEqual( Math.abs( fTermDerivation ), 0.0 ) ) if ( approxEqual( Math.abs( fTermDerivation ), 0 ) )
fXnew = fX + 1.1 * SCdEpsilon; fXnew = fX + 1.1 * eps;
else else
fXnew = fX - fTerm / fTermDerivation; fXnew = fX - fTerm / fTermDerivation;
nCount++; nCount++;
bFound = (Math.abs( fXnew - fX ) < SCdEpsilon); bFound = (Math.abs( fXnew - fX ) < eps);
fX = fXnew; fX = fXnew;
} }
} }
bValid = (fX >= -1.0); bValid = (fX >= -1);
} }
else { else {
fX = (fGuess.fGuest < -1.0) ? -1.0 : fGuess.fGuest; fX = (guess.fGuest < -1) ? -1 : guess.fGuest;
while ( bValid && !bFound && nCount < nIterationsMax ) { while ( bValid && !bFound && nCount < iterationMax ) {
if ( approxEqual( Math.abs( fX ), 0.0 ) ) { if ( approxEqual( Math.abs( fX ), 0 ) ) {
fGeoSeries = fNper; fGeoSeries = nper;
fGeoSeriesDerivation = fNper * (fNper - 1.0) / 2.0; fGeoSeriesDerivation = nper * (nper - 1) / 2;
} }
else { else {
fGeoSeries = (Math.pow( 1.0 + fX, fNper ) - 1.0) / fX; fGeoSeries = (Math.pow( 1 + fX, nper ) - 1) / fX;
fGeoSeriesDerivation = fNper * Math.pow( 1.0 + fX, fNper - 1.0 ) / fX - fGeoSeries / fX; fGeoSeriesDerivation = nper * Math.pow( 1 + fX, nper - 1 ) / fX - fGeoSeries / fX;
} }
fTerm = fFv + fPv * pow( 1.0 + fX, fNper ) + fPayment * fGeoSeries; fTerm = fv + pv * pow( 1 + fX, nper ) + payment * fGeoSeries;
fTermDerivation = fPv * fNper * Math.pow( 1.0 + fX, fNper - 1.0 ) + fPayment * fGeoSeriesDerivation; fTermDerivation = pv * nper * Math.pow( 1 + fX, nper - 1 ) + payment * fGeoSeriesDerivation;
if ( Math.abs( fTerm ) < fEpsilonSmall ) if ( Math.abs( fTerm ) < minEps )
bFound = true; bFound = true;
else { else {
if ( approxEqual( Math.abs( fTermDerivation ), 0.0 ) ) if ( approxEqual( Math.abs( fTermDerivation ), 0 ) )
fXnew = fX + 1.1 * SCdEpsilon; fXnew = fX + 1.1 * eps;
else else
fXnew = fX - fTerm / fTermDerivation; fXnew = fX - fTerm / fTermDerivation;
nCount++; nCount++;
bFound = (Math.abs( fXnew - fX ) < SCdEpsilon); bFound = (Math.abs( fXnew - fX ) < eps);
fX = fXnew; fX = fXnew;
bValid = (fX >= -1.0); bValid = (fX >= -1);
} }
} }
} }
fGuess.fGuess = fX; guess.fGuess = fX;
return bValid && bFound; return bValid && bFound;
} }
...@@ -137,8 +178,8 @@ function getcoupdays( settl, matur, frequency, basis ) { ...@@ -137,8 +178,8 @@ function getcoupdays( settl, matur, frequency, basis ) {
function getcoupnum( settl, matur, frequency ) { function getcoupnum( settl, matur, frequency ) {
var n = new Date( matur ); var n = new Date( matur );
lcl_GetCouppcd( settl, n, frequency ); lcl_GetCouppcd( settl, n, frequency );
var nMonths = (matur.getFullYear() - n.getFullYear()) * 12 + matur.getMonth() - n.getMonth(); var months = (matur.getFullYear() - n.getFullYear()) * 12 + matur.getMonth() - n.getMonth();
return Math.ceil( nMonths * frequency / 12 ); return Math.ceil( months * frequency / 12 );
} }
function getcoupdaysnc( settl, matur, frequency, basis ) { function getcoupdaysnc( settl, matur, frequency, basis ) {
...@@ -153,77 +194,75 @@ function getcoupdaysnc( settl, matur, frequency, basis ) { ...@@ -153,77 +194,75 @@ function getcoupdaysnc( settl, matur, frequency, basis ) {
return getcoupdays( new Date( settl ), new Date( matur ), frequency, basis ) - getcoupdaybs( new Date( settl ), new Date( matur ), frequency, basis ); return getcoupdays( new Date( settl ), new Date( matur ), frequency, basis ) - getcoupdaybs( new Date( settl ), new Date( matur ), frequency, basis );
} }
function getprice( nSettle, nMat, fRate, fYield, fRedemp, nFreq, nBase ) { function getprice( settle, mat, rate, _yield, redemp, freq, base ) {
var fE = getcoupdays( new Date( nSettle ), new Date( nMat ), nFreq, nBase ); var _coupdays = getcoupdays( new Date( settle ), new Date( mat ), freq, base );
var fDSC_E = getcoupdaysnc( new Date( nSettle ), new Date( nMat ), nFreq, nBase ) / fE; var _coupdaysnc = getcoupdaysnc( new Date( settle ), new Date( mat ), freq, base ) / _coupdays;
var fN = getcoupnum( new Date( nSettle ), (nMat), nFreq, nBase ); var _coupnum = getcoupnum( new Date( settle ), (mat), freq, base );
var fA = getcoupdaybs( new Date( nSettle ), new Date( nMat ), nFreq, nBase ); var _coupdaybs = getcoupdaybs( new Date( settle ), new Date( mat ), freq, base );
var fRet = fRedemp / ( Math.pow( 1.0 + fYield / nFreq, fN - 1.0 + fDSC_E ) ); var res = redemp / ( Math.pow( 1 + _yield / freq, _coupnum - 1 + _coupdaysnc ) );
fRet -= 100.0 * fRate / nFreq * fA / fE; res -= 100 * rate / freq * _coupdaybs / _coupdays;
var fT1 = 100.0 * fRate / nFreq; var fT1 = 100 * rate / freq;
var fT2 = 1.0 + fYield / nFreq; var fT2 = 1 + _yield / freq;
for ( var fK = 0.0; fK < fN; fK++ ) { for ( var i = 0; i < _coupnum; i++ ) {
fRet += fT1 / Math.pow( fT2, fK + fDSC_E ); res += fT1 / Math.pow( fT2, i + _coupdaysnc );
} }
return fRet; return res;
} }
function getYield( nSettle, nMat, fCoup, fPrice, fRedemp, nFreq, nBase ) { function getYield( settle, mat, coup, price, redemp, freq, base ) {
var fRate = fCoup, fPriceN = 0.0, fYield1 = 0.0, fYield2 = 1.0; var rate = coup, priceN = 0, yield1 = 0, yield2 = 1;
var fPrice1 = getprice( nSettle, nMat, fRate, fYield1, fRedemp, nFreq, nBase ); var price1 = getprice( settle, mat, rate, yield1, redemp, freq, base );
var fPrice2 = getprice( nSettle, nMat, fRate, fYield2, fRedemp, nFreq, nBase ); var price2 = getprice( settle, mat, rate, yield2, redemp, freq, base );
var fYieldN = ( fYield2 - fYield1 ) * 0.5; var yieldN = ( yield2 - yield1 ) * 0.5;
for ( var nIter = 0; nIter < 100 && fPriceN != fPrice; nIter++ ) { for ( var i = 0; i < 100 && priceN != price; i++ ) {
fPriceN = getprice( nSettle, nMat, fRate, fYieldN, fRedemp, nFreq, nBase ); priceN = getprice( settle, mat, rate, yieldN, redemp, freq, base );
if ( fPrice == fPrice1 ) if ( price == price1 )
return fYield1; return yield1;
else if ( fPrice == fPrice2 ) else if ( price == price2 )
return fYield2; return yield2;
else if ( fPrice == fPriceN ) else if ( price == priceN )
return fYieldN; return yieldN;
else if ( fPrice < fPrice2 ) { else if ( price < price2 ) {
fYield2 *= 2.0; yield2 *= 2.0;
fPrice2 = getprice( nSettle, nMat, fRate, fYield2, fRedemp, nFreq, nBase ); price2 = getprice( settle, mat, rate, yield2, redemp, freq, base );
fYieldN = ( fYield2 - fYield1 ) * 0.5; yieldN = ( yield2 - yield1 ) * 0.5;
} }
else { else {
if ( fPrice < fPriceN ) { if ( price < priceN ) {
fYield1 = fYieldN; yield1 = yieldN;
fPrice1 = fPriceN; price1 = priceN;
} }
else { else {
fYield2 = fYieldN; yield2 = yieldN;
fPrice2 = fPriceN; price2 = priceN;
} }
fYieldN = fYield2 - ( fYield2 - fYield1 ) * ( ( fPrice - fPrice2 ) / ( fPrice1 - fPrice2 ) ); yieldN = yield2 - ( yield2 - yield1 ) * ( ( price - price2 ) / ( price1 - price2 ) );
} }
} }
if ( Math.abs( fPrice - fPriceN ) > fPrice / 100.0 ) if ( Math.abs( price - priceN ) > price / 100 )
return new cError( cErrorType.not_numeric ); // result not precise enough return new cError( cErrorType.not_numeric ); // result not precise enough
return new cNumber( fYieldN ); return new cNumber( yieldN );
} }
function getyieldmat( nSettle, nMat, nIssue, fRate, fPrice, nBase ){ function getyieldmat( settle, mat, issue, rate, price, base ){
var fIssMat = yearFrac( nIssue, nMat, nBase ); var issMat = yearFrac( issue, mat, base );
var fIssSet = yearFrac( nIssue, nSettle, nBase ); var issSet = yearFrac( issue, settle, base );
var fSetMat = yearFrac( nSettle, nMat, nBase ); var setMat = yearFrac( settle, mat, base );
var y = 1.0 + fIssMat * fRate; var y = (1 + issMat * rate) / (price / 100 + issSet * rate) - 1;
y /= fPrice / 100.0 + fIssSet * fRate; y /= setMat;
y--;
y /= fSetMat;
return y; return y;
} }
...@@ -233,24 +272,23 @@ function getduration( settlement, maturity, coupon, yld, frequency, basis ){ ...@@ -233,24 +272,23 @@ function getduration( settlement, maturity, coupon, yld, frequency, basis ){
var yearfrac = yearFrac( new Date( settlement ), new Date( maturity ), basis ).getValue(); var yearfrac = yearFrac( new Date( settlement ), new Date( maturity ), basis ).getValue();
var numOfCoups = getcoupnum( new Date( settlement ), new Date( maturity ), frequency ); var numOfCoups = getcoupnum( new Date( settlement ), new Date( maturity ), frequency );
var duration = 0, f100 = 100; var duration = 0, f100 = 100;
coupon *= f100 / frequency; // fCoup is used as cash flow coupon *= f100 / frequency;
yld /= frequency; yld /= frequency;
yld += 1; yld += 1;
var nDiff = yearfrac * frequency - numOfCoups; var nDiff = yearfrac * frequency - numOfCoups;
var t, p = 0; var p = 0;
for ( t = 1; t < numOfCoups; t++ ) { for ( var i = 1; i < numOfCoups; i++ ) {
duration += ( t + nDiff ) * ( coupon ) / Math.pow( yld, t + nDiff ); duration += ( i + nDiff ) * ( coupon ) / Math.pow( yld, i + nDiff );
p += coupon / Math.pow( yld, t + nDiff ); p += coupon / Math.pow( yld, i + nDiff );
} }
duration += ( numOfCoups + nDiff ) * ( coupon + f100 ) / Math.pow( yld, numOfCoups + nDiff ); duration += ( numOfCoups + nDiff ) * ( coupon + f100 ) / Math.pow( yld, numOfCoups + nDiff );
p += ( coupon + f100 ) / Math.pow( yld, numOfCoups + nDiff ); p += ( coupon + f100 ) / Math.pow( yld, numOfCoups + nDiff );
duration /= p; duration = duration / p / frequency;
duration /= frequency;
return duration; return duration;
} }
...@@ -1407,7 +1445,7 @@ cCUMIPMT.prototype.Calculate = function ( arg ) { ...@@ -1407,7 +1445,7 @@ cCUMIPMT.prototype.Calculate = function ( arg ) {
if ( nStartPer < 1 || nEndPer < nStartPer || fRate <= 0 || nEndPer > nNumPeriods || nNumPeriods <= 0 || fVal <= 0 || ( nPayType != 0 && nPayType != 1 ) ) if ( nStartPer < 1 || nEndPer < nStartPer || fRate <= 0 || nEndPer > nNumPeriods || nNumPeriods <= 0 || fVal <= 0 || ( nPayType != 0 && nPayType != 1 ) )
return this.value = new cError( cErrorType.not_numeric ); return this.value = new cError( cErrorType.not_numeric );
fRmz = GetRmz( fRate, nNumPeriods, fVal, 0, nPayType ); fRmz = getPMT( fRate, nNumPeriods, fVal, 0, nPayType );
if ( nStartPer == 1 ) { if ( nStartPer == 1 ) {
if ( nPayType <= 0 ) if ( nPayType <= 0 )
...@@ -1418,9 +1456,9 @@ cCUMIPMT.prototype.Calculate = function ( arg ) { ...@@ -1418,9 +1456,9 @@ cCUMIPMT.prototype.Calculate = function ( arg ) {
for ( var i = nStartPer; i <= nEndPer; i++ ) { for ( var i = nStartPer; i <= nEndPer; i++ ) {
if ( nPayType > 0 ) if ( nPayType > 0 )
fZinsZ += GetZw( fRate, i - 2, fRmz, fVal, 1 ) - fRmz; fZinsZ += getFV( fRate, i - 2, fRmz, fVal, 1 ) - fRmz;
else else
fZinsZ += GetZw( fRate, i - 1, fRmz, fVal, 0 ); fZinsZ += getFV( fRate, i - 1, fRmz, fVal, 0 );
} }
fZinsZ *= fRate; fZinsZ *= fRate;
...@@ -1527,7 +1565,7 @@ cCUMPRINC.prototype.Calculate = function ( arg ) { ...@@ -1527,7 +1565,7 @@ cCUMPRINC.prototype.Calculate = function ( arg ) {
if ( nStartPer < 1 || nEndPer < nStartPer || nEndPer < 1 || fRate <= 0 || nNumPeriods <= 0 || fVal <= 0 || ( nPayType != 0 && nPayType != 1 ) ) if ( nStartPer < 1 || nEndPer < nStartPer || nEndPer < 1 || fRate <= 0 || nNumPeriods <= 0 || fVal <= 0 || ( nPayType != 0 && nPayType != 1 ) )
return this.value = new cError( cErrorType.not_numeric ); return this.value = new cError( cErrorType.not_numeric );
fRmz = GetRmz( fRate, nNumPeriods, fVal, 0.0, nPayType ); fRmz = getPMT( fRate, nNumPeriods, fVal, 0.0, nPayType );
fKapZ = 0.0; fKapZ = 0.0;
...@@ -1545,9 +1583,9 @@ cCUMPRINC.prototype.Calculate = function ( arg ) { ...@@ -1545,9 +1583,9 @@ cCUMPRINC.prototype.Calculate = function ( arg ) {
for ( var i = nStart; i <= nEnd; i++ ) { for ( var i = nStart; i <= nEnd; i++ ) {
if ( nPayType > 0 ) if ( nPayType > 0 )
fKapZ += fRmz - ( GetZw( fRate, i - 2, fRmz, fVal, 1 ) - fRmz ) * fRate; fKapZ += fRmz - ( getFV( fRate, i - 2, fRmz, fVal, 1 ) - fRmz ) * fRate;
else else
fKapZ += fRmz - GetZw( fRate, i - 1, fRmz, fVal, 0 ) * fRate; fKapZ += fRmz - getFV( fRate, i - 1, fRmz, fVal, 0 ) * fRate;
} }
return this.value = new cNumber( fKapZ ); return this.value = new cNumber( fKapZ );
...@@ -1561,14 +1599,209 @@ cCUMPRINC.prototype.getInfo = function () { ...@@ -1561,14 +1599,209 @@ cCUMPRINC.prototype.getInfo = function () {
} }
function cDB() { function cDB() {
cBaseFunction.call( this, "DB" ); // cBaseFunction.call( this, "DB" );
this.name = "DB";
this.type = cElementType.func;
this.value = null;
this.argumentsMin = 4;
this.argumentsCurrent = 0;
this.argumentsMax = 5;
this.formatType = {
def:-1, //подразумевается формат первой ячейки входящей в формулу.
noneFormat:-2
};
this.numFormat = this.formatType.noneFormat;
} }
cDB.prototype = Object.create( cBaseFunction.prototype ) cDB.prototype = Object.create( cBaseFunction.prototype )
cDB.prototype.Calculate = function ( arg ) {
var cast = arg[0],
salvage = arg[1],
life = arg[2],
period = arg[3],
month = arg[4] && !(arg[4] instanceof cEmpty) ? arg[4] : new cNumber( 12 );
if ( cast instanceof cArea || cast instanceof cArea3D ) {
cast = cast.cross( arguments[1].first );
}
else if ( cast instanceof cArray ) {
cast = cast.getElementRowCol( 0, 0 );
}
if ( salvage instanceof cArea || salvage instanceof cArea3D ) {
salvage = salvage.cross( arguments[1].first );
}
else if ( salvage instanceof cArray ) {
salvage = salvage.getElementRowCol( 0, 0 );
}
if ( life instanceof cArea || life instanceof cArea3D ) {
life = life.cross( arguments[1].first );
}
else if ( life instanceof cArray ) {
life = life.getElementRowCol( 0, 0 );
}
if ( period instanceof cArea || period instanceof cArea3D ) {
period = period.cross( arguments[1].first );
}
else if ( period instanceof cArray ) {
period = period.getElementRowCol( 0, 0 );
}
if ( month instanceof cArea || month instanceof cArea3D ) {
month = month.cross( arguments[1].first );
}
else if ( month instanceof cArray ) {
month = month.getElementRowCol( 0, 0 );
}
cast = cast.tocNumber();
salvage = salvage.tocNumber();
life = life.tocNumber();
period = period.tocNumber();
month = month.tocNumber();
if ( cast instanceof cError ) return this.value = cast;
if ( salvage instanceof cError ) return this.value = salvage;
if ( life instanceof cError ) return this.value = life;
if ( period instanceof cError ) return this.value = period;
if ( month instanceof cError ) return this.value = month;
cast = cast.getValue();
salvage = salvage.getValue();
life = life.getValue();
period = period.getValue();
month = month.getValue();
if ( month < 1 || month > 12 || life > 1200 || salvage < 0 ||
period > (life + 1) || salvage > cast || cast < 0 ) {
return this.value = new cError( cErrorType.wrong_value_type );
}
var nAbRate = 1 - Math.pow( salvage / cast, 1 / life );
nAbRate = Math.floor( (nAbRate * 1000) + 0.5 ) / 1000;
var nErsteAbRate = cast * nAbRate * month / 12;
var res = 0;
if ( Math.floor( period ) == 1 )
res = nErsteAbRate;
else {
var nSummAbRate = nErsteAbRate, nMin = life;
if ( nMin > period ) nMin = period;
var iMax = Math.floor( nMin );
for ( var i = 2; i <= iMax; i++ ) {
res = (cast - nSummAbRate) * nAbRate;
nSummAbRate += res;
}
if ( period > life )
res = ((cast - nSummAbRate) * nAbRate * (12 - month)) / 12;
}
this.value = new cNumber( res );
return this.value;
}
cDB.prototype.getInfo = function () {
return {
name:this.name,
args:"( cost , salvage , life , period [ , [ month ] ] )"
};
}
function cDDB() { function cDDB() {
cBaseFunction.call( this, "DDB" ); // cBaseFunction.call( this, "DDB" );
this.name = "DDB";
this.type = cElementType.func;
this.value = null;
this.argumentsMin = 4;
this.argumentsCurrent = 0;
this.argumentsMax = 5;
this.formatType = {
def:-1, //подразумевается формат первой ячейки входящей в формулу.
noneFormat:-2
};
this.numFormat = this.formatType.noneFormat;
} }
cDDB.prototype = Object.create( cBaseFunction.prototype ) cDDB.prototype = Object.create( cBaseFunction.prototype )
cDDB.prototype.Calculate = function ( arg ) {
var cost = arg[0],
salvage = arg[1],
life = arg[2],
period = arg[3],
factor = arg[4] && !(arg[4] instanceof cEmpty) ? arg[4] : new cNumber( 2 );
if ( cost instanceof cArea || cost instanceof cArea3D ) {
cost = cost.cross( arguments[1].first );
}
else if ( cost instanceof cArray ) {
cost = cost.getElementRowCol( 0, 0 );
}
if ( salvage instanceof cArea || salvage instanceof cArea3D ) {
salvage = salvage.cross( arguments[1].first );
}
else if ( salvage instanceof cArray ) {
salvage = salvage.getElementRowCol( 0, 0 );
}
if ( life instanceof cArea || life instanceof cArea3D ) {
life = life.cross( arguments[1].first );
}
else if ( life instanceof cArray ) {
life = life.getElementRowCol( 0, 0 );
}
if ( period instanceof cArea || period instanceof cArea3D ) {
period = period.cross( arguments[1].first );
}
else if ( period instanceof cArray ) {
period = period.getElementRowCol( 0, 0 );
}
if ( factor instanceof cArea || factor instanceof cArea3D ) {
factor = factor.cross( arguments[1].first );
}
else if ( factor instanceof cArray ) {
factor = factor.getElementRowCol( 0, 0 );
}
cost = cost.tocNumber();
salvage = salvage.tocNumber();
life = life.tocNumber();
period = period.tocNumber();
factor = factor.tocNumber();
if ( cost instanceof cError ) return this.value = cost;
if ( salvage instanceof cError ) return this.value = salvage;
if ( life instanceof cError ) return this.value = life;
if ( period instanceof cError ) return this.value = period;
if ( factor instanceof cError ) return this.value = factor;
cost = cost.getValue();
salvage = salvage.getValue();
life = life.getValue();
period = period.getValue();
factor = factor.getValue();
if (cost < 0.0 || salvage < 0.0 || factor <= 0.0 || salvage > cost || period < 1.0 || period > life) {
return this.value = new cError( cErrorType.wrong_value_type );
}
var res = getDDB(cost, salvage, life, period, factor);
this.value = new cNumber( res );
return this.value;
}
cDDB.prototype.getInfo = function () {
return {
name:this.name,
args:"( cost , salvage , life , period [ , factor ] )"
};
}
function cDISC() { function cDISC() {
// cBaseFunction.call( this, "DISC" ); // cBaseFunction.call( this, "DISC" );
...@@ -2201,9 +2434,107 @@ cINTRATE.prototype.getInfo = function () { ...@@ -2201,9 +2434,107 @@ cINTRATE.prototype.getInfo = function () {
} }
function cIPMT() { function cIPMT() {
cBaseFunction.call( this, "IPMT" ); // cBaseFunction.call( this, "IPMT" );
this.name = "IPMT";
this.type = cElementType.func;
this.value = null;
this.argumentsMin = 4;
this.argumentsCurrent = 0;
this.argumentsMax = 6;
this.formatType = {
def:-1, //подразумевается формат первой ячейки входящей в формулу.
noneFormat:-2
};
this.numFormat = this.formatType.noneFormat;
} }
cIPMT.prototype = Object.create( cBaseFunction.prototype ) cIPMT.prototype = Object.create( cBaseFunction.prototype )
cIPMT.prototype.Calculate = function ( arg ) {
var rate = arg[0], per = arg[1], nper = arg[2], pv = arg[3], fv = arg[4] ? arg[4] : new cNumber( 0 ), type = arg[5] ? arg[5] : new cNumber( 0 );
if ( rate instanceof cArea || rate instanceof cArea3D ) {
rate = rate.cross( arguments[1].first );
}
else if ( rate instanceof cArray ) {
rate = rate.getElementRowCol( 0, 0 );
}
if ( per instanceof cArea || per instanceof cArea3D ) {
per = per.cross( arguments[1].first );
}
else if ( per instanceof cArray ) {
per = per.getElementRowCol( 0, 0 );
}
if ( nper instanceof cArea || nper instanceof cArea3D ) {
nper = nper.cross( arguments[1].first );
}
else if ( nper instanceof cArray ) {
nper = nper.getElementRowCol( 0, 0 );
}
if ( pv instanceof cArea || pv instanceof cArea3D ) {
pv = pv.cross( arguments[1].first );
}
else if ( pv instanceof cArray ) {
pv = pv.getElementRowCol( 0, 0 );
}
if ( fv instanceof cArea || fv instanceof cArea3D ) {
fv = fv.cross( arguments[1].first );
}
else if ( fv instanceof cArray ) {
fv = fv.getElementRowCol( 0, 0 );
}
if ( type instanceof cArea || type instanceof cArea3D ) {
type = type.cross( arguments[1].first );
}
else if ( type instanceof cArray ) {
type = type.getElementRowCol( 0, 0 );
}
rate = rate.tocNumber();
per = per.tocNumber();
nper = nper.tocNumber();
pv = pv.tocNumber();
fv = fv.tocNumber();
type = type.tocNumber();
if ( rate instanceof cError ) return this.value = rate;
if ( per instanceof cError ) return this.value = per;
if ( nper instanceof cError ) return this.value = nper;
if ( pv instanceof cError ) return this.value = pv;
if ( fv instanceof cError ) return this.value = fv;
if ( type instanceof cError ) return this.value = type;
rate = rate.getValue();
per = per.getValue();
nper = nper.getValue();
pv = pv.getValue();
fv = fv.getValue();
type = type.getValue();
var res;
if ( per < 1 || per > nper ){
return this.value = new cError( cErrorType.wrong_value_type );
}
res = getPMT(rate, nper, pv, fv, type);
res = getIPMT(rate, per, pv, type, res);
this.value = new cNumber( res );
// this.value.numFormat = 9;
return this.value;
}
cIPMT.prototype.getInfo = function () {
return {
name:this.name,
args:"( rate , per , nper , pv [ , [ fv ] [ , [ type ] ] ] )"
};
}
function cIRR() { function cIRR() {
// cBaseFunction.call( this, "IRR" ); // cBaseFunction.call( this, "IRR" );
...@@ -2484,9 +2815,116 @@ cMDURATION.prototype.getInfo = function () { ...@@ -2484,9 +2815,116 @@ cMDURATION.prototype.getInfo = function () {
} }
function cMIRR() { function cMIRR() {
cBaseFunction.call( this, "MIRR" ); // cBaseFunction.call( this, "MIRR" );
this.name = "MIRR";
this.type = cElementType.func;
this.value = null;
this.argumentsMin = 3;
this.argumentsCurrent = 0;
this.argumentsMax = 3;
this.formatType = {
def:-1, //подразумевается формат первой ячейки входящей в формулу.
noneFormat:-2
};
this.numFormat = this.formatType.noneFormat;
} }
cMIRR.prototype = Object.create( cBaseFunction.prototype ) cMIRR.prototype = Object.create( cBaseFunction.prototype )
cMIRR.prototype.Calculate = function ( arg ) {
var arg0 = arg[0], arg1 = arg[1], arg2 = arg[2];
var valueArray = [];
if ( arg0 instanceof cArea ) {
arg0.foreach2( function ( c ) {
valueArray.push( c.tocNumber() );
} )
}
else if ( arg0 instanceof cArray ) {
arg0.foreach( function ( c ) {
valueArray.push( c.tocNumber() );
} )
}
else if ( arg0 instanceof cArea3D ) {
if ( arg0.wsFrom == arg0.wsTo ) {
valueArray = arg0.getMatrix()[0];
}
else
return this.value = new cError( cErrorType.wrong_value_type );
}
else {
arg0 = arg0.tocNumber();
if ( arg1 instanceof cError ) {
return this.value = new cError( cErrorType.not_numeric )
}
else
valueArray.push( arg0 );
}
if ( arg1 instanceof cArea || arg1 instanceof cArea3D ) {
arg1 = arg1.cross( arguments[1].first );
}
else if ( arg1 instanceof cArray ) {
arg1 = arg1.getElementRowCol( 0, 0 );
}
if ( arg2 instanceof cArea || arg2 instanceof cArea3D ) {
arg2 = arg2.cross( arguments[1].first );
}
else if ( arg2 instanceof cArray ) {
arg2 = arg2.getElementRowCol( 0, 0 );
}
var fRate1_invest = arg1.tocNumber(), fRate1_reinvest = arg2.tocNumber();
if( fRate1_invest instanceof cError ) return this.value = fRate1_invest;
if( fRate1_reinvest instanceof cError ) return this.value = fRate1_reinvest;
fRate1_invest = fRate1_invest.getValue() + 1;
fRate1_reinvest = fRate1_reinvest.getValue() + 1;
var fNPV_reinvest = 0, fPow_reinvest = 1, fNPV_invest = 0, fPow_invest = 1, fCellValue,
wasNegative = false, wasPositive = false;
for(var i = 0; i < valueArray.length; i++){
fCellValue = valueArray[i];
if( fCellValue instanceof cError )
return this.value = fCellValue;
fCellValue = valueArray[i].getValue();
if( fCellValue > 0 ){ // reinvestments
wasPositive = true;
fNPV_reinvest += fCellValue * fPow_reinvest;
}
else if( fCellValue < 0 ){ // investments
wasNegative = true;
fNPV_invest += fCellValue * fPow_invest;
}
fPow_reinvest /= fRate1_reinvest;
fPow_invest /= fRate1_invest;
}
if( !( wasNegative && wasPositive ) )
return this.value = new cError( cErrorType.division_by_zero );
var fResult = -fNPV_reinvest / fNPV_invest;
fResult *= Math.pow( fRate1_reinvest, valueArray.length - 1 );
fResult = Math.pow( fResult, 1 / (valueArray.length - 1) );
this.value = new cNumber( fResult - 1 );
this.value.numFormat = 9;
return this.value;
}
cMIRR.prototype.getInfo = function () {
return {
name:this.name,
args:"( values , finance-rate , reinvest-rate )"
};
}
function cNOMINAL() { function cNOMINAL() {
// cBaseFunction.call( this, "NOMINAL" ); // cBaseFunction.call( this, "NOMINAL" );
...@@ -3108,9 +3546,108 @@ cPMT.prototype.getInfo = function () { ...@@ -3108,9 +3546,108 @@ cPMT.prototype.getInfo = function () {
} }
function cPPMT() { function cPPMT() {
cBaseFunction.call( this, "PPMT" ); // cBaseFunction.call( this, "PPMT" );
this.name = "PPMT";
this.type = cElementType.func;
this.value = null;
this.argumentsMin = 4;
this.argumentsCurrent = 0;
this.argumentsMax = 6;
this.formatType = {
def:-1, //подразумевается формат первой ячейки входящей в формулу.
noneFormat:-2
};
this.numFormat = this.formatType.noneFormat;
} }
cPPMT.prototype = Object.create( cBaseFunction.prototype ) cPPMT.prototype = Object.create( cBaseFunction.prototype )
cPPMT.prototype.Calculate = function ( arg ) {
var rate = arg[0], per = arg[1], nper = arg[2], pv = arg[3], fv = arg[4] ? arg[4] : new cNumber( 0 ), type = arg[5] ? arg[5] : new cNumber( 0 );
if ( rate instanceof cArea || rate instanceof cArea3D ) {
rate = rate.cross( arguments[1].first );
}
else if ( rate instanceof cArray ) {
rate = rate.getElementRowCol( 0, 0 );
}
if ( per instanceof cArea || per instanceof cArea3D ) {
per = per.cross( arguments[1].first );
}
else if ( per instanceof cArray ) {
per = per.getElementRowCol( 0, 0 );
}
if ( nper instanceof cArea || nper instanceof cArea3D ) {
nper = nper.cross( arguments[1].first );
}
else if ( nper instanceof cArray ) {
nper = nper.getElementRowCol( 0, 0 );
}
if ( pv instanceof cArea || pv instanceof cArea3D ) {
pv = pv.cross( arguments[1].first );
}
else if ( pv instanceof cArray ) {
pv = pv.getElementRowCol( 0, 0 );
}
if ( fv instanceof cArea || fv instanceof cArea3D ) {
fv = fv.cross( arguments[1].first );
}
else if ( fv instanceof cArray ) {
fv = fv.getElementRowCol( 0, 0 );
}
if ( type instanceof cArea || type instanceof cArea3D ) {
type = type.cross( arguments[1].first );
}
else if ( type instanceof cArray ) {
type = type.getElementRowCol( 0, 0 );
}
rate = rate.tocNumber();
per = per.tocNumber();
nper = nper.tocNumber();
pv = pv.tocNumber();
fv = fv.tocNumber();
type = type.tocNumber();
if ( rate instanceof cError ) return this.value = rate;
if ( per instanceof cError ) return this.value = per;
if ( nper instanceof cError ) return this.value = nper;
if ( pv instanceof cError ) return this.value = pv;
if ( fv instanceof cError ) return this.value = fv;
if ( type instanceof cError ) return this.value = type;
rate = rate.getValue();
per = per.getValue();
nper = nper.getValue();
pv = pv.getValue();
fv = fv.getValue();
type = type.getValue();
var res;
if ( per < 1 || per > nper ){
return this.value = new cError( cErrorType.wrong_value_type );
}
var fRmz = getPMT(rate, nper, pv, fv, type);
res = fRmz - getIPMT(rate, per, pv, type, fRmz);
this.value = new cNumber( res );
// this.value.numFormat = 9;
return this.value;
}
cPPMT.prototype.getInfo = function () {
return {
name:this.name,
args:"( rate , per , nper , pv [ , [ fv ] [ , [ type ] ] ] )"
};
}
function cPRICE() { function cPRICE() {
// cBaseFunction.call( this, "PRICE" ); // cBaseFunction.call( this, "PRICE" );
...@@ -3719,9 +4256,75 @@ cRECEIVED.prototype.getInfo = function () { ...@@ -3719,9 +4256,75 @@ cRECEIVED.prototype.getInfo = function () {
} }
function cSLN() { function cSLN() {
cBaseFunction.call( this, "SLN" ); // cBaseFunction.call( this, "SLN" );
this.name = "SLN";
this.type = cElementType.func;
this.value = null;
this.argumentsMin = 3;
this.argumentsCurrent = 0;
this.argumentsMax = 3;
this.formatType = {
def:-1, //подразумевается формат первой ячейки входящей в формулу.
noneFormat:-2
};
this.numFormat = this.formatType.noneFormat;
} }
cSLN.prototype = Object.create( cBaseFunction.prototype ) cSLN.prototype = Object.create( cBaseFunction.prototype )
cSLN.prototype.Calculate = function ( arg ) {
var cost = arg[0],
salvage = arg[1],
life = arg[2];
if ( cost instanceof cArea || cost instanceof cArea3D ) {
cost = cost.cross( arguments[1].first );
}
else if ( cost instanceof cArray ) {
cost = cost.getElementRowCol( 0, 0 );
}
if ( salvage instanceof cArea || salvage instanceof cArea3D ) {
salvage = salvage.cross( arguments[1].first );
}
else if ( salvage instanceof cArray ) {
salvage = salvage.getElementRowCol( 0, 0 );
}
if ( life instanceof cArea || life instanceof cArea3D ) {
life = life.cross( arguments[1].first );
}
else if ( life instanceof cArray ) {
life = life.getElementRowCol( 0, 0 );
}
cost = cost.tocNumber();
salvage = salvage.tocNumber();
life = life.tocNumber();
if ( cost instanceof cError ) return this.value = cost;
if ( salvage instanceof cError ) return this.value = salvage;
if ( life instanceof cError ) return this.value = life;
cost = cost.getValue();
salvage = salvage.getValue();
life = life.getValue();
if( life == 0 )
return this.value = new cError( cErrorType.not_numeric );
var res = ( cost - salvage ) / life;
this.value = new cNumber( res );
return this.value;
}
cSLN.prototype.getInfo = function () {
return {
name:this.name,
args:"( cost , salvage , life )"
};
}
function cSYD() { function cSYD() {
// cBaseFunction.call( this, "SYD" ); // cBaseFunction.call( this, "SYD" );
...@@ -4184,7 +4787,7 @@ cXIRR.prototype.Calculate = function ( arg ) { ...@@ -4184,7 +4787,7 @@ cXIRR.prototype.Calculate = function ( arg ) {
return this.value = new cError( cErrorType.not_numeric ) return this.value = new cError( cErrorType.not_numeric )
} }
else else
valueArray[0] = arg0; valueArray.push( arg0 );
} }
if ( arg1 instanceof cArea ) { if ( arg1 instanceof cArea ) {
......
...@@ -4101,9 +4101,7 @@ RangeDataManager.prototype = { ...@@ -4101,9 +4101,7 @@ RangeDataManager.prototype = {
}, },
move : function(from, to) move : function(from, to)
{ {
//убираем fChange, чтобы потом послать его только на одну операцию, а не 2 var fOldChange = this.fChange;
var fOldChange = this.fChange;
this.fChange = null;
var nOffsetRow = to.r1 - from.r1; var nOffsetRow = to.r1 - from.r1;
var nOffsetCol = to.c1 - from.c1; var nOffsetCol = to.c1 - from.c1;
var oGetRes = this.get(from); var oGetRes = this.get(from);
...@@ -4113,10 +4111,13 @@ RangeDataManager.prototype = { ...@@ -4113,10 +4111,13 @@ RangeDataManager.prototype = {
var oNewBBox = new Asc.Range(elem.bbox.c1 + nOffsetCol, elem.bbox.r1 + nOffsetRow, elem.bbox.c2 + nOffsetCol, elem.bbox.r2 + nOffsetRow); var oNewBBox = new Asc.Range(elem.bbox.c1 + nOffsetCol, elem.bbox.r1 + nOffsetRow, elem.bbox.c2 + nOffsetCol, elem.bbox.r2 + nOffsetRow);
if(null != fOldChange) if(null != fOldChange)
fOldChange.call(this, elem.data, elem.bbox, oNewBBox); fOldChange.call(this, elem.data, elem.bbox, oNewBBox);
//убираем fChange, чтобы потом послать его только на одну операцию, а не 2
this.fChange = null;
this.removeElement(elem); this.removeElement(elem);
this.add(oNewBBox, elem.data); this.add(oNewBBox, elem.data);
this.fChange = fOldChange;
} }
this.fChange = fOldChange;
}, },
getAll : function() getAll : function()
{ {
...@@ -4312,9 +4313,7 @@ CellArea.prototype = { ...@@ -4312,9 +4313,7 @@ CellArea.prototype = {
}, },
move : function(from, to) move : function(from, to)
{ {
//убираем fChange, чтобы потом послать его только на одну операцию, а не 2 var fOldChange = this.fChange;
var fOldChange = this.fChange;
this.fChange = null;
var nOffsetRow = to.r1 - from.r1; var nOffsetRow = to.r1 - from.r1;
var nOffsetCol = to.c1 - from.c1; var nOffsetCol = to.c1 - from.c1;
var oGetRes = this.get(from); var oGetRes = this.get(from);
...@@ -4324,10 +4323,12 @@ CellArea.prototype = { ...@@ -4324,10 +4323,12 @@ CellArea.prototype = {
var oNewBBox = new Asc.Range(elem.bbox.c1 + nOffsetCol, elem.bbox.r1 + nOffsetRow, elem.bbox.c2 + nOffsetCol, elem.bbox.r2 + nOffsetRow); var oNewBBox = new Asc.Range(elem.bbox.c1 + nOffsetCol, elem.bbox.r1 + nOffsetRow, elem.bbox.c2 + nOffsetCol, elem.bbox.r2 + nOffsetRow);
if(null != fOldChange) if(null != fOldChange)
fOldChange.call(this, elem.data, elem.bbox, oNewBBox); fOldChange.call(this, elem.data, elem.bbox, oNewBBox);
//убираем fChange, чтобы потом послать его только на одну операцию, а не 2
this.fChange = null;
this.removeElement(elem); this.removeElement(elem);
this.add(oNewBBox.r1, oNewBBox.c1, elem.data); this.add(oNewBBox.r1, oNewBBox.c1, elem.data);
this.fChange = fOldChange;
} }
this.fChange = fOldChange;
}, },
getAll : function(){ getAll : function(){
var aRes = []; var aRes = [];
......
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