123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158 |
- using System;
- using NTERA.Engine.Compiler;
- namespace NTERA.Engine
- {
- public enum ValueType
- {
- Real,
- String
- }
- public struct Value
- {
- public ValueType Type { get; }
- public double Real { get; }
- public string String { get; }
- public Value(bool boolean) : this(boolean ? 1d : 0d) { }
- public Value(double real) : this()
- {
- Type = ValueType.Real;
- Real = real;
- String = real.ToString("0.##");
- }
- public Value(string str) : this()
- {
- Type = ValueType.String;
- String = str;
- Real = double.NaN;
- }
- public Value Operate(Value b, Token tok)
- {
- Value a = this;
- bool isStringOperation = a.Type == ValueType.String || b.Type == ValueType.String;
- if (isStringOperation)
- {
- switch (tok)
- {
- case Token.Plus: return new Value(a.String + b.String);
- case Token.Equal: return new Value(a.String == b.String);
- case Token.NotEqual: return new Value(a.String != b.String);
- }
- }
- else
- {
- switch (tok)
- {
- case Token.Plus: return new Value(a.Real + b.Real);
- case Token.Equal: return new Value(a.Real == b.Real);
- case Token.NotEqual: return new Value(a.Real != b.Real);
- 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.ShiftLeft: return new Value((int)a.Real << (int)b.Real);
- case Token.ShiftRight: return new Value((int)a.Real >> (int)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($"Invalid operation on value ({tok}) on {(isStringOperation ? "string" : "double")}");
- }
- public static Value operator +(Value value1, Value value2)
- {
- return value1.Operate(value2, Token.Plus);
- }
- public static Value operator ==(Value value1, Value value2)
- {
- return value1.Operate(value2, Token.Equal);
- }
- public static Value operator !=(Value value1, Value value2)
- {
- return value1.Operate(value2, Token.NotEqual);
- }
- public static Value operator <(Value value1, Value value2)
- {
- return value1.Operate(value2, Token.Less);
- }
- public static Value operator >(Value value1, Value value2)
- {
- return value1.Operate(value2, Token.More);
- }
- public static Value operator <=(Value value1, Value value2)
- {
- return value1.Operate(value2, Token.LessEqual);
- }
- public static Value operator >=(Value value1, Value value2)
- {
- return value1.Operate(value2, Token.MoreEqual);
- }
- public static implicit operator double(Value value)
- {
- return value.Real;
- }
- public static implicit operator string(Value value)
- {
- return value.String;
- }
- public static implicit operator bool(Value value)
- {
- return value.Real != 0;
- }
- public static implicit operator Value(double value)
- {
- return new Value(value);
- }
- public static implicit operator Value(string value)
- {
- return new Value(value);
- }
- public static implicit operator Value(bool value)
- {
- return new Value(value);
- }
- public override string ToString()
- {
- return String;
- }
- public override bool Equals(object obj)
- {
- if (ReferenceEquals(null, obj))
- return false;
- return obj is Value other && this == other;
- }
- public override int GetHashCode()
- {
- return Type == ValueType.String
- ? String.GetHashCode()
- : Real.GetHashCode();
- }
- }
- }
|