2021-04-18 10:16:06 -07:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
// BlockType type of block
|
|
|
|
type BlockType string
|
|
|
|
|
|
|
|
// BlockAlignment aligment of a Block
|
|
|
|
type BlockAlignment string
|
|
|
|
|
|
|
|
const (
|
|
|
|
// Prompt writes one or more Segments
|
|
|
|
Prompt BlockType = "prompt"
|
|
|
|
// LineBreak creates a line break in the prompt
|
|
|
|
LineBreak BlockType = "newline"
|
|
|
|
// RPrompt a right aligned prompt in ZSH and Powershell
|
|
|
|
RPrompt BlockType = "rprompt"
|
|
|
|
// Left aligns left
|
|
|
|
Left BlockAlignment = "left"
|
|
|
|
// Right aligns right
|
|
|
|
Right BlockAlignment = "right"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Block defines a part of the prompt with optional segments
|
|
|
|
type Block struct {
|
|
|
|
Type BlockType `config:"type"`
|
|
|
|
Alignment BlockAlignment `config:"alignment"`
|
|
|
|
HorizontalOffset int `config:"horizontal_offset"`
|
|
|
|
VerticalOffset int `config:"vertical_offset"`
|
|
|
|
Segments []*Segment `config:"segments"`
|
|
|
|
Newline bool `config:"newline"`
|
|
|
|
|
|
|
|
env environmentInfo
|
2021-11-02 06:39:51 -07:00
|
|
|
writer promptWriter
|
2021-04-20 12:30:46 -07:00
|
|
|
ansi *ansiUtils
|
2021-04-18 10:16:06 -07:00
|
|
|
activeSegment *Segment
|
|
|
|
previousActiveSegment *Segment
|
2021-10-25 10:22:58 -07:00
|
|
|
activeBackground string
|
|
|
|
activeForeground string
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
|
2021-11-02 06:39:51 -07:00
|
|
|
func (b *Block) init(env environmentInfo, writer promptWriter, ansi *ansiUtils) {
|
2021-04-18 10:16:06 -07:00
|
|
|
b.env = env
|
2021-04-20 12:30:46 -07:00
|
|
|
b.writer = writer
|
|
|
|
b.ansi = ansi
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
|
2021-05-21 13:10:20 -07:00
|
|
|
func (b *Block) initPlain(env environmentInfo, config *Config) {
|
|
|
|
b.ansi = &ansiUtils{}
|
|
|
|
b.ansi.init(plain)
|
2021-11-02 06:39:51 -07:00
|
|
|
b.writer = &AnsiWriter{
|
2021-05-21 13:10:20 -07:00
|
|
|
ansi: b.ansi,
|
|
|
|
terminalBackground: getConsoleBackgroundColor(env, config.TerminalBackground),
|
2021-11-22 13:05:15 -08:00
|
|
|
ansiColors: MakeColors(env, config),
|
2021-05-21 13:10:20 -07:00
|
|
|
}
|
|
|
|
b.env = env
|
|
|
|
}
|
|
|
|
|
2021-11-10 11:03:42 -08:00
|
|
|
func (b *Block) setActiveSegment(segment *Segment) {
|
|
|
|
b.activeSegment = segment
|
|
|
|
b.activeBackground = segment.background()
|
|
|
|
b.activeForeground = segment.foreground()
|
|
|
|
}
|
|
|
|
|
2021-04-18 10:16:06 -07:00
|
|
|
func (b *Block) enabled() bool {
|
|
|
|
if b.Type == LineBreak {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
for _, segment := range b.Segments {
|
|
|
|
if segment.active {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Block) setStringValues() {
|
|
|
|
wg := sync.WaitGroup{}
|
|
|
|
wg.Add(len(b.Segments))
|
|
|
|
defer wg.Wait()
|
|
|
|
for _, segment := range b.Segments {
|
|
|
|
go func(s *Segment) {
|
|
|
|
defer wg.Done()
|
2021-10-20 08:01:19 -07:00
|
|
|
s.setStringValue(b.env)
|
2021-04-18 10:16:06 -07:00
|
|
|
}(segment)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Block) renderSegments() string {
|
2021-04-20 12:30:46 -07:00
|
|
|
defer b.writer.reset()
|
2021-04-18 10:16:06 -07:00
|
|
|
for _, segment := range b.Segments {
|
|
|
|
if !segment.active {
|
|
|
|
continue
|
|
|
|
}
|
2021-12-16 01:57:28 -08:00
|
|
|
b.renderSegment(segment)
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
2021-12-16 01:57:28 -08:00
|
|
|
b.writePowerline(true)
|
2021-10-30 02:39:08 -07:00
|
|
|
b.writer.clearParentColors()
|
2021-04-20 12:30:46 -07:00
|
|
|
return b.writer.string()
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
|
2021-12-16 01:57:28 -08:00
|
|
|
func (b *Block) writePowerline(end bool) {
|
|
|
|
resolvePowerlineSymbol := func() string {
|
|
|
|
var symbol string
|
|
|
|
if b.previousActiveSegment != nil && b.previousActiveSegment.Style == Powerline {
|
|
|
|
symbol = b.previousActiveSegment.PowerlineSymbol
|
|
|
|
} else if b.activeSegment.Style == Powerline {
|
|
|
|
symbol = b.activeSegment.PowerlineSymbol
|
|
|
|
}
|
|
|
|
return symbol
|
|
|
|
}
|
|
|
|
symbol := resolvePowerlineSymbol()
|
|
|
|
if len(symbol) == 0 {
|
2021-05-03 22:40:27 -07:00
|
|
|
return
|
|
|
|
}
|
2021-12-16 01:57:28 -08:00
|
|
|
background := b.activeSegment.background()
|
|
|
|
if end || b.activeSegment.Style != Powerline {
|
|
|
|
background = Transparent
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
2021-12-16 01:57:28 -08:00
|
|
|
if b.activeSegment.Style == Diamond && len(b.activeSegment.LeadingDiamond) == 0 {
|
|
|
|
background = b.activeSegment.background()
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
if b.activeSegment.InvertPowerline {
|
2021-12-16 01:57:28 -08:00
|
|
|
b.writer.write(b.getPowerlineColor(), background, symbol)
|
2021-04-18 10:16:06 -07:00
|
|
|
return
|
|
|
|
}
|
2021-12-16 01:57:28 -08:00
|
|
|
b.writer.write(background, b.getPowerlineColor(), symbol)
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
|
2021-12-16 01:57:28 -08:00
|
|
|
func (b *Block) getPowerlineColor() string {
|
2021-04-18 10:16:06 -07:00
|
|
|
if b.previousActiveSegment == nil {
|
|
|
|
return Transparent
|
|
|
|
}
|
2021-05-03 22:40:27 -07:00
|
|
|
if b.previousActiveSegment.Style == Diamond && len(b.previousActiveSegment.TrailingDiamond) == 0 {
|
|
|
|
return b.previousActiveSegment.background()
|
|
|
|
}
|
|
|
|
if b.activeSegment.Style == Diamond && len(b.activeSegment.LeadingDiamond) == 0 {
|
2021-12-16 01:57:28 -08:00
|
|
|
return b.previousActiveSegment.background()
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
2021-12-16 01:57:28 -08:00
|
|
|
if b.previousActiveSegment.Style != Powerline {
|
2021-04-18 10:16:06 -07:00
|
|
|
return Transparent
|
|
|
|
}
|
|
|
|
return b.previousActiveSegment.background()
|
|
|
|
}
|
|
|
|
|
2021-12-16 01:57:28 -08:00
|
|
|
func (b *Block) renderSegment(segment *Segment) {
|
|
|
|
b.setActiveSegment(segment)
|
|
|
|
b.writePowerline(false)
|
|
|
|
b.writer.setColors(b.activeBackground, b.activeForeground)
|
2021-04-18 10:16:06 -07:00
|
|
|
switch b.activeSegment.Style {
|
2021-12-16 01:57:28 -08:00
|
|
|
case Plain, Powerline:
|
|
|
|
b.renderText(segment.stringValue)
|
2021-04-18 10:16:06 -07:00
|
|
|
case Diamond:
|
2021-12-16 01:57:28 -08:00
|
|
|
b.writer.write(Transparent, b.activeBackground, b.activeSegment.LeadingDiamond)
|
|
|
|
b.renderText(segment.stringValue)
|
|
|
|
b.writer.write(Transparent, b.activeBackground, b.activeSegment.TrailingDiamond)
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
b.previousActiveSegment = b.activeSegment
|
2021-10-25 10:22:58 -07:00
|
|
|
b.writer.setParentColors(b.activeBackground, b.activeForeground)
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Block) renderText(text string) {
|
|
|
|
defaultValue := " "
|
2021-10-25 10:22:58 -07:00
|
|
|
b.writer.write(b.activeBackground, b.activeForeground, b.activeSegment.getValue(Prefix, defaultValue))
|
|
|
|
b.writer.write(b.activeBackground, b.activeForeground, text)
|
|
|
|
b.writer.write(b.activeBackground, b.activeForeground, b.activeSegment.getValue(Postfix, defaultValue))
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Block) debug() (int, []*SegmentTiming) {
|
|
|
|
var segmentTimings []*SegmentTiming
|
|
|
|
largestSegmentNameLength := 0
|
|
|
|
for _, segment := range b.Segments {
|
|
|
|
err := segment.mapSegmentWithWriter(b.env)
|
2021-10-20 08:01:19 -07:00
|
|
|
if err != nil || !segment.shouldIncludeFolder() {
|
2021-04-18 10:16:06 -07:00
|
|
|
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()
|
2021-12-16 01:57:28 -08:00
|
|
|
segment.stringValue = segment.string()
|
2021-04-18 10:16:06 -07:00
|
|
|
segmentTiming.stringDuration = time.Since(start)
|
|
|
|
b.previousActiveSegment = nil
|
2021-12-16 01:57:28 -08:00
|
|
|
b.renderSegment(segment)
|
2021-04-18 10:16:06 -07:00
|
|
|
if b.activeSegment.Style == Powerline {
|
2021-12-16 01:57:28 -08:00
|
|
|
b.writePowerline(false)
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
2021-04-20 12:30:46 -07:00
|
|
|
segmentTiming.stringValue = b.writer.string()
|
|
|
|
b.writer.reset()
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
segmentTimings = append(segmentTimings, &segmentTiming)
|
|
|
|
}
|
|
|
|
return largestSegmentNameLength, segmentTimings
|
|
|
|
}
|