File: write.go

package info (click to toggle)
golang-github-cqroot-prompt 0.9.4-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 356 kB
  • sloc: makefile: 17; sh: 12
file content (138 lines) | stat: -rw-r--r-- 2,659 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
package write

import (
	"fmt"
	"strings"

	"github.com/charmbracelet/bubbles/help"
	"github.com/charmbracelet/bubbles/key"
	"github.com/charmbracelet/bubbles/textarea"
	tea "github.com/charmbracelet/bubbletea"
	"github.com/cqroot/prompt/constants"
)

type Model struct {
	textarea     textarea.Model
	validateFunc ValidateFunc

	quitting       bool
	err            error
	keys           KeyMap
	showHelp       bool
	help           help.Model
	teaProgramOpts []tea.ProgramOption
}

func New(defaultValue string, opts ...Option) *Model {
	ta := textarea.New()
	ta.Placeholder = defaultValue
	ta.ShowLineNumbers = false
	ta.Focus()

	m := &Model{
		textarea:       ta,
		quitting:       false,
		err:            nil,
		keys:           DefaultKeyMap,
		showHelp:       false,
		help:           help.New(),
		teaProgramOpts: make([]tea.ProgramOption, 0),
	}

	for _, opt := range opts {
		opt(m)
	}

	return m
}

func (m Model) Data() string {
	if m.textarea.Value() == "" {
		return m.textarea.Placeholder
	} else {
		return m.textarea.Value()
	}
}

func (m Model) DataString() string {
	data := m.Data()
	if strings.Contains(data, "\n") {
		return fmt.Sprintf("...(%d bytes)", len(m.Data()))
	} else {
		return data
	}
}

func (m Model) Quitting() bool {
	return m.quitting
}

func (m Model) Error() error {
	return m.err
}

func (m Model) TeaProgramOpts() []tea.ProgramOption {
	return m.teaProgramOpts
}

func (m Model) KeyBindings() []key.Binding {
	return nil
}

func (m *Model) WithValidateFunc(vf ValidateFunc) *Model {
	m.validateFunc = vf
	return m
}

func (m Model) Init() tea.Cmd {
	return textarea.Blink
}

func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
	switch msg := msg.(type) {
	case tea.WindowSizeMsg:
		m.help.Width = msg.Width

	case tea.KeyMsg:
		switch {
		case key.Matches(msg, m.keys.Confirm):
			if m.err == nil && m.validateFunc != nil {
				currVal := m.textarea.Value()
				if currVal == "" {
					currVal = m.textarea.Placeholder
				}
				m.err = m.validateFunc(currVal)
			}
			m.quitting = true
			return m, tea.Quit

		case key.Matches(msg, m.keys.Quit):
			m.quitting = true
			m.err = constants.ErrUserQuit
			return m, tea.Quit
		}
	}

	var cmd tea.Cmd

	m.textarea, cmd = m.textarea.Update(msg)
	return m, cmd
}

func (m Model) View() string {
	view := "\n" + m.textarea.View()

	if m.textarea.Value() != "" && m.validateFunc != nil {
		err := m.validateFunc(m.textarea.Value())
		if err != nil {
			view = view + constants.DefaultErrorPromptPrefixStyle.Render("\n✖  ") +
				constants.DefaultNoteStyle.Render(err.Error())
		}
	}

	if m.showHelp {
		view += "\n\n"
		view += m.help.View(m.keys)
	}
	return view
}