lf/nav.go

513 lines
9.0 KiB
Go
Raw Normal View History

2016-08-13 12:49:04 +00:00
package main
import (
"errors"
"fmt"
"log"
"os"
"os/exec"
"path/filepath"
2016-11-10 21:18:56 +00:00
"sort"
2016-08-13 12:49:04 +00:00
"strings"
)
type LinkState int8
const (
NotLink LinkState = iota
Working
Broken
)
type File struct {
os.FileInfo
LinkState LinkState
Path string
2016-08-13 12:49:04 +00:00
}
2016-11-10 21:18:56 +00:00
type FilesSortable struct {
files []*File
less func(i, j int) bool
}
func (f FilesSortable) Len() int { return len(f.files) }
func (f FilesSortable) Swap(i, j int) { f.files[i], f.files[j] = f.files[j], f.files[i] }
func (f FilesSortable) Less(i, j int) bool { return f.less(i, j) }
// TODO: Replace with `sort.SliceStable` once available
func sortFilesStable(files []*File, less func(i, j int) bool) {
sort.Stable(FilesSortable{files: files, less: less})
}
func getFilesSorted(path string) []*File {
fi, err := readdir(path)
if err != nil {
log.Printf("reading directory: %s", err)
2016-08-13 12:49:04 +00:00
}
switch gOpts.sortby {
case "name":
sortFilesStable(fi, func(i, j int) bool {
return strings.ToLower(fi[i].Name()) < strings.ToLower(fi[j].Name())
})
2016-08-13 12:49:04 +00:00
case "size":
sortFilesStable(fi, func(i, j int) bool {
return fi[i].Size() < fi[j].Size()
})
2016-08-13 12:49:04 +00:00
case "time":
sortFilesStable(fi, func(i, j int) bool {
return fi[i].ModTime().Before(fi[j].ModTime())
})
2016-08-13 12:49:04 +00:00
default:
log.Printf("unknown sorting type: %s", gOpts.sortby)
}
if gOpts.dirfirst {
sortFilesStable(fi, func(i, j int) bool {
if fi[i].IsDir() == fi[j].IsDir() {
return i < j
}
return fi[i].IsDir()
})
}
//TODO this should be optional
sortFilesStable(fi, func(i, j int) bool {
nums1, rest1, numFirst1 := extractNums(fi[i].Name())
nums2, rest2, numFirst2 := extractNums(fi[j].Name())
if numFirst1 != numFirst2 {
return i < j
}
if numFirst1 {
if nums1[0] != nums2[0] {
return nums1[0] < nums2[0]
}
nums1 = nums1[1:]
nums2 = nums2[1:]
}
for k := 0; k < len(nums1) && k < len(nums2); k++ {
if rest1[k] != rest2[k] {
return i < j
}
if nums1[k] != nums2[k] {
return nums1[k] < nums2[k]
}
}
return i < j
})
2016-08-13 12:49:04 +00:00
return fi
}
func readdir(path string) ([]*File, error) {
f, err := os.Open(path)
2016-08-13 12:49:04 +00:00
if err != nil {
return nil, err
2016-08-13 12:49:04 +00:00
}
names, err := f.Readdirnames(-1)
fi := make([]*File, 0, len(names))
for _, filename := range names {
if !gOpts.hidden && filename[0] == '.' {
continue
}
fpath := filepath.Join(path, filename)
lstat, lerr := os.Lstat(fpath)
if os.IsNotExist(lerr) {
continue
}
if lerr != nil {
return fi, lerr
}
var linkState LinkState
if lstat.Mode()&os.ModeSymlink != 0 {
stat, serr := os.Stat(fpath)
if serr == nil {
linkState = Working
lstat = stat
} else {
linkState = Broken
log.Printf("getting link destination info: %s", serr)
}
}
fi = append(fi, &File{
FileInfo: lstat,
LinkState: linkState,
Path: fpath,
})
}
return fi, err
}
2016-08-13 12:49:04 +00:00
type Dir struct {
ind int // which entry is highlighted
pos int // which line in the ui highlighted entry is
path string
fi []*File
}
func newDir(path string) *Dir {
2016-08-13 12:49:04 +00:00
return &Dir{
path: path,
fi: getFilesSorted(path),
2016-08-13 12:49:04 +00:00
}
}
func (dir *Dir) renew(height int) {
var name string
if len(dir.fi) != 0 {
name = dir.fi[dir.ind].Name()
}
dir.fi = getFilesSorted(dir.path)
2016-08-13 12:49:04 +00:00
dir.load(dir.ind, dir.pos, height, name)
}
func (dir *Dir) load(ind, pos, height int, name string) {
if len(dir.fi) == 0 {
dir.ind, dir.pos = 0, 0
return
}
ind = max(0, min(ind, len(dir.fi)-1))
if dir.fi[ind].Name() != name {
for i, f := range dir.fi {
if f.Name() == name {
ind = i
break
}
}
edge := min(gOpts.scrolloff, len(dir.fi)-ind-1)
pos = min(ind, height-edge-1)
}
dir.ind = ind
dir.pos = pos
}
type Nav struct {
dirs []*Dir
inds map[string]int
poss map[string]int
names map[string]string
marks map[string]bool
saves map[string]bool
2016-08-13 12:49:04 +00:00
height int
}
func getDirs(wd string, height int) []*Dir {
var dirs []*Dir
for curr, base := wd, ""; !isRoot(base); curr, base = filepath.Dir(curr), filepath.Base(curr) {
2016-08-13 12:49:04 +00:00
dir := newDir(curr)
for i, f := range dir.fi {
if f.Name() == base {
dir.ind = i
edge := min(gOpts.scrolloff, len(dir.fi)-dir.ind-1)
dir.pos = min(i, height-edge-1)
break
}
}
dirs = append(dirs, dir)
}
for i, j := 0, len(dirs)-1; i < j; i, j = i+1, j-1 {
dirs[i], dirs[j] = dirs[j], dirs[i]
}
return dirs
}
func newNav(height int) *Nav {
wd, err := os.Getwd()
if err != nil {
2016-08-17 20:22:11 +00:00
log.Printf("getting current directory: %s", err)
2016-08-13 12:49:04 +00:00
}
dirs := getDirs(wd, height)
return &Nav{
dirs: dirs,
inds: make(map[string]int),
poss: make(map[string]int),
names: make(map[string]string),
marks: make(map[string]bool),
saves: make(map[string]bool),
2016-08-13 12:49:04 +00:00
height: height,
}
}
func (nav *Nav) renew(height int) {
nav.height = height
for _, d := range nav.dirs {
d.renew(nav.height)
}
for m := range nav.marks {
if _, err := os.Stat(m); os.IsNotExist(err) {
delete(nav.marks, m)
}
}
}
func (nav *Nav) up(dist int) {
2016-08-14 15:39:02 +00:00
dir := nav.currDir()
if dir.ind == 0 {
return
}
dir.ind -= dist
dir.ind = max(0, dir.ind)
2016-08-14 15:39:02 +00:00
dir.pos -= dist
2016-08-14 15:39:02 +00:00
edge := min(gOpts.scrolloff, dir.ind)
dir.pos = max(dir.pos, edge)
}
func (nav *Nav) down(dist int) {
2016-08-13 12:49:04 +00:00
dir := nav.currDir()
maxind := len(dir.fi) - 1
if dir.ind >= maxind {
return
}
dir.ind += dist
dir.ind = min(maxind, dir.ind)
2016-08-13 12:49:04 +00:00
dir.pos += dist
2016-08-13 12:49:04 +00:00
edge := min(gOpts.scrolloff, maxind-dir.ind)
2016-09-02 19:47:11 +00:00
// use a smaller value when the height is even and scrolloff is maxed
// in order to stay at the same row as much as possible while up/down
2016-09-02 20:06:25 +00:00
edge = min(edge, nav.height/2+nav.height%2-1)
2016-09-02 19:47:11 +00:00
2016-08-13 12:49:04 +00:00
dir.pos = min(dir.pos, nav.height-edge-1)
dir.pos = min(dir.pos, maxind)
}
func (nav *Nav) updir() error {
if len(nav.dirs) <= 1 {
return nil
}
dir := nav.currDir()
nav.inds[dir.path] = dir.ind
nav.poss[dir.path] = dir.pos
if len(dir.fi) != 0 {
nav.names[dir.path] = dir.fi[dir.ind].Name()
}
nav.dirs = nav.dirs[:len(nav.dirs)-1]
if err := os.Chdir(filepath.Dir(dir.path)); err != nil {
2016-08-13 12:49:04 +00:00
return fmt.Errorf("updir: %s", err)
}
return nil
}
var ErrNotDir = fmt.Errorf("not a directory")
2016-08-13 12:49:04 +00:00
func (nav *Nav) open() error {
curr, err := nav.currFile()
if err != nil {
return err
}
if !curr.IsDir() {
return ErrNotDir
}
path := curr.Path
2016-08-13 12:49:04 +00:00
2016-08-14 12:37:22 +00:00
dir := newDir(path)
2016-08-14 12:37:22 +00:00
dir.load(nav.inds[path], nav.poss[path], nav.height, nav.names[path])
2016-08-13 12:49:04 +00:00
2016-08-14 12:37:22 +00:00
nav.dirs = append(nav.dirs, dir)
2016-08-13 12:49:04 +00:00
2016-08-17 20:28:42 +00:00
if err := os.Chdir(path); err != nil {
2016-08-14 12:37:22 +00:00
return fmt.Errorf("open: %s", err)
2016-08-13 12:49:04 +00:00
}
return nil
}
func (nav *Nav) bot() {
dir := nav.currDir()
dir.ind = len(dir.fi) - 1
dir.pos = min(dir.ind, nav.height-1)
}
func (nav *Nav) top() {
dir := nav.currDir()
dir.ind = 0
dir.pos = 0
}
func (nav *Nav) cd(wd string) error {
2016-08-14 15:38:33 +00:00
wd = strings.Replace(wd, "~", envHome, -1)
2016-10-01 21:39:03 +00:00
wd = filepath.Clean(wd)
2016-08-14 15:38:33 +00:00
if !filepath.IsAbs(wd) {
wd = filepath.Join(nav.currDir().path, wd)
2016-08-13 12:49:04 +00:00
}
2016-08-17 20:28:42 +00:00
if err := os.Chdir(wd); err != nil {
2016-08-13 12:49:04 +00:00
return fmt.Errorf("cd: %s", err)
}
nav.dirs = getDirs(wd, nav.height)
// TODO: save/load ind and pos from the map
return nil
}
func (nav *Nav) toggleMark(path string) {
2016-08-13 12:49:04 +00:00
if nav.marks[path] {
delete(nav.marks, path)
} else {
nav.marks[path] = true
}
}
func (nav *Nav) toggle() {
curr, err := nav.currFile()
if err != nil {
return
}
nav.toggleMark(curr.Path)
2016-08-13 12:49:04 +00:00
nav.down(1)
2016-08-13 12:49:04 +00:00
}
func (nav *Nav) invert() {
last := nav.currDir()
for _, f := range last.fi {
path := filepath.Join(last.path, f.Name())
nav.toggleMark(path)
}
}
func (nav *Nav) save(copy bool) error {
2016-08-13 12:49:04 +00:00
if len(nav.marks) == 0 {
curr, err := nav.currFile()
if err != nil {
return errors.New("no file selected")
}
if err := saveFiles([]string{curr.Path}, copy); err != nil {
2016-08-13 12:49:04 +00:00
return err
}
nav.saves = make(map[string]bool)
nav.saves[curr.Path] = copy
2016-08-13 12:49:04 +00:00
} else {
var fs []string
for f := range nav.marks {
fs = append(fs, f)
}
if err := saveFiles(fs, copy); err != nil {
2016-08-13 12:49:04 +00:00
return err
}
nav.saves = make(map[string]bool)
for f := range nav.marks {
nav.saves[f] = copy
}
2016-08-13 12:49:04 +00:00
}
return nil
}
2016-11-06 15:09:18 +00:00
func (nav *Nav) put() error {
list, copy, err := loadFiles()
2016-08-13 12:49:04 +00:00
if err != nil {
return err
}
if len(list) == 0 {
return errors.New("no file in yank/delete buffer")
}
dir := nav.currDir()
var args []string
2016-08-13 12:49:04 +00:00
var sh string
if copy {
2016-08-13 12:49:04 +00:00
sh = "cp"
args = append(args, "-r")
2016-08-13 12:49:04 +00:00
} else {
sh = "mv"
}
args = append(args, "--backup=numbered")
args = append(args, list...)
args = append(args, dir.path)
2016-08-13 12:49:04 +00:00
cmd := exec.Command(sh, args...)
if err := cmd.Run(); err != nil {
return fmt.Errorf("%s: %s", sh, err)
}
// TODO: async?
return nil
}
func (nav *Nav) sync() error {
list, copy, err := loadFiles()
if err != nil {
return err
}
nav.saves = make(map[string]bool)
for _, f := range list {
nav.saves[f] = copy
}
return nil
}
2016-08-13 12:49:04 +00:00
func (nav *Nav) currDir() *Dir {
return nav.dirs[len(nav.dirs)-1]
}
func (nav *Nav) currFile() (*File, error) {
2016-08-13 12:49:04 +00:00
last := nav.dirs[len(nav.dirs)-1]
if len(last.fi) == 0 {
return nil, fmt.Errorf("empty directory")
}
return last.fi[last.ind], nil
2016-08-13 12:49:04 +00:00
}
func (nav *Nav) currMarks() []string {
marks := make([]string, 0, len(nav.marks))
for m := range nav.marks {
marks = append(marks, m)
}
return marks
}