oh-my-posh/src/engine.go

273 lines
7.5 KiB
Go
Raw Normal View History

2019-03-13 04:14:30 -07:00
package main
import (
"fmt"
"strings"
"sync"
"time"
2019-03-13 04:14:30 -07:00
)
type engine struct {
2021-03-20 11:32:15 -07:00
config *Config
2019-03-13 04:14:30 -07:00
env environmentInfo
2020-12-17 23:59:45 -08:00
color *AnsiColor
renderer *AnsiRenderer
2020-12-26 10:51:21 -08:00
consoleTitle *consoleTitle
2019-03-13 04:14:30 -07:00
activeBlock *Block
activeSegment *Segment
previousActiveSegment *Segment
2020-12-17 23:59:45 -08:00
rprompt string
2019-03-13 04:14:30 -07:00
}
2021-01-05 03:28:33 -08:00
// SegmentTiming holds the timing context for a segment
type SegmentTiming struct {
name string
nameLength int
enabled bool
stringValue string
enabledDuration time.Duration
stringDuration time.Duration
}
2019-03-13 04:14:30 -07:00
func (e *engine) getPowerlineColor(foreground bool) string {
if e.previousActiveSegment == nil {
2020-09-21 06:44:09 -07:00
return Transparent
2019-03-13 04:14:30 -07:00
}
if !foreground && e.activeSegment.Style != Powerline {
2020-09-21 06:44:09 -07:00
return Transparent
2019-03-13 04:14:30 -07:00
}
if foreground && e.previousActiveSegment.Style != Powerline {
2020-09-21 06:44:09 -07:00
return Transparent
2019-03-13 04:14:30 -07:00
}
return e.previousActiveSegment.background()
2019-03-13 04:14:30 -07:00
}
func (e *engine) writePowerLineSeparator(background, foreground string, end bool) {
symbol := e.activeSegment.PowerlineSymbol
if end {
symbol = e.previousActiveSegment.PowerlineSymbol
}
2020-09-27 00:37:50 -07:00
if e.activeSegment.InvertPowerline {
2020-12-17 23:59:45 -08:00
e.color.write(foreground, background, symbol)
2019-03-13 04:14:30 -07:00
return
}
2020-12-17 23:59:45 -08:00
e.color.write(background, foreground, symbol)
2019-03-13 04:14:30 -07:00
}
func (e *engine) endPowerline() {
if e.activeSegment != nil &&
e.activeSegment.Style != Powerline &&
e.previousActiveSegment != nil &&
e.previousActiveSegment.Style == Powerline {
e.writePowerLineSeparator(e.getPowerlineColor(false), e.previousActiveSegment.background(), true)
2019-03-13 04:14:30 -07:00
}
}
func (e *engine) renderPowerLineSegment(text string) {
e.writePowerLineSeparator(e.activeSegment.background(), e.getPowerlineColor(true), false)
2019-03-13 04:14:30 -07:00
e.renderText(text)
}
func (e *engine) renderPlainSegment(text string) {
e.renderText(text)
}
func (e *engine) renderDiamondSegment(text string) {
e.color.write(Transparent, e.activeSegment.background(), e.activeSegment.LeadingDiamond)
2019-03-13 04:14:30 -07:00
e.renderText(text)
e.color.write(Transparent, e.activeSegment.background(), e.activeSegment.TrailingDiamond)
2019-03-13 04:14:30 -07:00
}
func (e *engine) renderText(text string) {
defaultValue := " "
if e.activeSegment.background() != "" {
defaultValue = fmt.Sprintf("<%s>\u2588</>", e.activeSegment.background())
}
text = e.color.formats.generateHyperlink(text)
2020-12-02 00:20:58 -08:00
prefix := e.activeSegment.getValue(Prefix, defaultValue)
postfix := e.activeSegment.getValue(Postfix, defaultValue)
e.color.write(e.activeSegment.background(), e.activeSegment.foreground(), fmt.Sprintf("%s%s%s", prefix, text, postfix))
2019-03-13 04:14:30 -07:00
}
func (e *engine) renderSegmentText(text string) {
switch e.activeSegment.Style {
case Plain:
e.renderPlainSegment(text)
case Diamond:
e.renderDiamondSegment(text)
case Powerline:
2019-03-13 04:14:30 -07:00
e.renderPowerLineSegment(text)
}
e.previousActiveSegment = e.activeSegment
}
func (e *engine) renderBlockSegments(block *Block) string {
2020-12-17 23:59:45 -08:00
defer e.resetBlock()
2019-03-13 04:14:30 -07:00
e.activeBlock = block
e.setStringValues(block.Segments)
2019-03-13 04:14:30 -07:00
for _, segment := range block.Segments {
if !segment.active {
2019-03-13 04:14:30 -07:00
continue
}
e.activeSegment = segment
e.endPowerline()
e.renderSegmentText(segment.stringValue)
2019-03-13 04:14:30 -07:00
}
if e.previousActiveSegment != nil && e.previousActiveSegment.Style == Powerline {
e.writePowerLineSeparator(Transparent, e.previousActiveSegment.background(), true)
2019-03-13 04:14:30 -07:00
}
2020-12-17 23:59:45 -08:00
return e.color.string()
2019-03-13 04:14:30 -07:00
}
func (e *engine) setStringValues(segments []*Segment) {
wg := sync.WaitGroup{}
wg.Add(len(segments))
defer wg.Wait()
cwd := e.env.getcwd()
for _, segment := range segments {
go func(s *Segment) {
defer wg.Done()
s.setStringValue(e.env, cwd)
}(segment)
}
}
func (e *engine) render() {
2021-03-20 11:32:15 -07:00
for _, block := range e.config.Blocks {
2019-03-13 04:14:30 -07:00
// if line break, append a line break
2020-12-17 23:59:45 -08:00
switch block.Type {
case LineBreak:
2021-02-14 05:09:43 -08:00
e.renderer.write("\n")
2020-12-17 23:59:45 -08:00
case Prompt:
if block.VerticalOffset != 0 {
e.renderer.changeLine(block.VerticalOffset)
}
switch block.Alignment {
case Right:
e.renderer.carriageForward()
blockText := e.renderBlockSegments(block)
e.renderer.setCursorForRightWrite(blockText, block.HorizontalOffset)
2021-02-14 05:09:43 -08:00
e.renderer.write(blockText)
2020-12-17 23:59:45 -08:00
case Left:
2021-02-14 05:09:43 -08:00
e.renderer.write(e.renderBlockSegments(block))
2020-12-17 23:59:45 -08:00
}
case RPrompt:
e.rprompt = e.renderBlockSegments(block)
2019-03-13 04:14:30 -07:00
}
}
2021-03-20 11:32:15 -07:00
if e.config.ConsoleTitle {
2021-02-14 05:09:43 -08:00
e.renderer.write(e.consoleTitle.getConsoleTitle())
2020-10-12 00:02:33 -07:00
}
e.renderer.creset()
2021-03-20 11:32:15 -07:00
if e.config.FinalSpace {
2021-02-14 05:09:43 -08:00
e.renderer.write(" ")
2019-03-13 04:14:30 -07:00
}
2021-03-20 11:32:15 -07:00
if !e.config.OSC99 {
2021-02-15 13:19:19 -08:00
e.print()
return
}
cwd := e.env.getcwd()
if e.env.isWsl() {
cwd, _ = e.env.runCommand("wslpath", "-m", cwd)
}
e.renderer.osc99(cwd)
2021-02-14 05:09:43 -08:00
e.print()
2020-12-17 23:59:45 -08:00
}
// debug will loop through your config file and output the timings for each segments
func (e *engine) debug() {
var segmentTimings []SegmentTiming
largestSegmentNameLength := 0
2021-02-14 05:09:43 -08:00
e.renderer.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{
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 {
for _, segment := range block.Segments {
err := segment.mapSegmentWithWriter(e.env)
2021-02-27 20:05:51 -08:00
if err != nil || !segment.shouldIncludeFolder(e.env.getcwd()) {
continue
}
var segmentTiming SegmentTiming
segmentTiming.name = string(segment.Type)
segmentTiming.nameLength = len(segmentTiming.name)
if segmentTiming.nameLength > largestSegmentNameLength {
largestSegmentNameLength = segmentTiming.nameLength
}
// enabled() timing
start := time.Now()
segmentTiming.enabled = segment.enabled()
segmentTiming.enabledDuration = time.Since(start)
// string() timing
if segmentTiming.enabled {
start = time.Now()
segmentTiming.stringValue = segment.string()
segmentTiming.stringDuration = time.Since(start)
e.previousActiveSegment = nil
e.activeSegment = segment
e.renderSegmentText(segmentTiming.stringValue)
if e.activeSegment.Style == Powerline {
e.writePowerLineSeparator(Transparent, e.activeSegment.background(), true)
}
segmentTiming.stringValue = e.color.string()
e.color.builder.Reset()
}
segmentTimings = append(segmentTimings, segmentTiming)
}
}
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-02-14 05:09:43 -08:00
e.renderer.write(fmt.Sprintf("%-*s - %3d ms - %s\n", largestSegmentNameLength, segmentName, duration, segment.stringValue))
}
fmt.Print(e.renderer.string())
}
2021-02-14 05:09:43 -08:00
func (e *engine) print() {
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
fmt.Printf("PS1=\"%s\"", strings.ReplaceAll(e.renderer.string(), "\"", "\"\""))
2020-12-17 23:59:45 -08:00
fmt.Printf("\nRPROMPT=\"%s\"", e.rprompt)
2020-12-23 04:31:21 -08:00
return
}
case pwsh, powershell5, bash:
if e.rprompt != "" {
e.renderer.saveCursorPosition()
e.renderer.carriageForward()
e.renderer.setCursorForRightWrite(e.rprompt, 0)
2021-02-14 05:09:43 -08:00
e.renderer.write(e.rprompt)
2020-12-23 04:31:21 -08:00
e.renderer.restoreCursorPosition()
2020-12-17 23:59:45 -08:00
}
2020-12-15 05:58:15 -08:00
}
2020-12-17 23:59:45 -08:00
fmt.Print(e.renderer.string())
2019-03-13 04:14:30 -07:00
}
2020-12-17 23:59:45 -08:00
func (e *engine) resetBlock() {
e.color.reset()
2019-03-13 04:14:30 -07:00
e.previousActiveSegment = nil
e.activeBlock = nil
}