| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267 | import { StoreObject } from '../store/storeObject';import * as Commands from './../../ast/commands';import { Types } from './../../typeSystem/types';import { toReal } from "./../../typeSystem/parsers";import { Decimal } from 'decimal.js';import { MultiType } from '../../typeSystem/multiType';import { CompoundType } from '../../typeSystem/compoundType';import { Modes } from '../modes';import { Config } from '../../util/config';/** * sin * cos * tan * sqrt * pow * log * abs * negate * invert * max * min */function convertToRadians (degrees) {  return degrees.times(Decimal.acos(-1)).div(180);}export function createSinFun () {   const sinFun = (sto, _) => {     const x = sto.applyStore('x');     const angle = x.value.mod(360);     let result = null;     if(angle.eq(90)) {       result = new Decimal(1);     } else if (angle.eq(180)) {      result = new Decimal(0);     } else if (angle.eq(270)) {       result = new Decimal(-1);     } else {       result = Decimal.sin(convertToRadians(angle));     }     if(result.dp() > Config.decimalPlaces) {      result = new Decimal(result.toFixed(Config.decimalPlaces));    }     const temp = new StoreObject(Types.REAL, result);     sto.mode = Modes.RETURN;     return Promise.resolve(sto.updateStore('$', temp));   };  const block = new Commands.CommandBlock([],  [new Commands.SysCall(sinFun)]);  const func = new Commands.Function('$sin', Types.REAL,    [new Commands.FormalParameter(new MultiType([Types.INTEGER, Types.REAL]), 'x', false)],    block);  return func;}export function createCosFun () {  const cosFun = (sto, _) => {    const x = sto.applyStore('x');    const angle = x.value.mod(360);    let result = null;    if(angle.eq(90)) {      result = new Decimal(0);    } else if (angle.eq(180)) {      result = new Decimal(-1);    } else if (angle.eq(270)) {      result = new Decimal(0)    }    result = Decimal.cos(convertToRadians(angle));    if(result.dp() > Config.decimalPlaces) {      result = new Decimal(result.toFixed(Config.decimalPlaces));    }    const temp = new StoreObject(Types.REAL, result);    sto.mode = Modes.RETURN;    return Promise.resolve(sto.updateStore('$', temp));  }; const block = new Commands.CommandBlock([],  [new Commands.SysCall(cosFun)]); const func = new Commands.Function('$cos', Types.REAL,   [new Commands.FormalParameter(new MultiType([Types.INTEGER, Types.REAL]), 'x', false)],   block); return func;}export function createTanFun () {  const tanFun = (sto, _) => {    const x = sto.applyStore('x');    const angle = x.value.mod(360);    if(angle.eq(90) || angle.eq(270)) {      return Promise.reject("Tangent of "+x.value.toNumber()+"° is undefined.");    }    let result = Decimal.tan(convertToRadians(angle));    if(result.dp() > Config.decimalPlaces) {      result = new Decimal(result.toFixed(Config.decimalPlaces));    }    const temp = new StoreObject(Types.REAL, result);    sto.mode = Modes.RETURN;    return Promise.resolve(sto.updateStore('$', temp));  }; const block = new Commands.CommandBlock([],  [new Commands.SysCall(tanFun)]); const func = new Commands.Function('$tan', Types.REAL,   [new Commands.FormalParameter(new MultiType([Types.INTEGER, Types.REAL]), 'x', false)],   block); return func;}export function createSqrtFun () {  const sqrtFun = (sto, _) => {    const x = sto.applyStore('x');    let result = x.value.sqrt();    if(result.dp() > Config.decimalPlaces) {      result = new Decimal(result.toFixed(Config.decimalPlaces));    }    const temp = new StoreObject(Types.REAL, result);    sto.mode = Modes.RETURN;    return Promise.resolve(sto.updateStore('$', temp));  }; const block = new Commands.CommandBlock([],  [new Commands.SysCall(sqrtFun)]); const func = new Commands.Function('$sqrt', Types.REAL,   [new Commands.FormalParameter(new MultiType([Types.INTEGER, Types.REAL]), 'x', false)],   block); return func;}export function createPowFun () {  const powFun = (sto, _) => {    const x = sto.applyStore('x');    const y = sto.applyStore('y');    let result = x.value.pow(y.value);    if(result.dp() > Config.decimalPlaces) {      result = new Decimal(result.toFixed(Config.decimalPlaces));    }    const temp = new StoreObject(Types.REAL, result);    sto.mode = Modes.RETURN;    return Promise.resolve(sto.updateStore('$', temp));  }; const block = new Commands.CommandBlock([],  [new Commands.SysCall(powFun)]); const func = new Commands.Function('$pow', Types.REAL,   [new Commands.FormalParameter(new MultiType([Types.INTEGER, Types.REAL]), 'x', false),    new Commands.FormalParameter(new MultiType([Types.INTEGER, Types.REAL]), 'y', false)],   block); return func;}export function createLogFun () {  const logFun = (sto, _) => {    const x = sto.applyStore('x');    if (x.value.isNegative()) {      return Promise.reject(new Error("the value passed to log function cannot be negative"));    }    let result = Decimal.log10(x.value);    if(result.dp() > Config.decimalPlaces) {      result = new Decimal(result.toFixed(Config.decimalPlaces));    }    const temp = new StoreObject(Types.REAL, result);    sto.mode = Modes.RETURN;    return Promise.resolve(sto.updateStore('$', temp));  }; const block = new Commands.CommandBlock([],  [new Commands.SysCall(logFun)]); const func = new Commands.Function('$log', Types.REAL,   [new Commands.FormalParameter(new MultiType([Types.INTEGER, Types.REAL]), 'x', false)],   block); return func;}export function createAbsFun () {  const absFun = (sto, _) => {    const x = sto.applyStore('x');    const result = x.value.abs();    const temp = new StoreObject(x.type, result);    sto.updateStore('$', temp)    sto.mode = Modes.RETURN;    return Promise.resolve(sto);  }; const block = new Commands.CommandBlock([],  [new Commands.SysCall(absFun)]); const func = new Commands.Function('$abs', new MultiType([Types.INTEGER, Types.REAL]),   [new Commands.FormalParameter(new MultiType([Types.INTEGER, Types.REAL]), 'x', false)],   block); return func;}export function createNegateFun () {  const negateFun = (sto, _) => {    const x = sto.applyStore('x');    const result = x.value.negated();    const temp = new StoreObject(x.type, result);    sto.mode = Modes.RETURN;    return Promise.resolve(sto.updateStore('$', temp));  }; const block = new Commands.CommandBlock([],  [new Commands.SysCall(negateFun)]); const func = new Commands.Function('$negate', new MultiType([Types.INTEGER, Types.REAL]),   [new Commands.FormalParameter(new MultiType([Types.INTEGER, Types.REAL]), 'x', false)],   block); return func;}export function createInvertFun () {  const invertFun = (sto, _) => {    const x = sto.applyStore('x');    let result = toReal(1).dividedBy(x.value);    if(result.dp() > Config.decimalPlaces) {      result = new Decimal(result.toFixed(Config.decimalPlaces));    }    const temp = new StoreObject(Types.REAL, result);    sto.mode = Modes.RETURN;    return Promise.resolve(sto.updateStore('$', temp));  }; const block = new Commands.CommandBlock([],  [new Commands.SysCall(invertFun)]); const func = new Commands.Function('$invert', Types.REAL,   [new Commands.FormalParameter(new MultiType([Types.INTEGER, Types.REAL]), 'x', false)],   block); return func;}export function createMaxFun () {  const maxFun = (sto, _) => {    const x = sto.applyStore('x');    const numbers = x.value.map(stoObj => stoObj.value);    const result = Decimal.max(...numbers);    const temp = new StoreObject(x.type.innerType, result);    sto.mode = Modes.RETURN;    return Promise.resolve(sto.updateStore('$', temp));  }; const paramType = new CompoundType(new MultiType([Types.INTEGER, Types.REAL]), 1); const block = new Commands.CommandBlock([],  [new Commands.SysCall(maxFun)]); const func = new Commands.Function('$max', new MultiType([Types.INTEGER, Types.REAL]),   [new Commands.FormalParameter(paramType, 'x', false)],   block); return func;}export function createMinFun () {  const minFun = (sto, _) => {    const x = sto.applyStore('x');    const numbers = x.value.map(stoObj => stoObj.value);    const result = Decimal.min(...numbers);    const temp = new StoreObject(x.type.innerType, result);    sto.mode = Modes.RETURN;    return Promise.resolve(sto.updateStore('$', temp));  }; const paramType = new CompoundType(new MultiType([Types.INTEGER, Types.REAL]), 1); const block = new Commands.CommandBlock([],  [new Commands.SysCall(minFun)]); const func = new Commands.Function('$min', new MultiType([Types.INTEGER, Types.REAL]),   [new Commands.FormalParameter(paramType, 'x', false)],   block); return func;}export function createRandFun () {  const randFun = (sto, _) => {    const val = Math.random();    const result = new StoreObject(Types.REAL, new Decimal(val));    sto.mode = Modes.RETURN;    return Promise.resolve(sto.updateStore("$", result));  };  const block = new Commands.CommandBlock([],  [new Commands.SysCall(randFun)]);  const func = new Commands.Function('$rand', Types.REAL, [], block);  return func;}
 |