123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110 |
- using System;
- namespace NTERA.Interpreter
- {
- public enum ValueType
- {
- Real,
- String
- }
- public struct Value
- {
- public static readonly Value Zero = new Value(0);
- public ValueType Type { get; set; }
- public double Real { get; set; }
- public string String { get; set; }
- public Value(double real) : this()
- {
- Type = ValueType.Real;
- Real = real;
- }
- public Value(string str)
- : this()
- {
- Type = ValueType.String;
- String = str;
- }
- public Value Convert(ValueType type)
- {
- if (Type != type)
- {
- switch (type)
- {
- case ValueType.Real:
- Real = double.Parse(String);
- Type = ValueType.Real;
- break;
- case ValueType.String:
- String = Real.ToString();
- Type = ValueType.String;
- break;
- }
- }
- return this;
- }
- public Value BinOp(Value b, Token tok)
- {
- Value a = this;
- if (a.Type != b.Type)
- {
- if (a.Type > b.Type)
- b = b.Convert(a.Type);
- else
- a = a.Convert(b.Type);
- }
- if (tok == Token.Plus)
- {
- if (a.Type == ValueType.Real)
- return new Value(a.Real + b.Real);
- else
- return new Value(a.String + b.String);
- }
- else if(tok == Token.Equal)
- {
- if (a.Type == ValueType.Real)
- return new Value(a.Real == b.Real ? 1 : 0);
- else
- return new Value(a.String == b.String ? 1 : 0);
- }
- else if(tok == Token.NotEqual)
- {
- if (a.Type == ValueType.Real)
- return new Value(a.Real == b.Real ? 0 : 1);
- else
- return new Value(a.String == b.String ? 0 : 1);
- }
- else
- {
- if (a.Type == ValueType.String)
- throw new Exception("Cannot do binop on strings(except +).");
- switch (tok)
- {
- case Token.Minus: return new Value(a.Real - b.Real);
- case Token.Asterisk: return new Value(a.Real * b.Real);
- case Token.Slash: return new Value(a.Real / b.Real);
- case Token.Caret: return new Value(Math.Pow(a.Real, b.Real));
- case Token.Less: return new Value(a.Real < b.Real ? 1 : 0);
- case Token.More: return new Value(a.Real > b.Real ? 1 : 0);
- case Token.LessEqual: return new Value(a.Real <= b.Real ? 1 : 0);
- case Token.MoreEqual: return new Value(a.Real >= b.Real ? 1 : 0);
- }
- }
- throw new Exception("Unkown binop");
- }
- public override string ToString()
- {
- if (Type == ValueType.Real)
- return Real.ToString();
- return String;
- }
- }
- }
|