120 lines
2.9 KiB
Go
Raw Normal View History

2016-04-10 23:39:38 +02:00
package language
import (
"fmt"
"strconv"
"strings"
)
2020-08-10 00:29:54 +02:00
// Operands is a representation of http://unicode.org/reports/tr35/tr35-numbers.html#Operands
type Operands struct {
2016-04-10 23:39:38 +02:00
N float64 // absolute value of the source number (integer and decimals)
I int64 // integer digits of n
V int64 // number of visible fraction digits in n, with trailing zeros
W int64 // number of visible fraction digits in n, without trailing zeros
F int64 // visible fractional digits in n, with trailing zeros
T int64 // visible fractional digits in n, without trailing zeros
}
2020-08-10 00:29:54 +02:00
// NequalsAny returns true if o represents an integer equal to any of the arguments.
func (o *Operands) NequalsAny(any ...int64) bool {
2016-04-10 23:39:38 +02:00
for _, i := range any {
if o.I == i && o.T == 0 {
return true
}
}
return false
}
2020-08-10 00:29:54 +02:00
// NmodEqualsAny returns true if o represents an integer equal to any of the arguments modulo mod.
func (o *Operands) NmodEqualsAny(mod int64, any ...int64) bool {
2016-04-10 23:39:38 +02:00
modI := o.I % mod
for _, i := range any {
if modI == i && o.T == 0 {
return true
}
}
return false
}
2020-08-10 00:29:54 +02:00
// NinRange returns true if o represents an integer in the closed interval [from, to].
func (o *Operands) NinRange(from, to int64) bool {
2016-04-10 23:39:38 +02:00
return o.T == 0 && from <= o.I && o.I <= to
}
// NmodInRange returns true if o represents an integer in the closed interval [from, to] modulo mod.
2020-08-10 00:29:54 +02:00
func (o *Operands) NmodInRange(mod, from, to int64) bool {
2016-04-10 23:39:38 +02:00
modI := o.I % mod
return o.T == 0 && from <= modI && modI <= to
}
2020-08-10 00:29:54 +02:00
func newOperands(v interface{}) (*Operands, error) {
2016-04-10 23:39:38 +02:00
switch v := v.(type) {
case int:
return newOperandsInt64(int64(v)), nil
case int8:
return newOperandsInt64(int64(v)), nil
case int16:
return newOperandsInt64(int64(v)), nil
case int32:
return newOperandsInt64(int64(v)), nil
case int64:
return newOperandsInt64(v), nil
case string:
return newOperandsString(v)
case float32, float64:
return nil, fmt.Errorf("floats should be formatted into a string")
default:
return nil, fmt.Errorf("invalid type %T; expected integer or string", v)
}
}
2020-08-10 00:29:54 +02:00
func newOperandsInt64(i int64) *Operands {
2016-04-10 23:39:38 +02:00
if i < 0 {
i = -i
}
2020-08-10 00:29:54 +02:00
return &Operands{float64(i), i, 0, 0, 0, 0}
2016-04-10 23:39:38 +02:00
}
2020-08-10 00:29:54 +02:00
func newOperandsString(s string) (*Operands, error) {
2016-04-10 23:39:38 +02:00
if s[0] == '-' {
s = s[1:]
}
n, err := strconv.ParseFloat(s, 64)
if err != nil {
return nil, err
}
2020-08-10 00:29:54 +02:00
ops := &Operands{N: n}
2016-04-10 23:39:38 +02:00
parts := strings.SplitN(s, ".", 2)
ops.I, err = strconv.ParseInt(parts[0], 10, 64)
if err != nil {
return nil, err
}
if len(parts) == 1 {
return ops, nil
}
fraction := parts[1]
ops.V = int64(len(fraction))
for i := ops.V - 1; i >= 0; i-- {
if fraction[i] != '0' {
ops.W = i + 1
break
}
}
if ops.V > 0 {
f, err := strconv.ParseInt(fraction, 10, 0)
if err != nil {
return nil, err
}
ops.F = f
}
if ops.W > 0 {
t, err := strconv.ParseInt(fraction[:ops.W], 10, 0)
if err != nil {
return nil, err
}
ops.T = t
}
return ops, nil
}