Logo Search packages:      
Sourcecode: edenmath.app version File versions  Download package

EMResponder.m

//
//  EMResponder.m
//  EdenMath
//
//  Created by admin on Thu Feb 21 2002.
//  Copyright (c) 2002-2004 Edenwaith. All rights reserved.
//

#import "EMResponder.h"

#include <math.h>
#include <stdlib.h>

#ifdef GNUSTEP
#include <time.h>
#endif

@implementation EMResponder


// -------------------------------------------------------
// (id) init
// -------------------------------------------------------
- (id) init
{
    current_value = 0.0;
    previous_value      = 0.0;
    op_type       = NO_OP;
    angle_type          = DEGREE;
    e_value       = M_E;            // 2.7182818285
    trailing_digits     = 0;
    isNewDigit          = YES;
    
    return self;
}

// -------------------------------------------------------
// (double) getCurrentValue
// Simple return function which returns the current
// value
// -------------------------------------------------------
- (double) getCurrentValue
{
    return current_value;
}


// -------------------------------------------------------
// (int) getTrailingDigits
// Simple return function which returns the number of
// trailing digits on the currently displayed number
// so the proper amount of precision can be displayed.
// Limit the number of trailing digits precision so odd
// errors don't occur.
// -------------------------------------------------------
- (int) getTrailingDigits
{
    if (trailing_digits == 0)
    {
        trailing_digits = 0;
    }
    else if (trailing_digits > 10)
    {
        trailing_digits = 10;
    }
    
    return trailing_digits;
}

// -------------------------------------------------------
// (void) setCurrentValue:(double)num
// -------------------------------------------------------
- (void) setCurrentValue:(double)num
{
    current_value = num;
}


// -------------------------------------------------------
// (void) setState:(NSDictionary *)stateDictionary
// -------------------------------------------------------
- (void)setState:(NSDictionary *)stateDictionary 
{
    current_value       = [[stateDictionary objectForKey: @"current_value"] doubleValue];
    previous_value      = [[stateDictionary objectForKey: @"previous_value"] doubleValue];
    op_type       = [[stateDictionary objectForKey: @"op_type"] intValue];
    trailing_digits     = [[stateDictionary objectForKey: @"trailing_digits"] intValue];
    isNewDigit          = [[stateDictionary objectForKey: @"isNewDigit"] boolValue];
}

// -------------------------------------------------------
// (NSDictionary *) state
// -------------------------------------------------------
- (NSDictionary *)state 
{
    NSDictionary *stateDictionary = [NSDictionary dictionaryWithObjectsAndKeys:
        [NSNumber numberWithDouble: current_value], @"current_value",
        [NSNumber numberWithDouble: previous_value], @"previous_value",
        [NSNumber numberWithInt: op_type], @"op_type",
        [NSNumber numberWithInt: trailing_digits], @"trailing_digits",
        [NSNumber numberWithBool: isNewDigit], @"isNewDigit",
        nil]; 
    return stateDictionary;
}

// -------------------------------------------------------
// (void)newDigit:(int)digit
// -------------------------------------------------------
- (void)newDigit:(int)digit 
{

    if (isNewDigit) 
    {
        previous_value = current_value;
        isNewDigit  = NO;
        current_value = digit;
    } 
    else 
    {
        BOOL negative = NO;
        if (current_value < 0) 
        {
            current_value = - current_value;
            negative = YES;
        }
        
        if (trailing_digits == 0) 
        {
            current_value = current_value * 10 + digit;
        } 
        else 
        {
            current_value = current_value + (digit/pow(10.0, trailing_digits));

            trailing_digits++;
        }
        
        if (negative == YES)
        {
            current_value = - current_value;
        }
    }
}

// -------------------------------------------------------
// (void) period
// -------------------------------------------------------
- (void)period 
{
    if (isNewDigit) 
    {
        current_value = 0.0;
        isNewDigit = NO;
    }
    if (trailing_digits == 0) 
    {
        trailing_digits = 1;
    }
}

// -------------------------------------------------------
// (void) pi
// Display the constant pi (3.141592653589793)
// -------------------------------------------------------
- (void) pi 
{
    current_value       = M_PI;
    trailing_digits     = 0;
    isNewDigit          = YES;
}

// -------------------------------------------------------
// (void) trig_constant
// Display the constant pi (3.141592653589793)
// -------------------------------------------------------
- (void) trig_constant: (double) trig_const 
{
    current_value       = trig_const;
    trailing_digits     = 0;
    isNewDigit          = YES;
}

// -------------------------------------------------------
// (void) e
// Display the constant e (2.718281828459045)
// -------------------------------------------------------
- (void)e
{
    current_value       = M_E; 
    trailing_digits     = 0;
    isNewDigit          = YES;
}

// -------------------------------------------------------
// (void) clear
// clear the displayField and reset several variables
// -------------------------------------------------------
- (void) clear 
{
    current_value       = 0.0;
    previous_value      = 0.0;
    op_type             = NO_OP;
    trailing_digits     = 0;
    isNewDigit    = YES;
}

// -------------------------------------------------------
// (void)operation:(OpType)new_op_type
// -------------------------------------------------------
- (void)operation:(OpType)new_op_type
{
    if (op_type == NO_OP) 
    {
        previous_value  = current_value;
        isNewDigit      = YES;
        op_type   = new_op_type;
        trailing_digits = 0;
    } 
    else 
    {
        // cascading operations
        [self enter]; // calculate previous value, first
        previous_value = current_value;
        isNewDigit = YES;
        op_type = new_op_type;
        trailing_digits = 0;        
    }
}

// -------------------------------------------------------
// (void)enter
// For binary operators (+, -, x, /, etc.), calculate
// the value and place into current_value
// -------------------------------------------------------
- (void)enter 
{
    switch (op_type) 
    {
        case NO_OP:
            break;
        case ADD_OP:
            current_value = previous_value + current_value;
            break;
        case SUBTRACT_OP:
            current_value = previous_value - current_value;
            break;
        case MULTIPLY_OP:
            current_value = previous_value * current_value;
            break;
        case DIVIDE_OP:
            if (current_value != 0.0)
            {
                current_value = previous_value / current_value;
            }
            break;
        case EXPONENT_OP: // x^y
            current_value = pow(previous_value, current_value);
            break;
        case XROOT_OP: // yx
            current_value = pow(previous_value, 1/current_value);
            break;
        case MOD_OP:
            current_value = (int)previous_value % (int)current_value;
            break;
        case EE_OP:
            current_value = previous_value * pow(10, current_value);
            break;
        case NPR_OP: // n!/(n-r)!
            current_value = [self factorial:previous_value] / [self factorial:(previous_value - current_value)];
            break;
        case NCR_OP: // n!/(r! * (n-r)!)
            current_value = [self factorial:previous_value] / ([self factorial:current_value] * [self factorial:(previous_value - current_value)]);
            break;
    }
    previous_value      = 0.0;
    op_type             = NO_OP;
    trailing_digits     = 0;
    isNewDigit          = YES;
}

// =====================================================================================
// ALGEBRAIC FUNCTIONS
// =====================================================================================

// -------------------------------------------------------
// (void) reverse_sign
// -------------------------------------------------------
- (void) reverse_sign
{
    current_value = - current_value;
}

// -------------------------------------------------------
// (void)percentage
// -------------------------------------------------------
- (void)percentage
{
    current_value = current_value * 0.01;
    isNewDigit      = YES;
    trailing_digits = 0;
}

// -------------------------------------------------------
// (void) squared
// -------------------------------------------------------
- (void) squared
{
    current_value = pow(current_value, 2);
    isNewDigit    = YES;
}

// -------------------------------------------------------
// (void) cubed
// -------------------------------------------------------
- (void) cubed
{
    current_value = pow(current_value, 3);
    isNewDigit    = true;
}

// -------------------------------------------------------
// (void) square_root
// -------------------------------------------------------
- (void) square_root
{
    if (current_value >= 0.0)
    {
        current_value = sqrt(current_value);
    }
    
    isNewDigit = YES;
}

// -------------------------------------------------------
// (void) cubed_root
// -------------------------------------------------------
- (void)cubed_root
{
    //if (current_value >= 0.0)
    //{
        current_value = pow(current_value, 0.3333333333333333);
    //}
    
    isNewDigit = YES;
}

// -------------------------------------------------------
// (void) ln
// natural log
// -------------------------------------------------------
- (void)ln
{
    if (current_value > 0.0)
    {
        current_value = log(current_value);
    }
    
    isNewDigit = YES;
}

// -------------------------------------------------------
// (void) cubed_root
// logarithm base 10
// -------------------------------------------------------
- (void)logarithm
{
    if (current_value > 0.0)
    {
        current_value = log10(current_value);
    }
    
    isNewDigit = YES;
}

// -------------------------------------------------------
// (void) factorial
// This new function replaces the old factorial function
// with a gamma function.  For more information, read the
// lgamma man page or 
// http://www.gnu.org/software/libc/manual/html_node/Special-Functions.html
// Version: 8. March 2004 23:40
// -------------------------------------------------------
- (void) factorial
{
    double lg;
    
    // 170.5 seems to be the max value which EdenMath can handle
    if (current_value < 170.5)
    {
        lg = lgamma(current_value+1.0);
        current_value = signgam*exp(lg); /* signgam is a predefined variable */
    }
    
    trailing_digits = 0; // this allows for more precise decimal precision
    isNewDigit = YES;

}

// -------------------------------------------------------
// (double) factorial: (double) n
// This is required for the permutations and combinations
// calls.
// Version: 9. March 2004 23:48
// -------------------------------------------------------
- (double) factorial: (double) n
{
    double lg;
    
    // 170.5 seems to be the max value which EdenMath can handle
    if (n < 170.5)
    {
        lg = lgamma(n+1.0);
        n = signgam*exp(lg);
    }
    
    return (n);
    
}

// -------------------------------------------------------
// (void) powerE
// e^x
// M_E is a constant hidden somewhere in the included
// libraries
// -------------------------------------------------------
- (void) powerE
{
    current_value = pow(M_E, current_value);
    isNewDigit    = YES;
}

// -------------------------------------------------------
// (void) 10^x
// -------------------------------------------------------
- (void) power10
{
    current_value = pow(10, current_value);
    isNewDigit      = YES;
}

// -------------------------------------------------------
// (void) inverse
// -------------------------------------------------------
- (void)inverse
{
    if (current_value != 0.0)
    {
        current_value = 1/current_value;
    }
    
    isNewDigit = YES;
}

// =====================================================================================
// TRIGOMETRIC FUNCTIONS
// =====================================================================================

// -------------------------------------------------------
// (void) setAngleType:(AngleType)aType
// Modify the type of angles using degrees, radians, or
// gradients.  EM 1.1.1 reduced this code down to one
// line, eliminating multiple IF-ELSE statements.
// -------------------------------------------------------
// Version: 31. May 2003
// -------------------------------------------------------
- (void)setAngleType:(AngleType)aType
{
    angle_type = aType;
}

// -------------------------------------------------------
// (double)deg_to_rad:(double)degrees
// Convert from degrees to radians
// -------------------------------------------------------
- (double)deg_to_rad:(double)degrees
{
    double radians = 0.0;
    radians = degrees * M_PI / 180;
    
    return radians;
}

// -------------------------------------------------------
// (double)rad_to_deg:(double)radians
// Convert from radians to degrees
// -------------------------------------------------------
// Created: 31. May 2003
// Version: 31. May 2003
// -------------------------------------------------------
- (double)rad_to_deg:(double)radians
{
    double degrees = 0.0;
    degrees = radians * 180 / M_PI;
    
    return degrees;
}

// -------------------------------------------------------
// (double)grad_to_rad:(double)gradients
// Convert from gradients to radians
// http://www.onlineconversion.com/angles.htm
// 1 gradient = 0.015707963267948966192 radians
// 1 gradient = 0.9 degrees
// Created higher precision for the conversion so 
// Tan(50g) would equal 1.  Otherwise, if the conversion
// isn't precise enough, it comes out to be 1.000003672.
// -------------------------------------------------------
// Version: 31. May 2003
// -------------------------------------------------------
- (double)grad_to_rad:(double)gradients
{
    double radians = 0.0;
    radians = gradients * 0.015707963267948966192;
    
    return radians;
}

// -------------------------------------------------------
// (double)rad_to_grad:(double)radians
// Convert from radians to gradients
// http://www.onlineconversion.com/angles.htm
// 1 gradient = 0.015707963267948966192 radians
// 1 gradient = 0.9 degrees
// -------------------------------------------------------
// Created: 31. May 2003
// Version: 31. May 2003
// -------------------------------------------------------
- (double)rad_to_grad:(double)radians
{
    double gradients = 0.0;
    gradients = radians / 0.015707963267948966192;
    
    return gradients;
}

// -------------------------------------------------------
// (void) sine
// -------------------------------------------------------
- (void)sine
{
    if (angle_type == DEGREE)
    {
        current_value = [self deg_to_rad:current_value];
    }
    else if (angle_type == GRADIENT)
    {
        current_value = [self grad_to_rad:current_value];
    }
    
    current_value = sin(current_value);
    isNewDigit      = YES;
}

// -------------------------------------------------------
// (void) cosine
// -------------------------------------------------------
- (void)cosine
{
    if (angle_type == DEGREE)
    {
        current_value = [self deg_to_rad:current_value];
    }
    else if (angle_type == GRADIENT)
    {
        current_value = [self grad_to_rad:current_value];
    }
    
    current_value = cos(current_value);
    isNewDigit      = YES;
}

// -------------------------------------------------------
// (void) tangent
// -------------------------------------------------------
- (void)tangent
{
    if (angle_type == DEGREE)
    {
        current_value = [self deg_to_rad:current_value];
    }
    else if (angle_type == GRADIENT)
    {
        current_value = [self grad_to_rad:current_value];
    }
    
    if ( ( current_value == M_PI/2) || (current_value == 3 * M_PI / 2) || 
         ( current_value == - M_PI/2) || (current_value == -3 * M_PI / 2) )
    {
        NSBeep();
        NSRunAlertPanel(@"Warning", @"Tan cannot calculate values of /2 or 3/2",
                        @"OK", nil, nil);
    }
    else // otherwise, tan will still be calculated on /2 or 3/2, which is wrong.
    {
        current_value = tan(current_value);
    }
    
    isNewDigit    = YES;
}

// -------------------------------------------------------
// (void) arcsine
// -------------------------------------------------------
// Version: 31. May 2003
// -------------------------------------------------------
- (void)arcsine
{
    current_value = asin(current_value);
    isNewDigit     = YES;
    
    if (angle_type == DEGREE)
    {
        current_value = [self rad_to_deg:current_value];
    }
    else if (angle_type == GRADIENT)
    {
        current_value = [self rad_to_grad:current_value];
    }
}

// -------------------------------------------------------
// (void) arccosine
// -------------------------------------------------------
// Version: 31. May 2003
// -------------------------------------------------------
- (void)arccosine
{ 
    current_value = acos(current_value);
    isNewDigit      = YES;
    
    if (angle_type == DEGREE)
    {
        current_value = [self rad_to_deg:current_value];
    }
    else if (angle_type == GRADIENT)
    {
        current_value = [self rad_to_grad:current_value];
    }    
}

// -------------------------------------------------------
// (void) arctangent
// -------------------------------------------------------
// Version: 31. May 2003
// -------------------------------------------------------
- (void)arctangent
{    
    current_value = atan(current_value);
    isNewDigit      = YES;
    
    if (angle_type == DEGREE)
    {
        current_value = [self rad_to_deg:current_value];
    }
    else if (angle_type == GRADIENT)
    {
        current_value = [self rad_to_grad:current_value];
    }    
}

// =====================================================================================
// PROBABILITY FUNCTIONS
// The Probability and Combination functions are in the enter function since they
// act as binary operators
// =====================================================================================

// -------------------------------------------------------
// (void) generate_random_num
// -------------------------------------------------------
- (double) generate_random_num
{
    static int seeded = 0;
    double value = 0.0;
    
    if (seeded == 0)
    {
        srand((unsigned)time((time_t *)NULL));
        seeded = 1;
    }
    
    value = rand();
    
    return value;
}

// -------------------------------------------------------
// (void) random_num
// -------------------------------------------------------
- (void) random_num
{
    current_value = [self generate_random_num];
    isNewDigit     = YES;
}

@end

Generated by  Doxygen 1.6.0   Back to index