mirror of
https://github.com/jesseduffield/lazygit.git
synced 2024-12-04 10:34:55 +02:00
4fe512ff3a
type safe view access
184 lines
3.9 KiB
Go
184 lines
3.9 KiB
Go
package gui
|
|
|
|
import (
|
|
"os/exec"
|
|
|
|
"github.com/jesseduffield/gocui"
|
|
)
|
|
|
|
type viewUpdateOpts struct {
|
|
title string
|
|
|
|
// awkwardly calling this noWrap because of how hard Go makes it to have
|
|
// a boolean option that defaults to true
|
|
noWrap bool
|
|
|
|
highlight bool
|
|
|
|
task updateTask
|
|
}
|
|
|
|
type refreshMainOpts struct {
|
|
main *viewUpdateOpts
|
|
secondary *viewUpdateOpts
|
|
}
|
|
|
|
// constants for updateTask's kind field
|
|
type TaskKind int
|
|
|
|
const (
|
|
RENDER_STRING TaskKind = iota
|
|
RENDER_STRING_WITHOUT_SCROLL
|
|
RUN_FUNCTION
|
|
RUN_COMMAND
|
|
RUN_PTY
|
|
)
|
|
|
|
type updateTask interface {
|
|
GetKind() TaskKind
|
|
}
|
|
|
|
type renderStringTask struct {
|
|
str string
|
|
}
|
|
|
|
func (t *renderStringTask) GetKind() TaskKind {
|
|
return RENDER_STRING
|
|
}
|
|
|
|
func NewRenderStringTask(str string) *renderStringTask {
|
|
return &renderStringTask{str: str}
|
|
}
|
|
|
|
type renderStringWithoutScrollTask struct {
|
|
str string
|
|
}
|
|
|
|
func (t *renderStringWithoutScrollTask) GetKind() TaskKind {
|
|
return RENDER_STRING_WITHOUT_SCROLL
|
|
}
|
|
|
|
func NewRenderStringWithoutScrollTask(str string) *renderStringWithoutScrollTask {
|
|
return &renderStringWithoutScrollTask{str: str}
|
|
}
|
|
|
|
type runCommandTask struct {
|
|
cmd *exec.Cmd
|
|
prefix string
|
|
}
|
|
|
|
func (t *runCommandTask) GetKind() TaskKind {
|
|
return RUN_COMMAND
|
|
}
|
|
|
|
func NewRunCommandTask(cmd *exec.Cmd) *runCommandTask {
|
|
return &runCommandTask{cmd: cmd}
|
|
}
|
|
|
|
func NewRunCommandTaskWithPrefix(cmd *exec.Cmd, prefix string) *runCommandTask {
|
|
return &runCommandTask{cmd: cmd, prefix: prefix}
|
|
}
|
|
|
|
type runPtyTask struct {
|
|
cmd *exec.Cmd
|
|
prefix string
|
|
}
|
|
|
|
func (t *runPtyTask) GetKind() TaskKind {
|
|
return RUN_PTY
|
|
}
|
|
|
|
func NewRunPtyTask(cmd *exec.Cmd) *runPtyTask {
|
|
return &runPtyTask{cmd: cmd}
|
|
}
|
|
|
|
// currently unused
|
|
// func (gui *Gui) createRunPtyTaskWithPrefix(cmd *exec.Cmd, prefix string) *runPtyTask {
|
|
// return &runPtyTask{cmd: cmd, prefix: prefix}
|
|
// }
|
|
|
|
type runFunctionTask struct {
|
|
f func(chan struct{}) error
|
|
}
|
|
|
|
func (t *runFunctionTask) GetKind() TaskKind {
|
|
return RUN_FUNCTION
|
|
}
|
|
|
|
// currently unused
|
|
// func (gui *Gui) createRunFunctionTask(f func(chan struct{}) error) *runFunctionTask {
|
|
// return &runFunctionTask{f: f}
|
|
// }
|
|
|
|
func (gui *Gui) runTaskForView(viewName string, task updateTask) error {
|
|
gui.Log.Warn("running new task for view")
|
|
|
|
switch task.GetKind() {
|
|
case RENDER_STRING:
|
|
specificTask := task.(*renderStringTask)
|
|
return gui.newStringTask(viewName, specificTask.str)
|
|
|
|
case RENDER_STRING_WITHOUT_SCROLL:
|
|
specificTask := task.(*renderStringWithoutScrollTask)
|
|
return gui.newStringTaskWithoutScroll(viewName, specificTask.str)
|
|
|
|
case RUN_FUNCTION:
|
|
specificTask := task.(*runFunctionTask)
|
|
return gui.newTask(viewName, specificTask.f)
|
|
|
|
case RUN_COMMAND:
|
|
specificTask := task.(*runCommandTask)
|
|
return gui.newCmdTask(viewName, specificTask.cmd, specificTask.prefix)
|
|
|
|
case RUN_PTY:
|
|
specificTask := task.(*runPtyTask)
|
|
return gui.newPtyTask(viewName, specificTask.cmd, specificTask.prefix)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (gui *Gui) refreshMainView(opts *viewUpdateOpts, view *gocui.View) error {
|
|
view.Title = opts.title
|
|
view.Wrap = !opts.noWrap
|
|
view.Highlight = opts.highlight
|
|
|
|
if err := gui.runTaskForView(view.Name(), opts.task); err != nil {
|
|
gui.Log.Error(err)
|
|
return nil
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (gui *Gui) refreshMainViews(opts refreshMainOpts) error {
|
|
if opts.main != nil {
|
|
if err := gui.refreshMainView(opts.main, gui.Views.Main); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if opts.secondary != nil {
|
|
if err := gui.refreshMainView(opts.secondary, gui.Views.Secondary); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
gui.splitMainPanel(opts.secondary != nil)
|
|
|
|
return nil
|
|
}
|
|
|
|
func (gui *Gui) splitMainPanel(splitMainPanel bool) {
|
|
gui.State.SplitMainPanel = splitMainPanel
|
|
|
|
// no need to set view on bottom when splitMainPanel is false: it will have zero size anyway thanks to our view arrangement code.
|
|
if splitMainPanel {
|
|
gui.Views.Secondary.Visible = false
|
|
}
|
|
}
|
|
|
|
func (gui *Gui) isMainPanelSplit() bool {
|
|
return gui.State.SplitMainPanel
|
|
}
|