2019-03-13 04:14:30 -07:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2021-03-20 00:49:49 -07:00
|
|
|
"strings"
|
2020-12-27 08:53:58 -08:00
|
|
|
"time"
|
2019-03-13 04:14:30 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
type engine struct {
|
2021-04-18 10:16:06 -07:00
|
|
|
config *Config
|
|
|
|
env environmentInfo
|
2021-04-20 12:30:46 -07:00
|
|
|
colorWriter colorWriter
|
|
|
|
ansi *ansiUtils
|
2021-04-18 10:16:06 -07:00
|
|
|
consoleTitle *consoleTitle
|
2021-04-20 12:30:46 -07:00
|
|
|
|
|
|
|
console strings.Builder
|
2021-04-18 10:16:06 -07:00
|
|
|
rprompt string
|
2020-10-12 23:57:46 -07:00
|
|
|
}
|
|
|
|
|
2021-04-20 12:30:46 -07:00
|
|
|
func (e *engine) write(text string) {
|
|
|
|
e.console.WriteString(text)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *engine) string() string {
|
|
|
|
return e.console.String()
|
|
|
|
}
|
|
|
|
|
2021-05-22 07:50:34 -07:00
|
|
|
func (e *engine) canWriteRPrompt() bool {
|
|
|
|
prompt := e.string()
|
|
|
|
consoleWidth, err := e.env.getTerminalWidth()
|
2021-05-25 09:31:23 -07:00
|
|
|
if err != nil || consoleWidth == 0 {
|
2021-05-22 07:50:34 -07:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
promptWidth := e.ansi.lenWithoutANSI(prompt)
|
|
|
|
availableSpace := consoleWidth - promptWidth
|
|
|
|
if promptWidth > consoleWidth {
|
|
|
|
availableSpace = promptWidth - (promptWidth % consoleWidth)
|
|
|
|
}
|
|
|
|
promptBreathingRoom := 30
|
|
|
|
return (availableSpace - e.ansi.lenWithoutANSI(e.rprompt)) >= promptBreathingRoom
|
|
|
|
}
|
|
|
|
|
2021-04-04 11:28:41 -07:00
|
|
|
func (e *engine) render() string {
|
2021-03-20 11:32:15 -07:00
|
|
|
for _, block := range e.config.Blocks {
|
2021-04-18 10:16:06 -07:00
|
|
|
e.renderBlock(block)
|
2019-03-13 04:14:30 -07:00
|
|
|
}
|
2021-03-20 11:32:15 -07:00
|
|
|
if e.config.ConsoleTitle {
|
2021-04-20 12:30:46 -07:00
|
|
|
e.write(e.consoleTitle.getConsoleTitle())
|
2020-10-12 00:02:33 -07:00
|
|
|
}
|
2021-04-20 12:30:46 -07:00
|
|
|
e.write(e.ansi.creset)
|
2021-03-20 11:32:15 -07:00
|
|
|
if e.config.FinalSpace {
|
2021-04-20 12:30:46 -07:00
|
|
|
e.write(" ")
|
2019-03-13 04:14:30 -07:00
|
|
|
}
|
2021-02-14 23:26:52 -08:00
|
|
|
|
2021-03-20 11:32:15 -07:00
|
|
|
if !e.config.OSC99 {
|
2021-04-04 11:28:41 -07:00
|
|
|
return e.print()
|
2021-02-15 13:19:19 -08:00
|
|
|
}
|
2021-02-14 23:26:52 -08:00
|
|
|
cwd := e.env.getcwd()
|
|
|
|
if e.env.isWsl() {
|
|
|
|
cwd, _ = e.env.runCommand("wslpath", "-m", cwd)
|
|
|
|
}
|
2021-04-20 12:30:46 -07:00
|
|
|
e.write(e.ansi.consolePwd(cwd))
|
2021-04-04 11:28:41 -07:00
|
|
|
return e.print()
|
2020-12-17 23:59:45 -08:00
|
|
|
}
|
|
|
|
|
2021-04-18 10:16:06 -07:00
|
|
|
func (e *engine) renderBlock(block *Block) {
|
2021-05-21 13:10:20 -07:00
|
|
|
// when in bash, for rprompt blocks we need to write plain
|
|
|
|
// and wrap in escaped mode or the prompt will not render correctly
|
|
|
|
if block.Type == RPrompt && e.env.getShellName() == bash {
|
|
|
|
block.initPlain(e.env, e.config)
|
|
|
|
} else {
|
|
|
|
block.init(e.env, e.colorWriter, e.ansi)
|
|
|
|
}
|
2021-04-18 10:16:06 -07:00
|
|
|
block.setStringValues()
|
|
|
|
if !block.enabled() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if block.Newline {
|
2021-04-20 12:30:46 -07:00
|
|
|
e.write("\n")
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
switch block.Type {
|
|
|
|
// This is deprecated but leave if to not break current configs
|
|
|
|
// It is encouraged to used "newline": true on block level
|
|
|
|
// rather than the standalone the linebreak block
|
|
|
|
case LineBreak:
|
2021-04-20 12:30:46 -07:00
|
|
|
e.write("\n")
|
2021-04-18 10:16:06 -07:00
|
|
|
case Prompt:
|
|
|
|
if block.VerticalOffset != 0 {
|
2021-04-20 12:30:46 -07:00
|
|
|
e.write(e.ansi.changeLine(block.VerticalOffset))
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
switch block.Alignment {
|
|
|
|
case Right:
|
2021-04-20 12:30:46 -07:00
|
|
|
e.write(e.ansi.carriageForward())
|
2021-04-18 10:16:06 -07:00
|
|
|
blockText := block.renderSegments()
|
2021-04-20 12:30:46 -07:00
|
|
|
e.write(e.ansi.getCursorForRightWrite(blockText, block.HorizontalOffset))
|
|
|
|
e.write(blockText)
|
2021-04-18 10:16:06 -07:00
|
|
|
case Left:
|
2021-04-20 12:30:46 -07:00
|
|
|
e.write(block.renderSegments())
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
case RPrompt:
|
2021-05-21 13:10:20 -07:00
|
|
|
blockText := block.renderSegments()
|
|
|
|
if e.env.getShellName() == bash {
|
|
|
|
blockText = fmt.Sprintf(e.ansi.bashFormat, blockText)
|
|
|
|
}
|
|
|
|
e.rprompt = blockText
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
2021-04-21 12:20:18 -07:00
|
|
|
// Due to a bug in Powershell, the end of the line needs to be cleared.
|
|
|
|
// If this doesn't happen, the portion after the prompt gets colored in the background
|
|
|
|
// color of the line above the new input line. Clearing the line fixes this,
|
|
|
|
// but can hopefully one day be removed when this is resolved natively.
|
|
|
|
if e.ansi.shell == pwsh || e.ansi.shell == powershell5 {
|
2021-06-25 11:08:00 -07:00
|
|
|
e.write(e.ansi.clearAfter())
|
2021-04-21 12:20:18 -07:00
|
|
|
}
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
|
2021-01-09 07:18:37 -08:00
|
|
|
// debug will loop through your config file and output the timings for each segments
|
2021-04-04 11:28:41 -07:00
|
|
|
func (e *engine) debug() string {
|
2021-04-18 10:16:06 -07:00
|
|
|
var segmentTimings []*SegmentTiming
|
2020-12-31 05:00:46 -08:00
|
|
|
largestSegmentNameLength := 0
|
2021-04-20 12:30:46 -07:00
|
|
|
e.write("\n\x1b[1mHere are the timings of segments in your prompt:\x1b[0m\n\n")
|
2021-01-14 21:10:36 -08:00
|
|
|
|
|
|
|
// console title timing
|
|
|
|
start := time.Now()
|
|
|
|
consoleTitle := e.consoleTitle.getTemplateText()
|
|
|
|
duration := time.Since(start)
|
2021-04-18 10:16:06 -07:00
|
|
|
segmentTiming := &SegmentTiming{
|
2021-01-14 21:10:36 -08:00
|
|
|
name: "ConsoleTitle",
|
|
|
|
nameLength: 12,
|
2021-03-20 11:32:15 -07:00
|
|
|
enabled: e.config.ConsoleTitle,
|
2021-01-14 21:10:36 -08:00
|
|
|
stringValue: consoleTitle,
|
|
|
|
enabledDuration: 0,
|
|
|
|
stringDuration: duration,
|
|
|
|
}
|
|
|
|
segmentTimings = append(segmentTimings, segmentTiming)
|
2020-12-27 08:53:58 -08:00
|
|
|
// loop each segments of each blocks
|
2021-03-20 11:32:15 -07:00
|
|
|
for _, block := range e.config.Blocks {
|
2021-04-20 12:30:46 -07:00
|
|
|
block.init(e.env, e.colorWriter, e.ansi)
|
2021-04-18 10:16:06 -07:00
|
|
|
longestSegmentName, timings := block.debug()
|
|
|
|
segmentTimings = append(segmentTimings, timings...)
|
|
|
|
if longestSegmentName > largestSegmentNameLength {
|
|
|
|
largestSegmentNameLength = longestSegmentName
|
2020-12-27 08:53:58 -08:00
|
|
|
}
|
|
|
|
}
|
2021-01-14 21:10:36 -08:00
|
|
|
|
2020-12-31 05:00:46 -08:00
|
|
|
// pad the output so the tabs render correctly
|
|
|
|
largestSegmentNameLength += 7
|
2020-12-27 08:53:58 -08:00
|
|
|
for _, segment := range segmentTimings {
|
|
|
|
duration := segment.enabledDuration.Milliseconds()
|
|
|
|
if segment.enabled {
|
|
|
|
duration += segment.stringDuration.Milliseconds()
|
|
|
|
}
|
2020-12-31 05:00:46 -08:00
|
|
|
segmentName := fmt.Sprintf("%s(%t)", segment.name, segment.enabled)
|
2021-04-20 12:30:46 -07:00
|
|
|
e.write(fmt.Sprintf("%-*s - %3d ms - %s\n", largestSegmentNameLength, segmentName, duration, segment.stringValue))
|
2020-12-27 08:53:58 -08:00
|
|
|
}
|
2021-04-20 12:30:46 -07:00
|
|
|
return e.string()
|
2020-12-27 08:53:58 -08:00
|
|
|
}
|
|
|
|
|
2021-04-04 11:28:41 -07:00
|
|
|
func (e *engine) print() string {
|
2020-12-23 04:31:21 -08:00
|
|
|
switch e.env.getShellName() {
|
|
|
|
case zsh:
|
2021-05-21 13:10:20 -07:00
|
|
|
if !*e.env.getArgs().Eval {
|
|
|
|
break
|
2020-12-23 04:31:21 -08:00
|
|
|
}
|
2021-05-21 13:10:20 -07:00
|
|
|
// escape double quotes contained in the prompt
|
|
|
|
prompt := fmt.Sprintf("PS1=\"%s\"", strings.ReplaceAll(e.string(), "\"", "\"\""))
|
|
|
|
prompt += fmt.Sprintf("\nRPROMPT=\"%s\"", e.rprompt)
|
|
|
|
return prompt
|
|
|
|
case pwsh, powershell5, bash, plain:
|
2021-05-22 07:50:34 -07:00
|
|
|
if e.rprompt == "" || !e.canWriteRPrompt() {
|
2021-05-21 13:10:20 -07:00
|
|
|
break
|
2020-12-17 23:59:45 -08:00
|
|
|
}
|
2021-05-21 13:10:20 -07:00
|
|
|
e.write(e.ansi.saveCursorPosition)
|
|
|
|
e.write(e.ansi.carriageForward())
|
|
|
|
e.write(e.ansi.getCursorForRightWrite(e.rprompt, 0))
|
|
|
|
e.write(e.rprompt)
|
|
|
|
e.write(e.ansi.restoreCursorPosition)
|
2020-12-15 05:58:15 -08:00
|
|
|
}
|
2021-04-20 12:30:46 -07:00
|
|
|
return e.string()
|
2019-03-13 04:14:30 -07:00
|
|
|
}
|
2021-06-05 07:14:44 -07:00
|
|
|
|
|
|
|
func (e *engine) renderTooltip(tip string) string {
|
|
|
|
tip = strings.Trim(tip, " ")
|
|
|
|
var tooltip *Segment
|
|
|
|
for _, tp := range e.config.Tooltips {
|
|
|
|
if !tp.shouldInvokeWithTip(tip) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
tooltip = tp
|
|
|
|
}
|
|
|
|
if tooltip == nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
if err := tooltip.mapSegmentWithWriter(e.env); err != nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
if !tooltip.enabled() {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
tooltip.stringValue = tooltip.string()
|
|
|
|
// little hack to reuse the current logic
|
|
|
|
block := &Block{
|
|
|
|
Alignment: Right,
|
|
|
|
Segments: []*Segment{tooltip},
|
|
|
|
}
|
|
|
|
switch e.env.getShellName() {
|
|
|
|
case zsh:
|
|
|
|
block.init(e.env, e.colorWriter, e.ansi)
|
|
|
|
return block.renderSegments()
|
|
|
|
case pwsh, powershell5:
|
|
|
|
block.initPlain(e.env, e.config)
|
|
|
|
tooltipText := block.renderSegments()
|
2021-06-25 11:08:00 -07:00
|
|
|
e.write(e.ansi.clearAfter())
|
2021-06-05 07:14:44 -07:00
|
|
|
e.write(e.ansi.carriageForward())
|
|
|
|
e.write(e.ansi.getCursorForRightWrite(tooltipText, 0))
|
|
|
|
e.write(tooltipText)
|
|
|
|
return e.string()
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
2021-06-15 12:23:08 -07:00
|
|
|
|
|
|
|
func (e *engine) renderTransientPrompt(command string) string {
|
2021-06-25 11:08:00 -07:00
|
|
|
newlines := strings.Count(command, ";")
|
|
|
|
if strings.HasSuffix(command, ";") {
|
|
|
|
newlines--
|
|
|
|
}
|
|
|
|
command = strings.Replace(command, ";", fmt.Sprintf(";%s", e.ansi.newLine()), newlines)
|
2021-06-15 12:23:08 -07:00
|
|
|
promptTemplate := e.config.TransientPrompt.Template
|
|
|
|
if len(promptTemplate) == 0 {
|
|
|
|
promptTemplate = "{{ .Shell }}> <#f7dc66>{{ .Command }}</>"
|
|
|
|
}
|
|
|
|
template := &textTemplate{
|
|
|
|
Template: promptTemplate,
|
|
|
|
Env: e.env,
|
|
|
|
}
|
|
|
|
context := make(map[string]interface{})
|
|
|
|
context["Command"] = command
|
|
|
|
prompt := template.renderPlainContextTemplate(context)
|
|
|
|
e.colorWriter.write(e.config.TransientPrompt.Background, e.config.TransientPrompt.Foreground, prompt)
|
|
|
|
transientPrompt := e.ansi.carriageBackward()
|
2021-06-23 13:00:40 -07:00
|
|
|
// calculate offset for multiline prompt
|
|
|
|
lineOffset := 0
|
|
|
|
for _, block := range e.config.Blocks {
|
|
|
|
if block.Newline {
|
|
|
|
lineOffset--
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if lineOffset != 0 {
|
|
|
|
transientPrompt += e.ansi.changeLine(lineOffset)
|
2021-06-15 12:23:08 -07:00
|
|
|
}
|
2021-06-25 11:08:00 -07:00
|
|
|
return transientPrompt + e.colorWriter.string() + e.ansi.clearAfter()
|
2021-06-15 12:23:08 -07:00
|
|
|
}
|