ncdu: implement multi selection
Co-authored-by: buengese <buengese@protonmail.com>
This commit is contained in:
parent
50c2e37aac
commit
0279bf3abb
1 changed files with 102 additions and 1 deletions
103
cmd/ncdu/ncdu.go
103
cmd/ncdu/ncdu.go
|
@ -92,6 +92,9 @@ func helpText() (tr []string) {
|
||||||
" u toggle human-readable format",
|
" u toggle human-readable format",
|
||||||
" n,s,C,A sort by name,size,count,average size",
|
" n,s,C,A sort by name,size,count,average size",
|
||||||
" d delete file/directory",
|
" d delete file/directory",
|
||||||
|
" v select file/directory",
|
||||||
|
" V enter visual select mode",
|
||||||
|
" D delete selected files/directories",
|
||||||
}
|
}
|
||||||
if !clipboard.Unsupported {
|
if !clipboard.Unsupported {
|
||||||
tr = append(tr, " y copy current path to clipboard")
|
tr = append(tr, " y copy current path to clipboard")
|
||||||
|
@ -126,11 +129,13 @@ type UI struct {
|
||||||
showCounts bool // toggle showing counts
|
showCounts bool // toggle showing counts
|
||||||
showDirAverageSize bool // toggle average size
|
showDirAverageSize bool // toggle average size
|
||||||
humanReadable bool // toggle human-readable format
|
humanReadable bool // toggle human-readable format
|
||||||
|
visualSelectMode bool // toggle visual selection mode
|
||||||
sortByName int8 // +1 for normal, 0 for off, -1 for reverse
|
sortByName int8 // +1 for normal, 0 for off, -1 for reverse
|
||||||
sortBySize int8
|
sortBySize int8
|
||||||
sortByCount int8
|
sortByCount int8
|
||||||
sortByAverageSize int8
|
sortByAverageSize int8
|
||||||
dirPosMap map[string]dirPos // store for directory positions
|
dirPosMap map[string]dirPos // store for directory positions
|
||||||
|
selectedEntries map[string]dirPos // selected entries of current directory
|
||||||
}
|
}
|
||||||
|
|
||||||
// Where we have got to in the directory listing
|
// Where we have got to in the directory listing
|
||||||
|
@ -361,6 +366,7 @@ func (u *UI) Draw() error {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
attrs, err := u.d.AttrI(u.sortPerm[n])
|
attrs, err := u.d.AttrI(u.sortPerm[n])
|
||||||
|
_, isSelected := u.selectedEntries[entry.String()]
|
||||||
fg := termbox.ColorWhite
|
fg := termbox.ColorWhite
|
||||||
if attrs.EntriesHaveErrors {
|
if attrs.EntriesHaveErrors {
|
||||||
fg = termbox.ColorYellow
|
fg = termbox.ColorYellow
|
||||||
|
@ -368,6 +374,9 @@ func (u *UI) Draw() error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fg = termbox.ColorRed
|
fg = termbox.ColorRed
|
||||||
}
|
}
|
||||||
|
if isSelected {
|
||||||
|
fg = termbox.ColorLightYellow
|
||||||
|
}
|
||||||
bg := termbox.ColorBlack
|
bg := termbox.ColorBlack
|
||||||
if n == dirPos.entry {
|
if n == dirPos.entry {
|
||||||
fg, bg = bg, fg
|
fg, bg = bg, fg
|
||||||
|
@ -494,6 +503,11 @@ func (u *UI) move(d int) {
|
||||||
dirPos.offset = entries - 1
|
dirPos.offset = entries - 1
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// toggle the current file for selection in selection mode
|
||||||
|
if u.visualSelectMode {
|
||||||
|
u.toggleSelectForCursor()
|
||||||
|
}
|
||||||
|
|
||||||
// write dirPos back for later
|
// write dirPos back for later
|
||||||
u.dirPosMap[u.path] = dirPos
|
u.dirPosMap[u.path] = dirPos
|
||||||
}
|
}
|
||||||
|
@ -503,11 +517,19 @@ func (u *UI) removeEntry(pos int) {
|
||||||
u.setCurrentDir(u.d)
|
u.setCurrentDir(u.d)
|
||||||
}
|
}
|
||||||
|
|
||||||
// delete the entry at the current position
|
|
||||||
func (u *UI) delete() {
|
func (u *UI) delete() {
|
||||||
if u.d == nil || len(u.entries) == 0 {
|
if u.d == nil || len(u.entries) == 0 {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
if len(u.selectedEntries) > 0 {
|
||||||
|
u.deleteSelected()
|
||||||
|
} else {
|
||||||
|
u.deleteSingle()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// delete the entry at the current position
|
||||||
|
func (u *UI) deleteSingle() {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
cursorPos := u.dirPosMap[u.path]
|
cursorPos := u.dirPosMap[u.path]
|
||||||
dirPos := u.sortPerm[cursorPos.entry]
|
dirPos := u.sortPerm[cursorPos.entry]
|
||||||
|
@ -553,6 +575,62 @@ func (u *UI) delete() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (u *UI) deleteSelected() {
|
||||||
|
ctx := context.Background()
|
||||||
|
|
||||||
|
u.boxMenu = []string{"cancel", "confirm"}
|
||||||
|
|
||||||
|
u.boxMenuHandler = func(f fs.Fs, p string, o int) (string, error) {
|
||||||
|
if o != 1 {
|
||||||
|
return "Aborted!", nil
|
||||||
|
}
|
||||||
|
|
||||||
|
positionsToDelete := make([]int, len(u.selectedEntries))
|
||||||
|
i := 0
|
||||||
|
|
||||||
|
for key, cursorPos := range u.selectedEntries {
|
||||||
|
|
||||||
|
dirPos := u.sortPerm[cursorPos.entry]
|
||||||
|
dirEntry := u.entries[dirPos]
|
||||||
|
var err error
|
||||||
|
|
||||||
|
if obj, isFile := dirEntry.(fs.Object); isFile {
|
||||||
|
err = operations.DeleteFile(ctx, obj)
|
||||||
|
} else {
|
||||||
|
err = operations.Purge(ctx, f, dirEntry.String())
|
||||||
|
}
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
delete(u.selectedEntries, key)
|
||||||
|
positionsToDelete[i] = dirPos
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
|
||||||
|
// deleting all entries at once, as doing it during the deletions
|
||||||
|
// could cause issues.
|
||||||
|
sort.Slice(positionsToDelete, func(i, j int) bool {
|
||||||
|
return positionsToDelete[i] > positionsToDelete[j]
|
||||||
|
})
|
||||||
|
for _, dirPos := range positionsToDelete {
|
||||||
|
u.removeEntry(dirPos)
|
||||||
|
}
|
||||||
|
|
||||||
|
// move cursor at end if needed
|
||||||
|
cursorPos := u.dirPosMap[u.path]
|
||||||
|
if cursorPos.entry >= len(u.entries) {
|
||||||
|
u.move(-1)
|
||||||
|
}
|
||||||
|
|
||||||
|
return "Successfully deleted all items!", nil
|
||||||
|
}
|
||||||
|
u.popupBox([]string{
|
||||||
|
"Delete selected items?",
|
||||||
|
fmt.Sprintf("ALL %d items will be deleted", len(u.selectedEntries))})
|
||||||
|
}
|
||||||
|
|
||||||
func (u *UI) displayPath() {
|
func (u *UI) displayPath() {
|
||||||
u.togglePopupBox([]string{
|
u.togglePopupBox([]string{
|
||||||
"Current Path",
|
"Current Path",
|
||||||
|
@ -661,6 +739,8 @@ func (u *UI) setCurrentDir(d *scan.Dir) {
|
||||||
u.d = d
|
u.d = d
|
||||||
u.entries = d.Entries()
|
u.entries = d.Entries()
|
||||||
u.path = fspath.JoinRootPath(u.fsName, d.Path())
|
u.path = fspath.JoinRootPath(u.fsName, d.Path())
|
||||||
|
u.selectedEntries = make(map[string]dirPos)
|
||||||
|
u.visualSelectMode = false
|
||||||
u.sortCurrentDir()
|
u.sortCurrentDir()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -737,6 +817,20 @@ func (u *UI) toggleSort(sortType *int8) {
|
||||||
u.sortCurrentDir()
|
u.sortCurrentDir()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (u *UI) toggleSelectForCursor() {
|
||||||
|
cursorPos := u.dirPosMap[u.path]
|
||||||
|
dirPos := u.sortPerm[cursorPos.entry]
|
||||||
|
dirEntry := u.entries[dirPos]
|
||||||
|
|
||||||
|
_, present := u.selectedEntries[dirEntry.String()]
|
||||||
|
|
||||||
|
if present {
|
||||||
|
delete(u.selectedEntries, dirEntry.String())
|
||||||
|
} else {
|
||||||
|
u.selectedEntries[dirEntry.String()] = cursorPos
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// NewUI creates a new user interface for ncdu on f
|
// NewUI creates a new user interface for ncdu on f
|
||||||
func NewUI(f fs.Fs) *UI {
|
func NewUI(f fs.Fs) *UI {
|
||||||
return &UI{
|
return &UI{
|
||||||
|
@ -752,6 +846,7 @@ func NewUI(f fs.Fs) *UI {
|
||||||
sortBySize: 1,
|
sortBySize: 1,
|
||||||
sortByCount: 0,
|
sortByCount: 0,
|
||||||
dirPosMap: make(map[string]dirPos),
|
dirPosMap: make(map[string]dirPos),
|
||||||
|
selectedEntries: make(map[string]dirPos),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -845,6 +940,10 @@ outer:
|
||||||
u.toggleSort(&u.sortByName)
|
u.toggleSort(&u.sortByName)
|
||||||
case 's':
|
case 's':
|
||||||
u.toggleSort(&u.sortBySize)
|
u.toggleSort(&u.sortBySize)
|
||||||
|
case 'v':
|
||||||
|
u.toggleSelectForCursor()
|
||||||
|
case 'V':
|
||||||
|
u.visualSelectMode = !u.visualSelectMode
|
||||||
case 'C':
|
case 'C':
|
||||||
u.toggleSort(&u.sortByCount)
|
u.toggleSort(&u.sortByCount)
|
||||||
case 'A':
|
case 'A':
|
||||||
|
@ -857,6 +956,8 @@ outer:
|
||||||
u.delete()
|
u.delete()
|
||||||
case 'u':
|
case 'u':
|
||||||
u.humanReadable = !u.humanReadable
|
u.humanReadable = !u.humanReadable
|
||||||
|
case 'D':
|
||||||
|
u.deleteSelected()
|
||||||
case '?':
|
case '?':
|
||||||
u.togglePopupBox(helpText())
|
u.togglePopupBox(helpText())
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue