oh-my-posh/segment.go

189 lines
5.6 KiB
Go
Raw Normal View History

2019-03-13 04:14:30 -07:00
package main
import (
"errors"
"fmt"
"regexp"
"time"
)
2020-09-17 07:51:29 -07:00
// Segment represent a single segment and it's configuration
2019-03-13 04:14:30 -07:00
type Segment struct {
2020-09-27 00:37:50 -07:00
Type SegmentType `json:"type"`
Style SegmentStyle `json:"style"`
PowerlineSymbol string `json:"powerline_symbol"`
InvertPowerline bool `json:"invert_powerline"`
Foreground string `json:"foreground"`
Background string `json:"background"`
LeadingDiamond string `json:"leading_diamond"`
TrailingDiamond string `json:"trailing_diamond"`
Properties map[Property]interface{} `json:"properties"`
props *properties
2020-09-27 00:37:50 -07:00
writer SegmentWriter
stringValue string
active bool
timing time.Duration
2019-03-13 04:14:30 -07:00
}
// SegmentWriter is the interface used to define what and if to write to the prompt
2019-03-13 04:14:30 -07:00
type SegmentWriter interface {
enabled() bool
string() string
init(props *properties, env environmentInfo)
}
// SegmentStyle the syle of segment, for more information, see the constants
2019-03-13 04:14:30 -07:00
type SegmentStyle string
// SegmentType the type of segment, for more information, see the constants
2019-03-13 04:14:30 -07:00
type SegmentType string
const (
// Session represents the user info segment
2019-03-13 04:14:30 -07:00
Session SegmentType = "session"
// Path represents the current path segment
2019-03-13 04:14:30 -07:00
Path SegmentType = "path"
// Git represents the git status and information
2019-03-13 04:14:30 -07:00
Git SegmentType = "git"
// Exit writes the last exit code
2019-03-13 04:14:30 -07:00
Exit SegmentType = "exit"
// Python writes the virtual env name
Python SegmentType = "python"
// Root writes root symbol
2019-03-13 04:14:30 -07:00
Root SegmentType = "root"
// Time writes the current timestamp
2019-03-13 04:14:30 -07:00
Time SegmentType = "time"
// Text writes a text
2019-03-13 04:14:30 -07:00
Text SegmentType = "text"
// Cmd writes the output of a shell command
2019-03-13 04:14:30 -07:00
Cmd SegmentType = "command"
// Battery writes the battery percentage
2019-03-13 04:14:30 -07:00
Battery SegmentType = "battery"
// Spotify writes the Spotify status for Mac
2019-03-13 04:14:30 -07:00
Spotify SegmentType = "spotify"
// ShellInfo writes which shell we're currently in
2020-09-15 04:44:53 -07:00
ShellInfo SegmentType = "shell"
// Node writes which node version is currently active
2020-10-01 11:57:02 -07:00
Node SegmentType = "node"
// Os write os specific icon
2020-10-07 12:01:03 -07:00
Os SegmentType = "os"
// EnvVar writes the content of an environment variable
2020-10-09 10:22:32 -07:00
EnvVar SegmentType = "envvar"
// Az writes the Azure subscription info we're currently in
Az SegmentType = "az"
// Kubectl writes the Kubernetes context we're currently in
Kubectl SegmentType = "kubectl"
// Dotnet writes which dotnet version is currently active
Dotnet SegmentType = "dotnet"
// Terraform writes the terraform workspace we're currently in
2020-10-16 07:25:38 -07:00
Terraform SegmentType = "terraform"
// Golang writes which go version is currently active
2020-10-22 04:47:42 -07:00
Golang SegmentType = "go"
2020-11-14 11:04:04 -08:00
// Julia writes which julia version is currently active
Julia SegmentType = "julia"
// Powerline writes it Powerline style
2019-03-13 04:14:30 -07:00
Powerline SegmentStyle = "powerline"
// Plain writes it without ornaments
2019-03-13 04:14:30 -07:00
Plain SegmentStyle = "plain"
// Diamond writes the prompt shaped with a leading and trailing symbol
2019-03-13 04:14:30 -07:00
Diamond SegmentStyle = "diamond"
// YTM writes YouTube Music information and status
YTM SegmentType = "ytm"
2020-12-06 13:03:40 -08:00
// ExecutionTime writes the execution time of the last run command
ExecutionTime SegmentType = "executiontime"
2019-03-13 04:14:30 -07:00
)
func (segment *Segment) string() string {
return segment.writer.string()
}
func (segment *Segment) enabled() bool {
segment.active = segment.writer.enabled()
return segment.active
2019-03-13 04:14:30 -07:00
}
2020-10-02 07:58:25 -07:00
func (segment *Segment) getValue(property Property, defaultValue string) string {
if value, ok := segment.Properties[property]; ok {
return parseString(value, defaultValue)
}
return defaultValue
}
func (segment *Segment) shouldIgnoreFolder(cwd string) bool {
if value, ok := segment.Properties[IgnoreFolders]; ok {
2020-10-02 07:58:25 -07:00
list := parseStringArray(value)
for _, element := range list {
pattern := fmt.Sprintf("^%s$", element)
matched, err := regexp.MatchString(pattern, cwd)
if err == nil && matched {
2020-10-02 07:58:25 -07:00
return true
}
}
return false
}
return false
}
func (segment *Segment) mapSegmentWithWriter(env environmentInfo) error {
2019-03-13 04:14:30 -07:00
functions := map[SegmentType]SegmentWriter{
2020-12-06 13:03:40 -08:00
Session: &session{},
Path: &path{},
Git: &git{},
Exit: &exit{},
Python: &python{},
Root: &root{},
Text: &text{},
Time: &tempus{},
Cmd: &command{},
Battery: &batt{},
Spotify: &spotify{},
ShellInfo: &shell{},
Node: &node{},
Os: &osInfo{},
EnvVar: &envvar{},
Az: &az{},
Kubectl: &kubectl{},
Dotnet: &dotnet{},
Terraform: &terraform{},
Golang: &golang{},
Julia: &julia{},
YTM: &ytm{},
ExecutionTime: &executiontime{},
2019-03-13 04:14:30 -07:00
}
if writer, ok := functions[segment.Type]; ok {
props := &properties{
values: segment.Properties,
foreground: segment.Foreground,
background: segment.Background,
}
writer.init(props, env)
segment.writer = writer
segment.props = props
return nil
2019-03-13 04:14:30 -07:00
}
return errors.New("unable to map writer")
2019-03-13 04:14:30 -07:00
}
func (segment *Segment) setStringValue(env environmentInfo, cwd string, debug bool) {
err := segment.mapSegmentWithWriter(env)
if err != nil || segment.shouldIgnoreFolder(cwd) {
return
}
// add timing only in debug
if debug {
start := time.Now()
defer (func() {
// force segment rendering to display the time it took
// to check if the segment is enabled or not
// depending on the segement, calling enabled()
// can be time consuming
segment.active = true
elapsed := time.Since(start)
segment.timing = elapsed
})()
}
if segment.enabled() {
segment.stringValue = segment.string()
}
}