mirror of
https://github.com/jesseduffield/lazygit.git
synced 2025-01-26 05:37:18 +02:00
8edad826ca
This begins a big refactor of moving more code out of the Gui struct into contexts, controllers, and helpers. We also move some code into structs in the gui package purely for the sake of better encapsulation
135 lines
3.3 KiB
Go
135 lines
3.3 KiB
Go
package gui
|
|
|
|
import (
|
|
"os"
|
|
"path/filepath"
|
|
|
|
"github.com/fsnotify/fsnotify"
|
|
"github.com/jesseduffield/lazygit/pkg/commands/models"
|
|
"github.com/jesseduffield/lazygit/pkg/gui/types"
|
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
|
"github.com/sirupsen/logrus"
|
|
)
|
|
|
|
// macs for some bizarre reason cap the number of watchable files to 256.
|
|
// there's no obvious platform agnostic way to check the situation of the user's
|
|
// computer so we're just arbitrarily capping at 200. This isn't so bad because
|
|
// file watching is only really an added bonus for faster refreshing.
|
|
const MAX_WATCHED_FILES = 50
|
|
|
|
var _ types.IFileWatcher = new(fileWatcher)
|
|
|
|
type fileWatcher struct {
|
|
Watcher *fsnotify.Watcher
|
|
WatchedFilenames []string
|
|
Log *logrus.Entry
|
|
Disabled bool
|
|
}
|
|
|
|
func NewFileWatcher(log *logrus.Entry) *fileWatcher {
|
|
// TODO: get this going again, and ensure we don't see any crashes from it
|
|
return &fileWatcher{
|
|
Disabled: true,
|
|
}
|
|
}
|
|
|
|
func (w *fileWatcher) watchingFilename(filename string) bool {
|
|
for _, watchedFilename := range w.WatchedFilenames {
|
|
if watchedFilename == filename {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (w *fileWatcher) popOldestFilename() {
|
|
// shift the last off the array to make way for this one
|
|
oldestFilename := w.WatchedFilenames[0]
|
|
w.WatchedFilenames = w.WatchedFilenames[1:]
|
|
if err := w.Watcher.Remove(oldestFilename); err != nil {
|
|
// swallowing errors here because it doesn't really matter if we can't unwatch a file
|
|
w.Log.Error(err)
|
|
}
|
|
}
|
|
|
|
func (w *fileWatcher) watchFilename(filename string) {
|
|
if err := w.Watcher.Add(filename); err != nil {
|
|
// swallowing errors here because it doesn't really matter if we can't watch a file
|
|
w.Log.Error(err)
|
|
}
|
|
|
|
// assume we're watching it now to be safe
|
|
w.WatchedFilenames = append(w.WatchedFilenames, filename)
|
|
}
|
|
|
|
func (w *fileWatcher) AddFilesToFileWatcher(files []*models.File) error {
|
|
if w.Disabled {
|
|
return nil
|
|
}
|
|
|
|
if len(files) == 0 {
|
|
return nil
|
|
}
|
|
|
|
// watch the files for changes
|
|
dirName, err := os.Getwd()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
for _, file := range files[0:min(MAX_WATCHED_FILES, len(files))] {
|
|
if file.Deleted {
|
|
continue
|
|
}
|
|
filename := filepath.Join(dirName, file.Name)
|
|
if w.watchingFilename(filename) {
|
|
continue
|
|
}
|
|
if len(w.WatchedFilenames) > MAX_WATCHED_FILES {
|
|
w.popOldestFilename()
|
|
}
|
|
|
|
w.watchFilename(filename)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func min(a int, b int) int {
|
|
if a < b {
|
|
return a
|
|
}
|
|
return b
|
|
}
|
|
|
|
// NOTE: given that we often edit files ourselves, this may make us end up refreshing files too often
|
|
// TODO: consider watching the whole directory recursively (could be more expensive)
|
|
func (gui *Gui) WatchFilesForChanges() {
|
|
gui.fileWatcher = NewFileWatcher(gui.Log)
|
|
if gui.fileWatcher.Disabled {
|
|
return
|
|
}
|
|
go utils.Safe(func() {
|
|
for {
|
|
select {
|
|
// watch for events
|
|
case event := <-gui.fileWatcher.Watcher.Events:
|
|
if event.Op == fsnotify.Chmod {
|
|
// for some reason we pick up chmod events when they don't actually happen
|
|
continue
|
|
}
|
|
// only refresh if we're not already
|
|
if !gui.IsRefreshingFiles {
|
|
_ = gui.c.Refresh(types.RefreshOptions{Mode: types.ASYNC, Scope: []types.RefreshableView{types.FILES}})
|
|
}
|
|
|
|
// watch for errors
|
|
case err := <-gui.fileWatcher.Watcher.Errors:
|
|
if err != nil {
|
|
gui.c.Log.Error(err)
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|