fssort.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  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. /*
  15. Quick utilties to sort file list according to different modes
  16. */
  17. func SortFileList(filelistRealpath []string, fileInfos []fs.FileInfo, sortMode string) []string {
  18. //Build a filelist with information based on the given filelist
  19. parsedFilelist := []*sortBufferedStructure{}
  20. if len(filelistRealpath) != len(fileInfos) {
  21. //Invalid usage
  22. return filelistRealpath
  23. }
  24. for i, file := range filelistRealpath {
  25. thisFileInfo := sortBufferedStructure{
  26. Filename: filepath.Base(file),
  27. Filepath: file,
  28. }
  29. //Get Filesize
  30. fi := fileInfos[i]
  31. thisFileInfo.Filesize = fi.Size()
  32. thisFileInfo.ModTime = fi.ModTime().Unix()
  33. parsedFilelist = append(parsedFilelist, &thisFileInfo)
  34. }
  35. //Sort the filelist
  36. if sortMode == "default" {
  37. //Sort by name, convert filename to window sorting methods
  38. sort.Slice(parsedFilelist, func(i, j int) bool {
  39. return strings.ToLower(parsedFilelist[i].Filename) < strings.ToLower(parsedFilelist[j].Filename)
  40. })
  41. } else if sortMode == "reverse" {
  42. //Sort by reverse name
  43. sort.Slice(parsedFilelist, func(i, j int) bool {
  44. return strings.ToLower(parsedFilelist[i].Filename) > strings.ToLower(parsedFilelist[j].Filename)
  45. })
  46. } else if sortMode == "smallToLarge" {
  47. sort.Slice(parsedFilelist, func(i, j int) bool { return parsedFilelist[i].Filesize < parsedFilelist[j].Filesize })
  48. } else if sortMode == "largeToSmall" {
  49. sort.Slice(parsedFilelist, func(i, j int) bool { return parsedFilelist[i].Filesize > parsedFilelist[j].Filesize })
  50. } else if sortMode == "mostRecent" {
  51. sort.Slice(parsedFilelist, func(i, j int) bool { return parsedFilelist[i].ModTime > parsedFilelist[j].ModTime })
  52. } else if sortMode == "leastRecent" {
  53. sort.Slice(parsedFilelist, func(i, j int) bool { return parsedFilelist[i].ModTime < parsedFilelist[j].ModTime })
  54. } else if sortMode == "smart" {
  55. parsedFilelist = SortNaturalFilelist(parsedFilelist)
  56. }
  57. results := []string{}
  58. for _, sortedFile := range parsedFilelist {
  59. results = append(results, sortedFile.Filepath)
  60. }
  61. return results
  62. }
  63. func SortDirEntryList(dirEntries []fs.DirEntry, sortMode string) []fs.DirEntry {
  64. entries := map[string]fs.DirEntry{}
  65. fnames := []string{}
  66. fis := []fs.FileInfo{}
  67. for _, de := range dirEntries {
  68. fnames = append(fnames, de.Name())
  69. fstat, _ := de.Info()
  70. fis = append(fis, fstat)
  71. thisFsDirEntry := de
  72. entries[de.Name()] = thisFsDirEntry
  73. }
  74. //Sort it
  75. sortedNameList := SortFileList(fnames, fis, sortMode)
  76. //Update dirEntry sequence
  77. newDirEntry := []fs.DirEntry{}
  78. for _, key := range sortedNameList {
  79. newDirEntry = append(newDirEntry, entries[key])
  80. }
  81. return newDirEntry
  82. }
  83. func SortModeIsSupported(sortMode string) bool {
  84. return contains(sortMode, []string{"default", "reverse", "smallToLarge", "largeToSmall", "mostRecent", "leastRecent", "smart"})
  85. }
  86. func contains(item string, slice []string) bool {
  87. set := make(map[string]struct{}, len(slice))
  88. for _, s := range slice {
  89. set[s] = struct{}{}
  90. }
  91. _, ok := set[item]
  92. return ok
  93. }