/*****************************************************************************
 * Project: RooFit                                                           *
 * Package: RooFitCore                                                       *
 * @(#)root/roofitcore:$Id$
 * Authors:                                                                  *
 *   WV, Wouter Verkerke, UC Santa Barbara, verkerke@slac.stanford.edu       *
 *   DK, David Kirkby,    UC Irvine,         dkirkby@uci.edu                 *
 *                                                                           *
 * Copyright (c) 2000-2005, Regents of the University of California          *
 *                          and Stanford University. All rights reserved.    *
 *                                                                           *
 * Redistribution and use in source and binary forms,                        *
 * with or without modification, are permitted according to the terms        *
 * listed in LICENSE (http://roofit.sourceforge.net/license.txt)             *
 *****************************************************************************/

//////////////////////////////////////////////////////////////////////////////
//
// BEGIN_HTML
// RooSecondMoment represents the first, second, or third order derivative
// of any RooAbsReal as calculated (numerically) by the MathCore Richardson
// derivator class.
// END_HTML
//


#include "RooFit.h"

#include "Riostream.h"
#include "Riostream.h"
#include <math.h>
#include <string>

#include "RooSecondMoment.h"
#include "RooAbsReal.h"
#include "RooAbsPdf.h"
#include "RooErrorHandler.h"
#include "RooArgSet.h"
#include "RooMsgService.h"
#include "RooRealVar.h"
#include "RooFunctor.h"
#include "RooGlobalFunc.h"
#include "RooConstVar.h"
#include "RooRealIntegral.h"
#include "RooNumIntConfig.h"
#include "RooFormulaVar.h"
#include "RooLinearVar.h"
#include "RooProduct.h"
#include "RooGlobalFunc.h"
#include <string>
using namespace std ;


ClassImp(RooSecondMoment)
;


//_____________________________________________________________________________
RooSecondMoment::RooSecondMoment() 
{
  // Default constructor
}



//_____________________________________________________________________________
RooSecondMoment::RooSecondMoment(const char* name, const char* title, RooAbsReal& func, RooRealVar& x, Bool_t centr, Bool_t takeRoot) :
  RooAbsMoment(name, title,func,x,2,takeRoot),
  _xf("!xf","xf",this,kFALSE,kFALSE),
  _ixf("!ixf","ixf",this),
  _if("!if","if",this),
  _xfOffset(0)
{

  setExpensiveObjectCache(func.expensiveObjectCache()) ;
  
  RooAbsReal* XF(0) ;
  if (centr) {

    string m1name=Form("%s_moment1",GetName()) ;
    RooAbsReal* mom1 = func.mean(x) ;
    _mean.setArg(*mom1) ;
    
    string pname=Form("%s_product",name) ;
    _xfOffset = mom1->getVal() ;
    XF = new RooFormulaVar(pname.c_str(),Form("pow((@0-%f),2)*@1",_xfOffset),RooArgList(x,func)) ;  
        
  } else {

    string pname=Form("%s_product",name) ;
    XF = new RooProduct(pname.c_str(),pname.c_str(),RooArgList(x,x,func)) ;
  }
    
  XF->setExpensiveObjectCache(func.expensiveObjectCache()) ;

  if (func.isBinnedDistribution(x)) {
    XF->specialIntegratorConfig(kTRUE)->method1D().setLabel("RooBinIntegrator");
  }

  RooRealIntegral* intXF = (RooRealIntegral*) XF->createIntegral(x) ;
  RooRealIntegral* intF =  (RooRealIntegral*) func.createIntegral(x) ;
  intXF->setCacheNumeric(kTRUE) ;
  intF->setCacheNumeric(kTRUE) ;

  _xf.setArg(*XF) ;
  _ixf.setArg(*intXF) ;
  _if.setArg(*intF) ;
  addOwnedComponents(RooArgSet(*XF,*intXF,*intF)) ;
}

//_____________________________________________________________________________
RooSecondMoment::RooSecondMoment(const char* name, const char* title, RooAbsReal& func, RooRealVar& x, const RooArgSet& nset, 
		     Bool_t centr, Bool_t takeRoot, Bool_t intNSet) :
  RooAbsMoment(name, title,func,x,2,takeRoot),
  _xf("!xf","xf",this,kFALSE,kFALSE),
  _ixf("!ixf","ixf",this),
  _if("!if","if",this),
  _xfOffset(0)
{

  setExpensiveObjectCache(func.expensiveObjectCache()) ;

  _nset.add(nset) ;

  RooAbsReal* XF(0) ;
  if (centr) {

    string m1name=Form("%s_moment1",GetName()) ;
    RooAbsReal* mom1 = func.mean(x,nset) ;
    _mean.setArg(*mom1) ;
    
    string pname=Form("%s_product",name) ;
    _xfOffset = mom1->getVal() ;
    XF = new RooFormulaVar(pname.c_str(),Form("pow((@0-%f),2)*@1",_xfOffset),RooArgList(x,func)) ;  


  } else {

    string pname=Form("%s_product",name) ;
    XF = new RooProduct(pname.c_str(),pname.c_str(),RooArgList(x,x,func)) ;

  }
  
  XF->setExpensiveObjectCache(func.expensiveObjectCache()) ;

  if (func.isBinnedDistribution(x)) {
    XF->specialIntegratorConfig(kTRUE)->method1D().setLabel("RooBinIntegrator");
  }
  if (intNSet && _nset.getSize()>0 && func.isBinnedDistribution(_nset)) {
      XF->specialIntegratorConfig(kTRUE)->method2D().setLabel("RooBinIntegrator");
      XF->specialIntegratorConfig(kTRUE)->methodND().setLabel("RooBinIntegrator");
  }

  RooArgSet intSet(x) ;
  if (intNSet) intSet.add(_nset,kTRUE) ;
  RooRealIntegral* intXF = (RooRealIntegral*) XF->createIntegral(intSet,&_nset) ;
  RooRealIntegral* intF =  (RooRealIntegral*) func.createIntegral(intSet,&_nset) ;
  intXF->setCacheNumeric(kTRUE) ;
  intF->setCacheNumeric(kTRUE) ;

  _xf.setArg(*XF) ;
  _ixf.setArg(*intXF) ;
  _if.setArg(*intF) ;
  addOwnedComponents(RooArgSet(*XF,*intXF,*intF)) ;
}



//_____________________________________________________________________________
RooSecondMoment::RooSecondMoment(const RooSecondMoment& other, const char* name) :
  RooAbsMoment(other, name), 
  _xf("xf",this,other._xf),
  _ixf("ixf",this,other._ixf),
  _if("if",this,other._if),
  _xfOffset(other._xfOffset)
{
}



//_____________________________________________________________________________
RooSecondMoment::~RooSecondMoment() 
{
  // Destructor
}



//_____________________________________________________________________________
Double_t RooSecondMoment::evaluate() const 
{
  // Calculate value  
  Double_t ratio = _ixf / _if ;

  if (_mean.absArg()) {
    ratio -= (_mean - _xfOffset)*(_mean-_xfOffset) ;
  }  

  Double_t ret =  _takeRoot ? sqrt(ratio) : ratio ;
  return ret ;
}


 RooSecondMoment.cxx:1
 RooSecondMoment.cxx:2
 RooSecondMoment.cxx:3
 RooSecondMoment.cxx:4
 RooSecondMoment.cxx:5
 RooSecondMoment.cxx:6
 RooSecondMoment.cxx:7
 RooSecondMoment.cxx:8
 RooSecondMoment.cxx:9
 RooSecondMoment.cxx:10
 RooSecondMoment.cxx:11
 RooSecondMoment.cxx:12
 RooSecondMoment.cxx:13
 RooSecondMoment.cxx:14
 RooSecondMoment.cxx:15
 RooSecondMoment.cxx:16
 RooSecondMoment.cxx:17
 RooSecondMoment.cxx:18
 RooSecondMoment.cxx:19
 RooSecondMoment.cxx:20
 RooSecondMoment.cxx:21
 RooSecondMoment.cxx:22
 RooSecondMoment.cxx:23
 RooSecondMoment.cxx:24
 RooSecondMoment.cxx:25
 RooSecondMoment.cxx:26
 RooSecondMoment.cxx:27
 RooSecondMoment.cxx:28
 RooSecondMoment.cxx:29
 RooSecondMoment.cxx:30
 RooSecondMoment.cxx:31
 RooSecondMoment.cxx:32
 RooSecondMoment.cxx:33
 RooSecondMoment.cxx:34
 RooSecondMoment.cxx:35
 RooSecondMoment.cxx:36
 RooSecondMoment.cxx:37
 RooSecondMoment.cxx:38
 RooSecondMoment.cxx:39
 RooSecondMoment.cxx:40
 RooSecondMoment.cxx:41
 RooSecondMoment.cxx:42
 RooSecondMoment.cxx:43
 RooSecondMoment.cxx:44
 RooSecondMoment.cxx:45
 RooSecondMoment.cxx:46
 RooSecondMoment.cxx:47
 RooSecondMoment.cxx:48
 RooSecondMoment.cxx:49
 RooSecondMoment.cxx:50
 RooSecondMoment.cxx:51
 RooSecondMoment.cxx:52
 RooSecondMoment.cxx:53
 RooSecondMoment.cxx:54
 RooSecondMoment.cxx:55
 RooSecondMoment.cxx:56
 RooSecondMoment.cxx:57
 RooSecondMoment.cxx:58
 RooSecondMoment.cxx:59
 RooSecondMoment.cxx:60
 RooSecondMoment.cxx:61
 RooSecondMoment.cxx:62
 RooSecondMoment.cxx:63
 RooSecondMoment.cxx:64
 RooSecondMoment.cxx:65
 RooSecondMoment.cxx:66
 RooSecondMoment.cxx:67
 RooSecondMoment.cxx:68
 RooSecondMoment.cxx:69
 RooSecondMoment.cxx:70
 RooSecondMoment.cxx:71
 RooSecondMoment.cxx:72
 RooSecondMoment.cxx:73
 RooSecondMoment.cxx:74
 RooSecondMoment.cxx:75
 RooSecondMoment.cxx:76
 RooSecondMoment.cxx:77
 RooSecondMoment.cxx:78
 RooSecondMoment.cxx:79
 RooSecondMoment.cxx:80
 RooSecondMoment.cxx:81
 RooSecondMoment.cxx:82
 RooSecondMoment.cxx:83
 RooSecondMoment.cxx:84
 RooSecondMoment.cxx:85
 RooSecondMoment.cxx:86
 RooSecondMoment.cxx:87
 RooSecondMoment.cxx:88
 RooSecondMoment.cxx:89
 RooSecondMoment.cxx:90
 RooSecondMoment.cxx:91
 RooSecondMoment.cxx:92
 RooSecondMoment.cxx:93
 RooSecondMoment.cxx:94
 RooSecondMoment.cxx:95
 RooSecondMoment.cxx:96
 RooSecondMoment.cxx:97
 RooSecondMoment.cxx:98
 RooSecondMoment.cxx:99
 RooSecondMoment.cxx:100
 RooSecondMoment.cxx:101
 RooSecondMoment.cxx:102
 RooSecondMoment.cxx:103
 RooSecondMoment.cxx:104
 RooSecondMoment.cxx:105
 RooSecondMoment.cxx:106
 RooSecondMoment.cxx:107
 RooSecondMoment.cxx:108
 RooSecondMoment.cxx:109
 RooSecondMoment.cxx:110
 RooSecondMoment.cxx:111
 RooSecondMoment.cxx:112
 RooSecondMoment.cxx:113
 RooSecondMoment.cxx:114
 RooSecondMoment.cxx:115
 RooSecondMoment.cxx:116
 RooSecondMoment.cxx:117
 RooSecondMoment.cxx:118
 RooSecondMoment.cxx:119
 RooSecondMoment.cxx:120
 RooSecondMoment.cxx:121
 RooSecondMoment.cxx:122
 RooSecondMoment.cxx:123
 RooSecondMoment.cxx:124
 RooSecondMoment.cxx:125
 RooSecondMoment.cxx:126
 RooSecondMoment.cxx:127
 RooSecondMoment.cxx:128
 RooSecondMoment.cxx:129
 RooSecondMoment.cxx:130
 RooSecondMoment.cxx:131
 RooSecondMoment.cxx:132
 RooSecondMoment.cxx:133
 RooSecondMoment.cxx:134
 RooSecondMoment.cxx:135
 RooSecondMoment.cxx:136
 RooSecondMoment.cxx:137
 RooSecondMoment.cxx:138
 RooSecondMoment.cxx:139
 RooSecondMoment.cxx:140
 RooSecondMoment.cxx:141
 RooSecondMoment.cxx:142
 RooSecondMoment.cxx:143
 RooSecondMoment.cxx:144
 RooSecondMoment.cxx:145
 RooSecondMoment.cxx:146
 RooSecondMoment.cxx:147
 RooSecondMoment.cxx:148
 RooSecondMoment.cxx:149
 RooSecondMoment.cxx:150
 RooSecondMoment.cxx:151
 RooSecondMoment.cxx:152
 RooSecondMoment.cxx:153
 RooSecondMoment.cxx:154
 RooSecondMoment.cxx:155
 RooSecondMoment.cxx:156
 RooSecondMoment.cxx:157
 RooSecondMoment.cxx:158
 RooSecondMoment.cxx:159
 RooSecondMoment.cxx:160
 RooSecondMoment.cxx:161
 RooSecondMoment.cxx:162
 RooSecondMoment.cxx:163
 RooSecondMoment.cxx:164
 RooSecondMoment.cxx:165
 RooSecondMoment.cxx:166
 RooSecondMoment.cxx:167
 RooSecondMoment.cxx:168
 RooSecondMoment.cxx:169
 RooSecondMoment.cxx:170
 RooSecondMoment.cxx:171
 RooSecondMoment.cxx:172
 RooSecondMoment.cxx:173
 RooSecondMoment.cxx:174
 RooSecondMoment.cxx:175
 RooSecondMoment.cxx:176
 RooSecondMoment.cxx:177
 RooSecondMoment.cxx:178
 RooSecondMoment.cxx:179
 RooSecondMoment.cxx:180
 RooSecondMoment.cxx:181
 RooSecondMoment.cxx:182
 RooSecondMoment.cxx:183
 RooSecondMoment.cxx:184
 RooSecondMoment.cxx:185
 RooSecondMoment.cxx:186
 RooSecondMoment.cxx:187
 RooSecondMoment.cxx:188
 RooSecondMoment.cxx:189
 RooSecondMoment.cxx:190
 RooSecondMoment.cxx:191
 RooSecondMoment.cxx:192
 RooSecondMoment.cxx:193
 RooSecondMoment.cxx:194
 RooSecondMoment.cxx:195
 RooSecondMoment.cxx:196
 RooSecondMoment.cxx:197
 RooSecondMoment.cxx:198
 RooSecondMoment.cxx:199
 RooSecondMoment.cxx:200
 RooSecondMoment.cxx:201
 RooSecondMoment.cxx:202
 RooSecondMoment.cxx:203