2021-04-18 10:16:06 -07:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"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-04-20 12:30:46 -07:00
|
|
|
writer colorWriter
|
|
|
|
ansi *ansiUtils
|
2021-04-18 10:16:06 -07:00
|
|
|
activeSegment *Segment
|
|
|
|
previousActiveSegment *Segment
|
|
|
|
}
|
|
|
|
|
2021-04-20 12:30:46 -07:00
|
|
|
func (b *Block) init(env environmentInfo, writer colorWriter, 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)
|
|
|
|
b.writer = &AnsiColor{
|
|
|
|
ansi: b.ansi,
|
|
|
|
terminalBackground: getConsoleBackgroundColor(env, config.TerminalBackground),
|
|
|
|
}
|
|
|
|
b.env = env
|
|
|
|
}
|
|
|
|
|
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()
|
|
|
|
cwd := b.env.getcwd()
|
|
|
|
for _, segment := range b.Segments {
|
|
|
|
go func(s *Segment) {
|
|
|
|
defer wg.Done()
|
|
|
|
s.setStringValue(b.env, cwd)
|
|
|
|
}(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
|
|
|
|
}
|
|
|
|
b.activeSegment = segment
|
|
|
|
b.endPowerline()
|
|
|
|
b.renderSegmentText(segment.stringValue)
|
|
|
|
}
|
|
|
|
if b.previousActiveSegment != nil && b.previousActiveSegment.Style == Powerline {
|
|
|
|
b.writePowerLineSeparator(Transparent, b.previousActiveSegment.background(), true)
|
|
|
|
}
|
2021-04-20 12:30:46 -07:00
|
|
|
return b.writer.string()
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Block) endPowerline() {
|
2021-05-03 22:40:27 -07:00
|
|
|
if b.previousActiveSegment == nil || b.activeSegment == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if b.activeSegment.Style != Powerline &&
|
2021-04-18 10:16:06 -07:00
|
|
|
b.previousActiveSegment.Style == Powerline {
|
|
|
|
b.writePowerLineSeparator(b.getPowerlineColor(false), b.previousActiveSegment.background(), true)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Block) writePowerLineSeparator(background, foreground string, end bool) {
|
|
|
|
symbol := b.activeSegment.PowerlineSymbol
|
|
|
|
if end {
|
|
|
|
symbol = b.previousActiveSegment.PowerlineSymbol
|
|
|
|
}
|
|
|
|
if b.activeSegment.InvertPowerline {
|
2021-04-20 12:30:46 -07:00
|
|
|
b.writer.write(foreground, background, symbol)
|
2021-04-18 10:16:06 -07:00
|
|
|
return
|
|
|
|
}
|
2021-04-20 12:30:46 -07:00
|
|
|
b.writer.write(background, foreground, symbol)
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Block) getPowerlineColor(foreground bool) string {
|
|
|
|
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 {
|
|
|
|
return b.activeSegment.background()
|
|
|
|
}
|
2021-04-18 10:16:06 -07:00
|
|
|
if !foreground && b.activeSegment.Style != Powerline {
|
|
|
|
return Transparent
|
|
|
|
}
|
|
|
|
if foreground && b.previousActiveSegment.Style != Powerline {
|
|
|
|
return Transparent
|
|
|
|
}
|
|
|
|
return b.previousActiveSegment.background()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Block) renderSegmentText(text string) {
|
|
|
|
switch b.activeSegment.Style {
|
|
|
|
case Plain:
|
|
|
|
b.renderPlainSegment(text)
|
|
|
|
case Diamond:
|
|
|
|
b.renderDiamondSegment(text)
|
|
|
|
case Powerline:
|
|
|
|
b.renderPowerLineSegment(text)
|
|
|
|
}
|
|
|
|
b.previousActiveSegment = b.activeSegment
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Block) renderPowerLineSegment(text string) {
|
|
|
|
b.writePowerLineSeparator(b.activeSegment.background(), b.getPowerlineColor(true), false)
|
|
|
|
b.renderText(text)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Block) renderPlainSegment(text string) {
|
|
|
|
b.renderText(text)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Block) renderDiamondSegment(text string) {
|
2021-04-20 12:30:46 -07:00
|
|
|
b.writer.write(Transparent, b.activeSegment.background(), b.activeSegment.LeadingDiamond)
|
2021-04-18 10:16:06 -07:00
|
|
|
b.renderText(text)
|
2021-04-20 12:30:46 -07:00
|
|
|
b.writer.write(Transparent, b.activeSegment.background(), b.activeSegment.TrailingDiamond)
|
2021-04-18 10:16:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Block) renderText(text string) {
|
2021-04-20 12:30:46 -07:00
|
|
|
text = b.ansi.generateHyperlink(text)
|
2021-04-18 10:16:06 -07:00
|
|
|
defaultValue := " "
|
|
|
|
prefix := b.activeSegment.getValue(Prefix, defaultValue)
|
|
|
|
postfix := b.activeSegment.getValue(Postfix, defaultValue)
|
2021-04-20 12:30:46 -07:00
|
|
|
b.writer.write(b.activeSegment.background(), b.activeSegment.foreground(), fmt.Sprintf("%s%s%s", prefix, text, postfix))
|
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)
|
|
|
|
if err != nil || !segment.shouldIncludeFolder(b.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)
|
|
|
|
b.previousActiveSegment = nil
|
|
|
|
b.activeSegment = segment
|
|
|
|
b.renderSegmentText(segmentTiming.stringValue)
|
|
|
|
if b.activeSegment.Style == Powerline {
|
|
|
|
b.writePowerLineSeparator(Transparent, b.activeSegment.background(), true)
|
|
|
|
}
|
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
|
|
|
|
}
|