| // Copyright 2021 The Tint Authors. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| // Package list provides utilities for handling lists of dynamically-typed elements |
| package list |
| |
| import ( |
| "fmt" |
| "reflect" |
| ) |
| |
| // List is an interface to a list of dynamically-typed elements |
| type List interface { |
| // Count returns the number if items in the list |
| Count() int |
| |
| // Get returns the element at the index i |
| Get(i int) interface{} |
| |
| // Set assigns the element at the index i with v |
| Set(i int, v interface{}) |
| |
| // Append adds a single item, list, or slice of items to this List |
| Append(v interface{}) |
| |
| // Copy copies the elements at [dst..dst+count) to [src..src+count) |
| Copy(dst, src, count int) |
| |
| // CopyFrom copies the elements [src..src+count) from the list l to the |
| // elements [dst..dst+count) in this list |
| CopyFrom(l List, dst, src, count int) |
| |
| // Reduces the size of the list to count elements |
| Resize(count int) |
| |
| // ElementType returns the type of the elements of the list |
| ElementType() reflect.Type |
| } |
| |
| // Wrap returns a List that wraps a slice pointer |
| func Wrap(s interface{}) List { |
| ptr := reflect.ValueOf(s) |
| if ptr.Kind() != reflect.Ptr || ptr.Elem().Kind() != reflect.Slice { |
| panic(fmt.Errorf("Wrap() must be called with a pointer to slice. Got: %T", s)) |
| } |
| return list{ptr.Elem()} |
| } |
| |
| // New returns a new list of element type elem for n items |
| func New(elem reflect.Type, count int) List { |
| slice := reflect.SliceOf(elem) |
| return list{reflect.MakeSlice(slice, count, count)} |
| } |
| |
| // Copy makes a shallow copy of the list |
| func Copy(l List) List { |
| out := New(l.ElementType(), l.Count()) |
| out.CopyFrom(l, 0, 0, l.Count()) |
| return out |
| } |
| |
| type list struct{ v reflect.Value } |
| |
| func (l list) Count() int { |
| return l.v.Len() |
| } |
| |
| func (l list) Get(i int) interface{} { |
| return l.v.Index(i).Interface() |
| } |
| |
| func (l list) Set(i int, v interface{}) { |
| l.v.Index(i).Set(reflect.ValueOf(v)) |
| } |
| |
| func (l list) Append(v interface{}) { |
| switch v := v.(type) { |
| case list: |
| l.v.Set(reflect.AppendSlice(l.v, reflect.Value(v.v))) |
| case List: |
| // v implements `List`, but isn't a `list`. Need to do a piece-wise copy |
| items := make([]reflect.Value, v.Count()) |
| for i := range items { |
| items[i] = reflect.ValueOf(v.Get(i)) |
| } |
| l.v.Set(reflect.Append(l.v, items...)) |
| default: |
| r := reflect.ValueOf(v) |
| if r.Type() == l.v.Type() { |
| l.v.Set(reflect.AppendSlice(l.v, r)) |
| return |
| } |
| l.v.Set(reflect.Append(l.v, reflect.ValueOf(v))) |
| } |
| } |
| |
| func (l list) Copy(dst, src, count int) { |
| reflect.Copy( |
| l.v.Slice(dst, dst+count), |
| l.v.Slice(src, src+count), |
| ) |
| } |
| |
| func (l list) CopyFrom(o List, dst, src, count int) { |
| if o, ok := o.(list); ok { |
| reflect.Copy( |
| l.v.Slice(dst, dst+count), |
| o.v.Slice(src, src+count), |
| ) |
| } |
| // v implements `List`, but isn't a `list`. Need to do a piece-wise copy |
| items := make([]reflect.Value, count) |
| for i := range items { |
| l.Set(dst+i, o.Get(src+i)) |
| } |
| } |
| |
| func (l list) Resize(count int) { |
| new := reflect.MakeSlice(l.v.Type(), count, count) |
| reflect.Copy(new, l.v) |
| l.v.Set(new) |
| } |
| |
| func (l list) ElementType() reflect.Type { |
| return l.v.Type().Elem() |
| } |