1
0
mirror of https://github.com/danog/liquid.git synced 2025-01-23 05:11:30 +01:00
liquid/generics/sort.go

77 lines
1.5 KiB
Go
Raw Normal View History

2017-06-27 10:28:39 -04:00
package generics
import (
"reflect"
"sort"
)
type genericSortable []interface{}
// Sort any []interface{} value.
func Sort(data []interface{}) {
sort.Sort(genericSortable(data))
}
// Len is part of sort.Interface.
func (s genericSortable) Len() int {
return len(s)
}
// Swap is part of sort.Interface.
func (s genericSortable) Swap(i, j int) {
s[i], s[j] = s[j], s[i]
}
// Less is part of sort.Interface.
func (s genericSortable) Less(i, j int) bool {
2017-06-28 09:36:39 -04:00
return Less(s[i], s[j])
2017-06-27 10:28:39 -04:00
}
// SortByProperty sorts maps on their key indices.
func SortByProperty(data []interface{}, key string) {
sort.Sort(sortableByProperty{data, key})
}
type sortableByProperty struct {
data []interface{}
key string
}
// Len is part of sort.Interface.
func (s sortableByProperty) Len() int {
return len(s.data)
}
// Swap is part of sort.Interface.
func (s sortableByProperty) Swap(i, j int) {
data := s.data
data[i], data[j] = data[j], data[i]
}
// Less is part of sort.Interface.
func (s sortableByProperty) Less(i, j int) bool {
// index returns the value at the s.key, if in is a map that contains this key
2017-06-27 21:30:47 -04:00
index := func(i int) interface{} {
value := s.data[i]
rt := reflect.ValueOf(value)
2017-06-27 10:28:39 -04:00
if rt.Kind() == reflect.Map && rt.Type().Key().Kind() == reflect.String {
2017-06-27 21:30:47 -04:00
elem := rt.MapIndex(reflect.ValueOf(s.key))
if elem.IsValid() {
return elem.Interface()
}
2017-06-27 10:28:39 -04:00
}
return nil
}
2017-06-27 21:30:47 -04:00
a, b := index(i), index(j)
2017-06-28 09:36:39 -04:00
nilFirst := true
switch {
case a == nil && b == nil:
return false
case a == nil:
return nilFirst
case b == nil:
return !nilFirst
}
2017-06-27 21:30:47 -04:00
return Less(a, b)
2017-06-27 10:28:39 -04:00
}