xgo/objects/iterator.go

217 lines
4.7 KiB
Go
Raw Normal View History

2019-01-09 10:17:42 +03:00
package objects
2019-01-11 13:27:28 +03:00
import "github.com/d5/tengo/compiler/token"
2019-01-09 10:17:42 +03:00
// Iterator represents an iterator for underlying data type.
2019-01-09 10:17:42 +03:00
type Iterator interface {
Object
// Next returns true if there are more elements to iterate.
2019-01-09 10:17:42 +03:00
Next() bool
// Key returns the key or index value of the current element.
2019-01-09 10:17:42 +03:00
Key() Object
// Value returns the value of the current element.
2019-01-09 10:17:42 +03:00
Value() Object
}
// ArrayIterator is an iterator for an array.
2019-01-09 10:17:42 +03:00
type ArrayIterator struct {
v []Object
i int
l int
}
// NewArrayIterator creates an ArrayIterator.
2019-01-09 10:17:42 +03:00
func NewArrayIterator(v *Array) Iterator {
return &ArrayIterator{
v: v.Value,
l: len(v.Value),
}
}
// TypeName returns the name of the type.
2019-01-09 10:17:42 +03:00
func (i *ArrayIterator) TypeName() string {
return "array-iterator"
}
func (i *ArrayIterator) String() string {
return "<array-iterator>"
}
// BinaryOp returns another object that is the result of
// a given binary operator and a right-hand side object.
2019-01-09 10:17:42 +03:00
func (i *ArrayIterator) BinaryOp(op token.Token, rhs Object) (Object, error) {
return nil, ErrInvalidOperator
}
// IsFalsy returns true if the value of the type is falsy.
2019-01-09 10:17:42 +03:00
func (i *ArrayIterator) IsFalsy() bool {
return true
}
// Equals returns true if the value of the type
// is equal to the value of another object.
2019-01-09 10:17:42 +03:00
func (i *ArrayIterator) Equals(Object) bool {
return false
}
// Copy returns a copy of the type.
2019-01-09 10:17:42 +03:00
func (i *ArrayIterator) Copy() Object {
return &ArrayIterator{v: i.v, i: i.i, l: i.l}
}
// Next returns true if there are more elements to iterate.
2019-01-09 10:17:42 +03:00
func (i *ArrayIterator) Next() bool {
i.i++
return i.i <= i.l
}
// Key returns the key or index value of the current element.
2019-01-09 10:17:42 +03:00
func (i *ArrayIterator) Key() Object {
return &Int{int64(i.i - 1)}
}
// Value returns the value of the current element.
2019-01-09 10:17:42 +03:00
func (i *ArrayIterator) Value() Object {
return i.v[i.i-1]
}
// MapIterator represents an iterator for the map.
2019-01-09 10:17:42 +03:00
type MapIterator struct {
v map[string]Object
k []string
i int
l int
}
// NewMapIterator creates a map iterator.
2019-01-09 10:17:42 +03:00
func NewMapIterator(v *Map) Iterator {
var keys []string
for k := range v.Value {
keys = append(keys, k)
}
return &MapIterator{
v: v.Value,
k: keys,
l: len(keys),
}
}
// TypeName returns the name of the type.
2019-01-09 10:17:42 +03:00
func (i *MapIterator) TypeName() string {
return "map-iterator"
}
func (i *MapIterator) String() string {
return "<map-iterator>"
}
// BinaryOp returns another object that is the result of
// a given binary operator and a right-hand side object.
2019-01-09 10:17:42 +03:00
func (i *MapIterator) BinaryOp(op token.Token, rhs Object) (Object, error) {
return nil, ErrInvalidOperator
}
// IsFalsy returns true if the value of the type is falsy.
2019-01-09 10:17:42 +03:00
func (i *MapIterator) IsFalsy() bool {
return true
}
// Equals returns true if the value of the type
// is equal to the value of another object.
2019-01-09 10:17:42 +03:00
func (i *MapIterator) Equals(Object) bool {
return false
}
// Copy returns a copy of the type.
2019-01-09 10:17:42 +03:00
func (i *MapIterator) Copy() Object {
return &MapIterator{v: i.v, k: i.k, i: i.i, l: i.l}
}
// Next returns true if there are more elements to iterate.
2019-01-09 10:17:42 +03:00
func (i *MapIterator) Next() bool {
i.i++
return i.i <= i.l
}
// Key returns the key or index value of the current element.
2019-01-09 10:17:42 +03:00
func (i *MapIterator) Key() Object {
k := i.k[i.i-1]
return &String{Value: k}
}
// Value returns the value of the current element.
2019-01-09 10:17:42 +03:00
func (i *MapIterator) Value() Object {
k := i.k[i.i-1]
return i.v[k]
}
// StringIterator represents an iterator for a string.
2019-01-09 10:17:42 +03:00
type StringIterator struct {
v []rune
i int
l int
}
// NewStringIterator creates a string iterator.
2019-01-09 10:17:42 +03:00
func NewStringIterator(v *String) Iterator {
r := []rune(v.Value)
return &StringIterator{
v: r,
l: len(r),
}
}
// TypeName returns the name of the type.
2019-01-09 10:17:42 +03:00
func (i *StringIterator) TypeName() string {
return "string-iterator"
}
func (i *StringIterator) String() string {
return "<string-iterator>"
}
// BinaryOp returns another object that is the result of
// a given binary operator and a right-hand side object.
2019-01-09 10:17:42 +03:00
func (i *StringIterator) BinaryOp(op token.Token, rhs Object) (Object, error) {
return nil, ErrInvalidOperator
}
// IsFalsy returns true if the value of the type is falsy.
2019-01-09 10:17:42 +03:00
func (i *StringIterator) IsFalsy() bool {
return true
}
// Equals returns true if the value of the type
// is equal to the value of another object.
2019-01-09 10:17:42 +03:00
func (i *StringIterator) Equals(Object) bool {
return false
}
// Copy returns a copy of the type.
2019-01-09 10:17:42 +03:00
func (i *StringIterator) Copy() Object {
return &StringIterator{v: i.v, i: i.i, l: i.l}
}
// Next returns true if there are more elements to iterate.
2019-01-09 10:17:42 +03:00
func (i *StringIterator) Next() bool {
i.i++
return i.i <= i.l
}
// Key returns the key or index value of the current element.
2019-01-09 10:17:42 +03:00
func (i *StringIterator) Key() Object {
return &Int{int64(i.i - 1)}
}
// Value returns the value of the current element.
2019-01-09 10:17:42 +03:00
func (i *StringIterator) Value() Object {
return &Char{Value: i.v[i.i-1]}
}