declare let RedNaoBasicManipulatorInstance:any;

class RedNaoFormula {
    public FormElement: sfFormElementBase<any>;
    public Formula: any;
    private _remote:SmartFormsRemote;

    constructor(formElement: sfFormElementBase<any>, formula) {
        this.FormElement = formElement;
        this.Formula = formula;
    }

    public GetRemote(){
        if(this._remote==null)
            this._remote=new SmartFormsRemote();
        return this._remote;
    }



    public FieldUsedInFormula(fieldName) {
        for (let i = 0; i < this.Formula.FieldsUsed.length; i++) {
            if (fieldName == this.Formula.FieldsUsed[i])
                return true;
        }

        return false;
    };

    public UpdateFieldWithValue(value) {
        try {
            let calculatedValue = this.GetValueFromFormula(value);
            if(calculatedValue instanceof Promise) {
                this.FormElement.ShowLoadingBox();
                calculatedValue.then((result) =>{this.FormElement.HideLoadingBox(); this.ProcessResult(result)});
            }
            else {

                this.ProcessResult(calculatedValue);
            }

        } catch (exception) {

        }
    };

    public GetValueFromFormula(values) {
        return RedNaoEventManager.Publish('CalculateFormula', {FormulaInstance:this,Formula: this.Formula, Values: values});
    }

    private ProcessResult(calculatedValue: any) {
        if (typeof calculatedValue == 'number' && isNaN(calculatedValue))
            calculatedValue = 0;
        let previousValue = RedNaoBasicManipulatorInstance.GetValue(this.FormElement.Options, this.Formula.PropertyName, this.Formula.additionalInformation);
        RedNaoBasicManipulatorInstance.SetValue(this.FormElement.Options, this.Formula.PropertyName, calculatedValue, this.Formula.additionalInformation);
        this.FormElement.RefreshElement(this.Formula.PropertyName, previousValue);
        this.FormElement.FirePropertyChanged();
    }
}

function RNFRound(value,decimals)
{
    return value.toFixed(decimals);
}

declare let smartFormsUserName;
function RNUserName()
{
    return smartFormsUserName;
}

declare let smartFormsFirstName;
function RNFirstName()
{
    return smartFormsFirstName;
}

declare let smartFormsLastName;
function RNLastName()
{
    return smartFormsLastName;
}

declare let smartFormsEmail;
function RNEmail()
{
    return smartFormsEmail;
}


function RNIf(condition,trueValue,falseValue)
{
    if(condition)
        return trueValue;

    return falseValue;
}

function RNDateDiff(date1,date2)
{
    let value1=date1.numericalValue;
    let value2=date2.numericalValue;
    if(value1==0||value2==0)
        return 0;

    date1=new Date(value1);
    date2=new Date(value2);

    let result=date2-date1;

    if(result==0)
        return 0;

    return Math.round(result/1000/60/60/24);
}

class SmartFormsRemote
{
    public static cache:RemoteCache[]=[];
    public lastPromise:any;
    public Post(url:string, args:any):Promise<any>
    {
        return this.Execute('post',url,args);
    }

    public Get(url:string, args:any):Promise<any>
    {
        return this.Execute('get',url,args);
    }

    private Execute(name:string,url:string, args:any):Promise<any>
    {
        if(args instanceof (window as any).SmartFormBasicDataStore)
        {
            args=Object.assign({},args);
        }
        let createdPromise= new Promise((resolve)=>{
            let id:string=this.GenerateRequestId(url,args);
            let cachedResult=SmartFormsRemote.cache.find(x=>x.Id==id);
            if(cachedResult!=null)
                resolve(cachedResult.Result);
            else
                rnJQuery[name](url,args)
                    .done((result)=>{
                        if(this.lastPromise!=createdPromise)
                            return;
                        try{
                            result=JSON.parse(result);
                        }catch (exception)
                        {

                        }
                        SmartFormsRemote.cache.push({Id:id,Result:result});
                        resolve(result);
                    })
                    .fail((e)=>{
                        console.log('An error ocurred processing the json request',e);
                    });
        });
        return this.lastPromise=createdPromise;
    }


    private GenerateRequestId(url: string, args: any) {
        let parsedArgs='';
        if(args!=null)
        {
            try{
                parsedArgs=JSON.stringify(args);
            }catch(ex)
            {
                parsedArgs=args.toString();
            }
        }

        return url+'@@'+parsedArgs;
    }
}

interface RemoteCache{
    Id:string;
    Result:string;
}

function RNPMT(rate,nper,pv,fv,type){if(!fv)fv=0;if(!type)type=0;if(rate==0)return-(pv+fv)/nper;var pvif=Math.pow(1+rate,nper);var pmt=rate/(pvif-1)*-(pv*pvif+fv);if(type==1){pmt/=(1+rate)};return pmt};function RNIPMT(pv,pmt,rate,per){var tmp=Math.pow(1+rate,per);return 0-(pv*tmp*rate+pmt*(tmp-1))};function RNPPMT(rate,per,nper,pv,fv,type){if(per<1||(per>=nper+1))return null;var pmt=RNPMT(rate,nper,pv,fv,type);var ipmt=RNIPMT(pv,pmt,rate,per-1);return pmt-ipmt}
function RNXNPV(rate,values,dates){var result=0;for(var i=0;i<values.length;i++){result+=values[i]/Math.pow(1+rate,RNDateDiff(dates[i],dates[0])/365)}
    return result};function RNXIRR(values,guess){if(!guess)guess=0.1;var x1=0.0;var x2=guess;var f1=RNXNPV(x1,values,null);var f2=RNXNPV(x2,values,null);for(var i=0;i<100;i++){if((f1*f2)<0.0)break;if(Math.abs(f1)<Math.abs(f2)){f1=RNXNPV(x1+=1.6*(x1-x2),values,null)}
else{f2=RNXNPV(x2+=1.6*(x2-x1),values,null)}};if((f1*f2)>0.0)return null;var f=RNXNPV(x1,values,null);if(f<0.0){var rtb:any=x1;var dx=x2-x1}
else{var rtb:any=x2;var dx=x1-x2};for(var i=0;i<100;i++){dx*=0.5;var x_mid=rtb+dx;var f_mid=RNXNPV(x_mid,values,null);if(f_mid<=0.0)rtb=x_mid;if((Math.abs(f_mid)<1.0e-6)||(Math.abs(dx)<1.0e-6))return x_mid};return null}

function RNFV(rate, nper, pmt, pv, type) {
    var pow = Math.pow(1 + rate, nper),
        fv;
    if (rate) {
        fv = (pmt*(1+rate*type)*(1-pow)/rate)-pv*pow;
    } else {
        fv = -1 * (pv + pmt * nper);
    }
    return fv.toFixed(2);
}