2021-03-30 14:56:59 +02:00
|
|
|
package filetree
|
|
|
|
|
2021-03-31 13:08:55 +02:00
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
)
|
2021-03-30 14:56:59 +02:00
|
|
|
|
|
|
|
type INode interface {
|
|
|
|
IsLeaf() bool
|
|
|
|
GetPath() string
|
|
|
|
GetChildren() []INode
|
|
|
|
SetChildren([]INode)
|
|
|
|
GetCompressionLevel() int
|
|
|
|
SetCompressionLevel(int)
|
|
|
|
}
|
|
|
|
|
|
|
|
func sortNode(node INode) {
|
|
|
|
sortChildren(node)
|
|
|
|
|
|
|
|
for _, child := range node.GetChildren() {
|
|
|
|
sortNode(child)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func sortChildren(node INode) {
|
|
|
|
if node.IsLeaf() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
children := node.GetChildren()
|
|
|
|
sortedChildren := make([]INode, len(children))
|
|
|
|
copy(sortedChildren, children)
|
|
|
|
|
|
|
|
sort.Slice(sortedChildren, func(i, j int) bool {
|
|
|
|
if !sortedChildren[i].IsLeaf() && sortedChildren[j].IsLeaf() {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if sortedChildren[i].IsLeaf() && !sortedChildren[j].IsLeaf() {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return sortedChildren[i].GetPath() < sortedChildren[j].GetPath()
|
|
|
|
})
|
|
|
|
|
|
|
|
// TODO: think about making this in-place
|
|
|
|
node.SetChildren(sortedChildren)
|
|
|
|
}
|
|
|
|
|
|
|
|
func forEachLeaf(node INode, cb func(INode) error) error {
|
|
|
|
if node.IsLeaf() {
|
|
|
|
if err := cb(node); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, child := range node.GetChildren() {
|
|
|
|
if err := forEachLeaf(child, cb); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func any(node INode, test func(INode) bool) bool {
|
|
|
|
if test(node) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, child := range node.GetChildren() {
|
|
|
|
if any(child, test) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2021-03-31 13:39:55 +02:00
|
|
|
func every(node INode, test func(INode) bool) bool {
|
|
|
|
if !test(node) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, child := range node.GetChildren() {
|
|
|
|
if !every(child, test) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2021-03-30 14:56:59 +02:00
|
|
|
func flatten(node INode, collapsedPaths map[string]bool) []INode {
|
|
|
|
result := []INode{}
|
|
|
|
result = append(result, node)
|
|
|
|
|
|
|
|
if !collapsedPaths[node.GetPath()] {
|
|
|
|
for _, child := range node.GetChildren() {
|
|
|
|
result = append(result, flatten(child, collapsedPaths)...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func getNodeAtIndex(node INode, index int, collapsedPaths map[string]bool) INode {
|
|
|
|
foundNode, _ := getNodeAtIndexAux(node, index, collapsedPaths)
|
|
|
|
|
|
|
|
return foundNode
|
|
|
|
}
|
|
|
|
|
|
|
|
func getNodeAtIndexAux(node INode, index int, collapsedPaths map[string]bool) (INode, int) {
|
|
|
|
offset := 1
|
|
|
|
|
|
|
|
if index == 0 {
|
|
|
|
return node, offset
|
|
|
|
}
|
|
|
|
|
|
|
|
if !collapsedPaths[node.GetPath()] {
|
|
|
|
for _, child := range node.GetChildren() {
|
|
|
|
foundNode, offsetChange := getNodeAtIndexAux(child, index-offset, collapsedPaths)
|
|
|
|
offset += offsetChange
|
|
|
|
if foundNode != nil {
|
|
|
|
return foundNode, offset
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, offset
|
|
|
|
}
|
|
|
|
|
|
|
|
func getIndexForPath(node INode, path string, collapsedPaths map[string]bool) (int, bool) {
|
|
|
|
offset := 0
|
|
|
|
|
|
|
|
if node.GetPath() == path {
|
|
|
|
return offset, true
|
|
|
|
}
|
|
|
|
|
|
|
|
if !collapsedPaths[node.GetPath()] {
|
|
|
|
for _, child := range node.GetChildren() {
|
|
|
|
offsetChange, found := getIndexForPath(child, path, collapsedPaths)
|
|
|
|
offset += offsetChange + 1
|
|
|
|
if found {
|
|
|
|
return offset, true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return offset, false
|
|
|
|
}
|
|
|
|
|
|
|
|
func size(node INode, collapsedPaths map[string]bool) int {
|
|
|
|
output := 1
|
|
|
|
|
|
|
|
if !collapsedPaths[node.GetPath()] {
|
|
|
|
for _, child := range node.GetChildren() {
|
|
|
|
output += size(child, collapsedPaths)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return output
|
|
|
|
}
|
|
|
|
|
|
|
|
func compressAux(node INode) INode {
|
|
|
|
if node.IsLeaf() {
|
|
|
|
return node
|
|
|
|
}
|
|
|
|
|
|
|
|
children := node.GetChildren()
|
|
|
|
for i := range children {
|
|
|
|
grandchildren := children[i].GetChildren()
|
2021-07-27 13:00:33 +02:00
|
|
|
for len(grandchildren) == 1 && !grandchildren[0].IsLeaf() {
|
2021-03-30 14:56:59 +02:00
|
|
|
grandchildren[0].SetCompressionLevel(children[i].GetCompressionLevel() + 1)
|
|
|
|
children[i] = grandchildren[0]
|
|
|
|
grandchildren = children[i].GetChildren()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := range children {
|
|
|
|
children[i] = compressAux(children[i])
|
|
|
|
}
|
|
|
|
|
|
|
|
node.SetChildren(children)
|
|
|
|
|
|
|
|
return node
|
|
|
|
}
|
|
|
|
|
|
|
|
func getPathsMatching(node INode, test func(INode) bool) []string {
|
|
|
|
paths := []string{}
|
|
|
|
|
|
|
|
if test(node) {
|
|
|
|
paths = append(paths, node.GetPath())
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, child := range node.GetChildren() {
|
|
|
|
paths = append(paths, getPathsMatching(child, test)...)
|
|
|
|
}
|
|
|
|
|
|
|
|
return paths
|
|
|
|
}
|
|
|
|
|
|
|
|
func getLeaves(node INode) []INode {
|
|
|
|
if node.IsLeaf() {
|
|
|
|
return []INode{node}
|
|
|
|
}
|
|
|
|
|
|
|
|
output := []INode{}
|
|
|
|
for _, child := range node.GetChildren() {
|
|
|
|
output = append(output, getLeaves(child)...)
|
|
|
|
}
|
|
|
|
|
|
|
|
return output
|
|
|
|
}
|
2021-03-31 13:08:55 +02:00
|
|
|
|
|
|
|
func renderAux(s INode, collapsedPaths CollapsedPaths, prefix string, depth int, renderLine func(INode, int) string) []string {
|
|
|
|
isRoot := depth == -1
|
|
|
|
|
|
|
|
renderLineWithPrefix := func() string {
|
|
|
|
return prefix + renderLine(s, depth)
|
|
|
|
}
|
|
|
|
|
|
|
|
if s.IsLeaf() {
|
|
|
|
if isRoot {
|
|
|
|
return []string{}
|
|
|
|
}
|
|
|
|
return []string{renderLineWithPrefix()}
|
|
|
|
}
|
|
|
|
|
|
|
|
if collapsedPaths.IsCollapsed(s.GetPath()) {
|
|
|
|
return []string{fmt.Sprintf("%s %s", renderLineWithPrefix(), COLLAPSED_ARROW)}
|
|
|
|
}
|
|
|
|
|
|
|
|
arr := []string{}
|
|
|
|
if !isRoot {
|
|
|
|
arr = append(arr, fmt.Sprintf("%s %s", renderLineWithPrefix(), EXPANDED_ARROW))
|
|
|
|
}
|
|
|
|
|
|
|
|
newPrefix := prefix
|
|
|
|
if strings.HasSuffix(prefix, LAST_ITEM) {
|
|
|
|
newPrefix = strings.TrimSuffix(prefix, LAST_ITEM) + NOTHING
|
|
|
|
} else if strings.HasSuffix(prefix, INNER_ITEM) {
|
|
|
|
newPrefix = strings.TrimSuffix(prefix, INNER_ITEM) + NESTED
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, child := range s.GetChildren() {
|
|
|
|
isLast := i == len(s.GetChildren())-1
|
|
|
|
|
|
|
|
var childPrefix string
|
|
|
|
if isRoot {
|
|
|
|
childPrefix = newPrefix
|
|
|
|
} else if isLast {
|
|
|
|
childPrefix = newPrefix + LAST_ITEM
|
|
|
|
} else {
|
|
|
|
childPrefix = newPrefix + INNER_ITEM
|
|
|
|
}
|
|
|
|
|
|
|
|
arr = append(arr, renderAux(child, collapsedPaths, childPrefix, depth+1+s.GetCompressionLevel(), renderLine)...)
|
|
|
|
}
|
|
|
|
|
|
|
|
return arr
|
|
|
|
}
|