fssort.go 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. package fssort
  2. import (
  3. "io/fs"
  4. "path/filepath"
  5. "sort"
  6. "strings"
  7. )
  8. type sortBufferedStructure struct {
  9. Filename string
  10. Filepath string
  11. Filesize int64
  12. ModTime int64
  13. }
  14. var ValidSortModes = []string{"default", "reverse", "smallToLarge", "largeToSmall", "mostRecent", "leastRecent", "smart", "fileTypeAsce", "fileTypeDesc"}
  15. /*
  16. Quick utilties to sort file list according to different modes
  17. */
  18. func SortFileList(filelistRealpath []string, fileInfos []fs.FileInfo, sortMode string) []string {
  19. //Build a filelist with information based on the given filelist
  20. parsedFilelist := []*sortBufferedStructure{}
  21. if len(filelistRealpath) != len(fileInfos) {
  22. //Invalid usage
  23. return filelistRealpath
  24. }
  25. for i, file := range filelistRealpath {
  26. thisFileInfo := sortBufferedStructure{
  27. Filename: filepath.Base(file),
  28. Filepath: file,
  29. }
  30. //Get Filesize
  31. fi := fileInfos[i]
  32. thisFileInfo.Filesize = fi.Size()
  33. thisFileInfo.ModTime = fi.ModTime().Unix()
  34. parsedFilelist = append(parsedFilelist, &thisFileInfo)
  35. }
  36. //Sort the filelist
  37. if sortMode == "default" {
  38. //Sort by name, convert filename to window sorting methods
  39. sort.Slice(parsedFilelist, func(i, j int) bool {
  40. return strings.ToLower(parsedFilelist[i].Filename) < strings.ToLower(parsedFilelist[j].Filename)
  41. })
  42. } else if sortMode == "reverse" {
  43. //Sort by reverse name
  44. sort.Slice(parsedFilelist, func(i, j int) bool {
  45. return strings.ToLower(parsedFilelist[i].Filename) > strings.ToLower(parsedFilelist[j].Filename)
  46. })
  47. } else if sortMode == "smallToLarge" {
  48. sort.Slice(parsedFilelist, func(i, j int) bool { return parsedFilelist[i].Filesize < parsedFilelist[j].Filesize })
  49. } else if sortMode == "largeToSmall" {
  50. sort.Slice(parsedFilelist, func(i, j int) bool { return parsedFilelist[i].Filesize > parsedFilelist[j].Filesize })
  51. } else if sortMode == "mostRecent" {
  52. sort.Slice(parsedFilelist, func(i, j int) bool {
  53. if parsedFilelist[i].ModTime == parsedFilelist[j].ModTime {
  54. return strings.ToLower(parsedFilelist[i].Filename) < strings.ToLower(parsedFilelist[j].Filename)
  55. }
  56. return parsedFilelist[i].ModTime > parsedFilelist[j].ModTime
  57. })
  58. } else if sortMode == "leastRecent" {
  59. sort.Slice(parsedFilelist, func(i, j int) bool {
  60. if parsedFilelist[i].ModTime == parsedFilelist[j].ModTime {
  61. return strings.ToLower(parsedFilelist[i].Filename) > strings.ToLower(parsedFilelist[j].Filename)
  62. }
  63. return parsedFilelist[i].ModTime < parsedFilelist[j].ModTime
  64. })
  65. } else if sortMode == "smart" {
  66. parsedFilelist = SortNaturalFilelist(parsedFilelist)
  67. } else if sortMode == "fileTypeAsce" {
  68. sort.Slice(parsedFilelist, func(i, j int) bool {
  69. exti := filepath.Ext(parsedFilelist[i].Filename)
  70. extj := filepath.Ext(parsedFilelist[j].Filename)
  71. exti = strings.TrimPrefix(exti, ".")
  72. extj = strings.TrimPrefix(extj, ".")
  73. return exti < extj
  74. })
  75. } else if sortMode == "fileTypeDesc" {
  76. sort.Slice(parsedFilelist, func(i, j int) bool {
  77. exti := filepath.Ext(parsedFilelist[i].Filename)
  78. extj := filepath.Ext(parsedFilelist[j].Filename)
  79. exti = strings.TrimPrefix(exti, ".")
  80. extj = strings.TrimPrefix(extj, ".")
  81. return exti > extj
  82. })
  83. }
  84. results := []string{}
  85. for _, sortedFile := range parsedFilelist {
  86. results = append(results, sortedFile.Filepath)
  87. }
  88. return results
  89. }
  90. func SortDirEntryList(dirEntries []fs.DirEntry, sortMode string) []fs.DirEntry {
  91. entries := map[string]fs.DirEntry{}
  92. fnames := []string{}
  93. fis := []fs.FileInfo{}
  94. for _, de := range dirEntries {
  95. fnames = append(fnames, de.Name())
  96. fstat, _ := de.Info()
  97. fis = append(fis, fstat)
  98. thisFsDirEntry := de
  99. entries[de.Name()] = thisFsDirEntry
  100. }
  101. //Sort it
  102. sortedNameList := SortFileList(fnames, fis, sortMode)
  103. //Update dirEntry sequence
  104. newDirEntry := []fs.DirEntry{}
  105. for _, key := range sortedNameList {
  106. newDirEntry = append(newDirEntry, entries[key])
  107. }
  108. return newDirEntry
  109. }
  110. func SortModeIsSupported(sortMode string) bool {
  111. return contains(sortMode, ValidSortModes)
  112. }
  113. func contains(item string, slice []string) bool {
  114. set := make(map[string]struct{}, len(slice))
  115. for _, s := range slice {
  116. set[s] = struct{}{}
  117. }
  118. _, ok := set[item]
  119. return ok
  120. }