oh-my-posh/src/engine.go

153 lines
4 KiB
Go
Raw Normal View History

2019-03-13 04:14:30 -07:00
package main
import (
"fmt"
"strings"
"time"
2019-03-13 04:14:30 -07:00
)
type engine struct {
config *Config
env environmentInfo
2021-04-20 12:30:46 -07:00
colorWriter colorWriter
ansi *ansiUtils
consoleTitle *consoleTitle
2021-04-20 12:30:46 -07:00
console strings.Builder
rprompt string
}
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()
}
func (e *engine) render() string {
2021-03-20 11:32:15 -07:00
for _, block := range e.config.Blocks {
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-03-20 11:32:15 -07:00
if !e.config.OSC99 {
return e.print()
2021-02-15 13:19:19 -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))
return e.print()
2020-12-17 23:59:45 -08:00
}
func (e *engine) renderBlock(block *Block) {
2021-04-20 12:30:46 -07:00
block.init(e.env, e.colorWriter, e.ansi)
block.setStringValues()
if !block.enabled() {
return
}
if block.Newline {
2021-04-20 12:30:46 -07:00
e.write("\n")
}
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")
case Prompt:
if block.VerticalOffset != 0 {
2021-04-20 12:30:46 -07:00
e.write(e.ansi.changeLine(block.VerticalOffset))
}
switch block.Alignment {
case Right:
2021-04-20 12:30:46 -07:00
e.write(e.ansi.carriageForward())
blockText := block.renderSegments()
2021-04-20 12:30:46 -07:00
e.write(e.ansi.getCursorForRightWrite(blockText, block.HorizontalOffset))
e.write(blockText)
case Left:
2021-04-20 12:30:46 -07:00
e.write(block.renderSegments())
}
case RPrompt:
e.rprompt = block.renderSegments()
}
// 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 {
e.write(e.ansi.clearEOL)
}
}
// debug will loop through your config file and output the timings for each segments
func (e *engine) debug() string {
var segmentTimings []*SegmentTiming
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)
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)
// 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)
longestSegmentName, timings := block.debug()
segmentTimings = append(segmentTimings, timings...)
if longestSegmentName > largestSegmentNameLength {
largestSegmentNameLength = longestSegmentName
}
}
2021-01-14 21:10:36 -08:00
// pad the output so the tabs render correctly
largestSegmentNameLength += 7
for _, segment := range segmentTimings {
duration := segment.enabledDuration.Milliseconds()
if segment.enabled {
duration += segment.stringDuration.Milliseconds()
}
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))
}
2021-04-20 12:30:46 -07:00
return e.string()
}
func (e *engine) print() string {
2020-12-23 04:31:21 -08:00
switch e.env.getShellName() {
case zsh:
if *e.env.getArgs().Eval {
// escape double quotes contained in the prompt
2021-04-20 12:30:46 -07:00
prompt := fmt.Sprintf("PS1=\"%s\"", strings.ReplaceAll(e.string(), "\"", "\"\""))
prompt += fmt.Sprintf("\nRPROMPT=\"%s\"", e.rprompt)
return prompt
2020-12-23 04:31:21 -08:00
}
case pwsh, powershell5, bash, shelly:
2020-12-23 04:31:21 -08:00
if e.rprompt != "" {
2021-04-20 12:30:46 -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-17 23:59:45 -08:00
}
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
}