aerc/widgets/dirlist.go

181 lines
3.9 KiB
Go
Raw Normal View History

2019-01-13 20:25:46 +01:00
package widgets
import (
"log"
"sort"
"github.com/gdamore/tcell"
2019-05-18 02:57:10 +02:00
"git.sr.ht/~sircmpwn/aerc/config"
"git.sr.ht/~sircmpwn/aerc/lib/ui"
"git.sr.ht/~sircmpwn/aerc/worker/types"
2019-01-13 20:25:46 +01:00
)
type DirectoryList struct {
ui.Invalidatable
conf *config.AccountConfig
dirs []string
logger *log.Logger
selecting string
selected string
spinner *Spinner
worker *types.Worker
2019-01-13 20:25:46 +01:00
}
func NewDirectoryList(conf *config.AccountConfig,
logger *log.Logger, worker *types.Worker) *DirectoryList {
2019-01-14 02:02:21 +01:00
dirlist := &DirectoryList{
conf: conf,
logger: logger,
spinner: NewSpinner(),
worker: worker,
}
dirlist.spinner.OnInvalidate(func(_ ui.Drawable) {
dirlist.Invalidate()
})
dirlist.spinner.Start()
return dirlist
2019-01-13 20:25:46 +01:00
}
func (dirlist *DirectoryList) UpdateList(done func(dirs []string)) {
var dirs []string
2019-01-13 20:25:46 +01:00
dirlist.worker.PostAction(
&types.ListDirectories{}, func(msg types.WorkerMessage) {
switch msg := msg.(type) {
case *types.Directory:
dirs = append(dirs, msg.Name)
2019-01-13 20:25:46 +01:00
case *types.Done:
sort.Strings(dirs)
dirlist.dirs = dirs
2019-01-14 02:02:21 +01:00
dirlist.spinner.Stop()
dirlist.Invalidate()
if done != nil {
done(dirs)
}
2019-01-13 20:25:46 +01:00
}
})
}
func (dirlist *DirectoryList) Select(name string) {
2019-03-15 03:34:34 +01:00
dirlist.selecting = name
dirlist.worker.PostAction(&types.OpenDirectory{Directory: name},
func(msg types.WorkerMessage) {
switch msg.(type) {
case *types.Error:
dirlist.selecting = ""
case *types.Done:
2019-03-15 03:34:34 +01:00
dirlist.selected = dirlist.selecting
2019-06-12 08:31:51 +02:00
dirlist.filterDirsByFoldersConfig()
hasSelected := false
for _, d := range dirlist.dirs {
if d == dirlist.selected {
hasSelected = true
break
}
}
if !hasSelected && dirlist.selected != "" {
dirlist.dirs = append(dirlist.dirs, dirlist.selected)
}
sort.Strings(dirlist.dirs)
2019-03-15 03:34:34 +01:00
}
dirlist.Invalidate()
})
dirlist.Invalidate()
}
2019-03-15 03:34:34 +01:00
func (dirlist *DirectoryList) Selected() string {
return dirlist.selected
}
2019-01-13 20:25:46 +01:00
func (dirlist *DirectoryList) Invalidate() {
dirlist.DoInvalidate(dirlist)
2019-01-13 20:25:46 +01:00
}
func (dirlist *DirectoryList) Draw(ctx *ui.Context) {
ctx.Fill(0, 0, ctx.Width(), ctx.Height(), ' ', tcell.StyleDefault)
2019-01-14 02:02:21 +01:00
if dirlist.spinner.IsRunning() {
dirlist.spinner.Draw(ctx)
return
}
row := 0
for _, name := range dirlist.dirs {
if row >= ctx.Height() {
break
}
if len(dirlist.conf.Folders) > 1 && name != dirlist.selected {
idx := sort.SearchStrings(dirlist.conf.Folders, name)
if idx == len(dirlist.conf.Folders) ||
dirlist.conf.Folders[idx] != name {
continue
}
}
style := tcell.StyleDefault
if name == dirlist.selected {
style = style.Reverse(true)
}
ctx.Fill(0, row, ctx.Width(), 1, ' ', style)
ctx.Printf(0, row, style, "%s", name)
row++
}
2019-01-13 20:25:46 +01:00
}
2019-03-11 02:15:24 +01:00
func (dirlist *DirectoryList) nextPrev(delta int) {
for i, dir := range dirlist.dirs {
if dir == dirlist.selected {
var j int
ndirs := len(dirlist.dirs)
for j = i + delta; j != i; j += delta {
if j < 0 {
j = ndirs - 1
}
if j >= ndirs {
j = 0
}
name := dirlist.dirs[j]
if len(dirlist.conf.Folders) > 1 && name != dirlist.selected {
idx := sort.SearchStrings(dirlist.conf.Folders, name)
if idx == len(dirlist.conf.Folders) ||
dirlist.conf.Folders[idx] != name {
continue
}
}
break
}
dirlist.Select(dirlist.dirs[j])
break
}
}
}
func (dirlist *DirectoryList) Next() {
dirlist.nextPrev(1)
}
func (dirlist *DirectoryList) Prev() {
dirlist.nextPrev(-1)
}
2019-06-12 08:31:51 +02:00
// filterDirsByFoldersConfig filters a folders list to only contain folders
// present in the account.folders config option
func (dirlist *DirectoryList) filterDirsByFoldersConfig() {
// config option defaults to show all if unset
if len(dirlist.conf.Folders) == 0 {
return
}
var filtered []string
for _, folder := range dirlist.dirs {
for _, cfgfolder := range dirlist.conf.Folders {
if folder == cfgfolder {
filtered = append(filtered, folder)
break
}
}
}
dirlist.dirs = filtered
}