2018-08-14 11:05:26 +02:00
package gui
2018-05-26 13:23:39 +10:00
import (
2018-07-21 15:51:18 +10:00
"fmt"
2018-05-26 13:23:39 +10:00
2018-07-21 15:51:18 +10:00
"github.com/jesseduffield/gocui"
2018-08-19 07:20:50 -04:00
"github.com/jesseduffield/lazygit/pkg/utils"
2018-08-16 01:53:53 -04:00
"github.com/spkg/bom"
2018-05-26 13:23:39 +10:00
)
2018-08-14 11:05:26 +02:00
func ( gui * Gui ) resetOrigin ( v * gocui . View ) error {
2019-04-25 21:37:19 +02:00
_ = v . SetCursor ( 0 , 0 )
2018-07-21 15:51:18 +10:00
return v . SetOrigin ( 0 , 0 )
2018-06-09 19:06:33 +10:00
}
2019-01-15 20:12:31 +11:00
func ( gui * Gui ) cleanString ( s string ) string {
2018-12-12 22:34:20 +11:00
output := string ( bom . Clean ( [ ] byte ( s ) ) )
2019-01-15 20:12:31 +11:00
return utils . NormalizeLinefeeds ( output )
}
2021-04-11 10:05:19 +10:00
func ( gui * Gui ) setViewContent ( v * gocui . View , s string ) {
2022-01-15 12:04:00 +11:00
v . SetContent ( gui . cleanString ( s ) )
2021-04-11 10:05:19 +10:00
}
2019-01-15 20:12:31 +11:00
// renderString resets the origin of a view and sets its content
2022-01-15 12:04:00 +11:00
func ( gui * Gui ) renderString ( view * gocui . View , s string ) error {
2021-04-05 00:31:52 +10:00
if err := view . SetOrigin ( 0 , 0 ) ; err != nil {
2020-08-16 18:25:08 +10:00
return err
}
2021-04-05 00:31:52 +10:00
if err := view . SetCursor ( 0 , 0 ) ; err != nil {
2020-08-16 18:25:08 +10:00
return err
}
2022-01-15 12:04:00 +11:00
gui . setViewContent ( view , s )
2020-08-16 18:25:08 +10:00
return nil
}
2019-02-25 22:11:35 +11:00
func ( gui * Gui ) currentViewName ( ) string {
currentView := gui . g . CurrentView ( )
2020-05-16 12:35:19 +10:00
if currentView == nil {
return ""
}
2018-08-14 11:05:26 +02:00
return currentView . Name ( )
}
2018-09-05 19:07:46 +10:00
2020-08-15 17:23:16 +10:00
func ( gui * Gui ) resizeCurrentPopupPanel ( ) error {
v := gui . g . CurrentView ( )
2020-08-18 22:41:14 +10:00
if v == nil {
return nil
}
2019-03-11 13:04:08 +11:00
if gui . isPopupPanel ( v . Name ( ) ) {
2021-10-17 13:00:44 +11:00
return gui . resizePopupPanel ( v , v . Buffer ( ) )
2018-09-05 19:07:46 +10:00
}
return nil
}
2021-10-17 13:00:44 +11:00
func ( gui * Gui ) resizePopupPanel ( v * gocui . View , content string ) error {
2018-09-05 19:07:46 +10:00
// If the confirmation panel is already displayed, just resize the width,
// otherwise continue
2020-08-15 17:23:16 +10:00
x0 , y0 , x1 , y1 := gui . getConfirmationPanelDimensions ( v . Wrap , content )
2018-09-05 19:07:46 +10:00
vx0 , vy0 , vx1 , vy1 := v . Dimensions ( )
if vx0 == x0 && vy0 == y0 && vx1 == x1 && vy1 == y1 {
return nil
}
2020-08-15 17:23:16 +10:00
_ , err := gui . g . SetView ( v . Name ( ) , x0 , y0 , x1 , y1 , 0 )
2018-09-05 19:07:46 +10:00
return err
}
2018-12-04 19:50:11 +11:00
2020-08-23 10:50:27 +10:00
func ( gui * Gui ) globalOptionsMap ( ) map [ string ] string {
2022-01-16 14:46:53 +11:00
keybindingConfig := gui . c . UserConfig . Keybinding
2020-10-03 14:54:55 +10:00
2020-08-23 10:50:27 +10:00
return map [ string ] string {
2022-01-16 14:46:53 +11:00
fmt . Sprintf ( "%s/%s" , gui . getKeyDisplay ( keybindingConfig . Universal . ScrollUpMain ) , gui . getKeyDisplay ( keybindingConfig . Universal . ScrollDownMain ) ) : gui . c . Tr . LcScroll ,
fmt . Sprintf ( "%s %s %s %s" , gui . getKeyDisplay ( keybindingConfig . Universal . PrevBlock ) , gui . getKeyDisplay ( keybindingConfig . Universal . NextBlock ) , gui . getKeyDisplay ( keybindingConfig . Universal . PrevItem ) , gui . getKeyDisplay ( keybindingConfig . Universal . NextItem ) ) : gui . c . Tr . LcNavigate ,
gui . getKeyDisplay ( keybindingConfig . Universal . Return ) : gui . c . Tr . LcCancel ,
gui . getKeyDisplay ( keybindingConfig . Universal . Quit ) : gui . c . Tr . LcQuit ,
gui . getKeyDisplay ( keybindingConfig . Universal . OptionMenu ) : gui . c . Tr . LcMenu ,
fmt . Sprintf ( "%s-%s" , gui . getKeyDisplay ( keybindingConfig . Universal . JumpToBlock [ 0 ] ) , gui . getKeyDisplay ( keybindingConfig . Universal . JumpToBlock [ len ( keybindingConfig . Universal . JumpToBlock ) - 1 ] ) ) : gui . c . Tr . LcJump ,
fmt . Sprintf ( "%s/%s" , gui . getKeyDisplay ( keybindingConfig . Universal . ScrollLeft ) , gui . getKeyDisplay ( keybindingConfig . Universal . ScrollRight ) ) : gui . c . Tr . LcScrollLeftRight ,
2020-08-23 10:50:27 +10:00
}
2020-03-29 10:31:34 +11:00
}
2019-03-11 13:04:08 +11:00
func ( gui * Gui ) isPopupPanel ( viewName string ) bool {
2022-02-23 19:44:48 +11:00
return viewName == "commitMessage" || viewName == "confirmation" || viewName == "menu"
2019-03-11 13:04:08 +11:00
}
2019-02-25 22:11:35 +11:00
func ( gui * Gui ) popupPanelFocused ( ) bool {
2019-03-11 13:04:08 +11:00
return gui . isPopupPanel ( gui . currentViewName ( ) )
2019-02-25 22:11:35 +11:00
}
2019-11-10 16:20:35 +11:00
2020-05-18 22:00:07 +10:00
// secondaryViewFocused tells us whether it appears that the secondary view is focused. The view is actually never focused for real: we just swap the main and secondary views and then you're still focused on the main view so that we can give you access to all its keybindings for free. I will probably regret this design decision soon enough.
func ( gui * Gui ) secondaryViewFocused ( ) bool {
2020-10-08 08:01:04 +11:00
state := gui . State . Panels . LineByLine
return state != nil && state . SecondaryFocused
2020-05-18 22:00:07 +10:00
}
2020-08-17 21:58:30 +10:00
2020-08-19 19:31:58 +10:00
func ( gui * Gui ) onViewTabClick ( viewName string , tabIndex int ) error {
2022-03-26 14:33:39 +11:00
context := gui . State . ViewTabContextMap [ viewName ] [ tabIndex ] . Context
2020-08-19 19:31:58 +10:00
2022-01-16 14:46:53 +11:00
return gui . c . PushContext ( context )
2020-08-19 19:31:58 +10:00
}
2021-04-02 19:20:40 +11:00
func ( gui * Gui ) handleNextTab ( ) error {
2021-06-06 12:41:55 +02:00
v := getTabbedView ( gui )
2021-04-02 20:51:52 +11:00
if v == nil {
2021-04-02 19:20:40 +11:00
return nil
}
2020-08-19 19:31:58 +10:00
return gui . onViewTabClick (
v . Name ( ) ,
utils . ModuloWithWrap ( v . TabIndex + 1 , len ( v . Tabs ) ) ,
)
}
2021-04-02 19:20:40 +11:00
func ( gui * Gui ) handlePrevTab ( ) error {
2021-06-06 12:41:55 +02:00
v := getTabbedView ( gui )
2021-04-02 20:51:52 +11:00
if v == nil {
2021-04-02 19:20:40 +11:00
return nil
}
2020-08-19 19:31:58 +10:00
return gui . onViewTabClick (
v . Name ( ) ,
utils . ModuloWithWrap ( v . TabIndex - 1 , len ( v . Tabs ) ) ,
)
}
2020-10-02 07:56:14 +10:00
// this is the distance we will move the cursor when paging up or down in a view
func ( gui * Gui ) pageDelta ( view * gocui . View ) int {
_ , height := view . Size ( )
delta := height - 1
if delta == 0 {
return 1
}
return delta
}
2021-06-06 12:41:55 +02:00
func getTabbedView ( gui * Gui ) * gocui . View {
2021-06-15 19:58:43 +02:00
// It safe assumption that only static contexts have tabs
context := gui . currentStaticContext ( )
view , _ := gui . g . View ( context . GetViewName ( ) )
return view
2021-06-06 12:41:55 +02:00
}
2021-11-01 09:35:54 +11:00
func ( gui * Gui ) render ( ) {
2022-01-15 12:04:00 +11:00
gui . OnUIThread ( func ( ) error { return nil } )
2021-11-01 09:35:54 +11:00
}