mirror of
https://github.com/Maks1mS/bubbles.git
synced 2025-01-11 14:38:10 +03:00
Add vertical layout
This commit is contained in:
parent
9b47f26bdd
commit
2e2a7eccb7
118
layouts/vertical.go
Normal file
118
layouts/vertical.go
Normal file
@ -0,0 +1,118 @@
|
|||||||
|
package layouts
|
||||||
|
|
||||||
|
import (
|
||||||
|
tea "github.com/charmbracelet/bubbletea"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Model is the Bubble Tea model for a vertical layout element.
|
||||||
|
type Model struct {
|
||||||
|
Index int
|
||||||
|
Items []tea.Model
|
||||||
|
|
||||||
|
// Focus indicates whether user focus should be on this component
|
||||||
|
focus bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type FocusItem interface {
|
||||||
|
Focus() tea.Model
|
||||||
|
Blur() tea.Model
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewModel creates a new model with default settings.
|
||||||
|
func NewModel() Model {
|
||||||
|
return Model{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update is the Tea update loop.
|
||||||
|
func (m Model) Update(msg tea.Msg) (Model, tea.Cmd) {
|
||||||
|
if !m.focus {
|
||||||
|
return m, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
switch msg := msg.(type) {
|
||||||
|
case tea.KeyMsg:
|
||||||
|
switch msg.String() {
|
||||||
|
case "shift+tab", "up":
|
||||||
|
m.Index--
|
||||||
|
if m.Index < 0 {
|
||||||
|
m.Index = len(m.Items) - 1
|
||||||
|
}
|
||||||
|
m.updateFocus()
|
||||||
|
|
||||||
|
case "tab", "down":
|
||||||
|
m.Index++
|
||||||
|
if m.Index >= len(m.Items) {
|
||||||
|
m.Index = 0
|
||||||
|
}
|
||||||
|
m.updateFocus()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cmd := m.updateItems(msg)
|
||||||
|
return m, cmd
|
||||||
|
}
|
||||||
|
|
||||||
|
// View renders the layout in its current state.
|
||||||
|
func (m Model) View() string {
|
||||||
|
var view string
|
||||||
|
|
||||||
|
for _, v := range m.Items {
|
||||||
|
if mi, ok := v.(tea.Model); ok {
|
||||||
|
view += mi.View() + "\n"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return view
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Model) updateFocus() {
|
||||||
|
for i, v := range m.Items {
|
||||||
|
if m.Index == i {
|
||||||
|
if fi, ok := v.(FocusItem); ok {
|
||||||
|
// new focused item
|
||||||
|
m.Items[i] = fi.Focus()
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if fi, ok := v.(FocusItem); ok {
|
||||||
|
m.Items[i] = fi.Blur()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pass messages and models through to text input components. Only text inputs
|
||||||
|
// with Focus() set will respond, so it's safe to simply update all of them
|
||||||
|
// here without any further logic.
|
||||||
|
func (m *Model) updateItems(msg tea.Msg) tea.Cmd {
|
||||||
|
var (
|
||||||
|
cmd tea.Cmd
|
||||||
|
cmds []tea.Cmd
|
||||||
|
)
|
||||||
|
|
||||||
|
for i, v := range m.Items {
|
||||||
|
if mi, ok := v.(tea.Model); ok {
|
||||||
|
m.Items[i], cmd = mi.Update(msg)
|
||||||
|
if cmd != nil {
|
||||||
|
cmds = append(cmds, cmd)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return tea.Batch(cmds...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Focused returns the focus state on the model.
|
||||||
|
func (m Model) Focused() bool {
|
||||||
|
return m.focus
|
||||||
|
}
|
||||||
|
|
||||||
|
// Focus sets the focus state on the model.
|
||||||
|
func (m *Model) Focus() {
|
||||||
|
m.focus = true
|
||||||
|
m.updateFocus()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Blur removes the focus state on the model.
|
||||||
|
func (m *Model) Blur() {
|
||||||
|
m.focus = false
|
||||||
|
}
|
@ -67,6 +67,7 @@ type Model struct {
|
|||||||
Cursor string
|
Cursor string
|
||||||
BlinkSpeed time.Duration
|
BlinkSpeed time.Duration
|
||||||
TextColor string
|
TextColor string
|
||||||
|
FocusedTextColor string
|
||||||
BackgroundColor string
|
BackgroundColor string
|
||||||
PlaceholderColor string
|
PlaceholderColor string
|
||||||
CursorColor string
|
CursorColor string
|
||||||
@ -114,6 +115,7 @@ func NewModel() Model {
|
|||||||
Placeholder: "",
|
Placeholder: "",
|
||||||
BlinkSpeed: defaultBlinkSpeed,
|
BlinkSpeed: defaultBlinkSpeed,
|
||||||
TextColor: "",
|
TextColor: "",
|
||||||
|
FocusedTextColor: "205",
|
||||||
PlaceholderColor: "240",
|
PlaceholderColor: "240",
|
||||||
CursorColor: "",
|
CursorColor: "",
|
||||||
EchoCharacter: '*',
|
EchoCharacter: '*',
|
||||||
@ -182,15 +184,19 @@ func (m Model) Focused() bool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Focus sets the focus state on the model.
|
// Focus sets the focus state on the model.
|
||||||
func (m *Model) Focus() {
|
func (m Model) Focus() tea.Model {
|
||||||
m.focus = true
|
m.focus = true
|
||||||
m.blink = m.cursorMode == cursorHide // show the cursor unless we've explicitly hidden it
|
m.blink = m.cursorMode == cursorHide // show the cursor unless we've explicitly hidden it
|
||||||
|
|
||||||
|
return m
|
||||||
}
|
}
|
||||||
|
|
||||||
// Blur removes the focus state on the model.
|
// Blur removes the focus state on the model.
|
||||||
func (m *Model) Blur() {
|
func (m Model) Blur() tea.Model {
|
||||||
m.focus = false
|
m.focus = false
|
||||||
m.blink = true
|
m.blink = true
|
||||||
|
|
||||||
|
return m
|
||||||
}
|
}
|
||||||
|
|
||||||
// Reset sets the input to its default state with no input. Returns whether
|
// Reset sets the input to its default state with no input. Returns whether
|
||||||
@ -450,8 +456,12 @@ func (m Model) echoTransform(v string) string {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (m Model) Init() tea.Cmd {
|
||||||
|
return Blink
|
||||||
|
}
|
||||||
|
|
||||||
// Update is the Bubble Tea update loop.
|
// Update is the Bubble Tea update loop.
|
||||||
func (m Model) Update(msg tea.Msg) (Model, tea.Cmd) {
|
func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
||||||
if !m.focus {
|
if !m.focus {
|
||||||
m.blink = true
|
m.blink = true
|
||||||
return m, nil
|
return m, nil
|
||||||
@ -560,9 +570,14 @@ func (m Model) Update(msg tea.Msg) (Model, tea.Cmd) {
|
|||||||
|
|
||||||
// View renders the textinput in its current state.
|
// View renders the textinput in its current state.
|
||||||
func (m Model) View() string {
|
func (m Model) View() string {
|
||||||
|
prompt := termenv.String(m.Prompt)
|
||||||
|
if m.focus {
|
||||||
|
prompt = prompt.Foreground(color(m.FocusedTextColor))
|
||||||
|
}
|
||||||
|
|
||||||
// Placeholder text
|
// Placeholder text
|
||||||
if len(m.value) == 0 && m.Placeholder != "" {
|
if len(m.value) == 0 && m.Placeholder != "" {
|
||||||
return m.placeholderView()
|
return prompt.String() + m.placeholderView()
|
||||||
}
|
}
|
||||||
|
|
||||||
value := m.value[m.offset:m.offsetRight]
|
value := m.value[m.offset:m.offsetRight]
|
||||||
@ -590,7 +605,7 @@ func (m Model) View() string {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
return m.Prompt + v
|
return prompt.String() + v
|
||||||
}
|
}
|
||||||
|
|
||||||
// placeholderView returns the prompt and placeholder view, if any.
|
// placeholderView returns the prompt and placeholder view, if any.
|
||||||
@ -610,7 +625,7 @@ func (m Model) placeholderView() string {
|
|||||||
// The rest of the placeholder text
|
// The rest of the placeholder text
|
||||||
v += m.colorPlaceholder(p[1:])
|
v += m.colorPlaceholder(p[1:])
|
||||||
|
|
||||||
return m.Prompt + v
|
return v
|
||||||
}
|
}
|
||||||
|
|
||||||
// cursorView styles the cursor.
|
// cursorView styles the cursor.
|
||||||
|
Loading…
Reference in New Issue
Block a user