oh-my-posh/src/segment.go
2021-05-14 21:28:16 +02:00

280 lines
8.1 KiB
Go

package main
import (
"errors"
"fmt"
"time"
)
// Segment represent a single segment and it's configuration
type Segment struct {
Type SegmentType `config:"type"`
Style SegmentStyle `config:"style"`
PowerlineSymbol string `config:"powerline_symbol"`
InvertPowerline bool `config:"invert_powerline"`
Foreground string `config:"foreground"`
ForegroundTemplates []string `config:"foreground_templates"`
Background string `config:"background"`
BackgroundTemplates []string `config:"background_templates"`
LeadingDiamond string `config:"leading_diamond"`
TrailingDiamond string `config:"trailing_diamond"`
Properties map[Property]interface{} `config:"properties"`
props *properties
writer SegmentWriter
stringValue string
active bool
}
// 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
}
// SegmentWriter is the interface used to define what and if to write to the prompt
type SegmentWriter interface {
enabled() bool
string() string
init(props *properties, env environmentInfo)
}
// SegmentStyle the syle of segment, for more information, see the constants
type SegmentStyle string
// SegmentType the type of segment, for more information, see the constants
type SegmentType string
const (
// Session represents the user info segment
Session SegmentType = "session"
// Path represents the current path segment
Path SegmentType = "path"
// Git represents the git status and information
Git SegmentType = "git"
// Exit writes the last exit code
Exit SegmentType = "exit"
// Python writes the virtual env name
Python SegmentType = "python"
// Root writes root symbol
Root SegmentType = "root"
// Time writes the current timestamp
Time SegmentType = "time"
// Text writes a text
Text SegmentType = "text"
// Cmd writes the output of a shell command
Cmd SegmentType = "command"
// Battery writes the battery percentage
Battery SegmentType = "battery"
// Spotify writes the Spotify status for Mac
Spotify SegmentType = "spotify"
// ShellInfo writes which shell we're currently in
ShellInfo SegmentType = "shell"
// Node writes which node version is currently active
Node SegmentType = "node"
// Os write os specific icon
Os SegmentType = "os"
// EnvVar writes the content of an environment variable
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
Terraform SegmentType = "terraform"
// Golang writes which go version is currently active
Golang SegmentType = "go"
// Julia writes which julia version is currently active
Julia SegmentType = "julia"
// Powerline writes it Powerline style
Powerline SegmentStyle = "powerline"
// Plain writes it without ornaments
Plain SegmentStyle = "plain"
// Diamond writes the prompt shaped with a leading and trailing symbol
Diamond SegmentStyle = "diamond"
// YTM writes YouTube Music information and status
YTM SegmentType = "ytm"
// ExecutionTime writes the execution time of the last run command
ExecutionTime SegmentType = "executiontime"
// Ruby writes which ruby version is currently active
Ruby SegmentType = "ruby"
// Aws writes the active aws context
Aws SegmentType = "aws"
// Java writes the active java version
Java SegmentType = "java"
// PoshGit writes the posh git prompt
PoshGit SegmentType = "poshgit"
// AZFunc writes current AZ func version
AZFunc SegmentType = "azfunc"
// Crystal writes the active crystal version
Crystal SegmentType = "crystal"
// Dart writes the active dart version
Dart SegmentType = "dart"
)
func (segment *Segment) string() string {
return segment.writer.string()
}
func (segment *Segment) enabled() bool {
segment.active = segment.writer.enabled()
return segment.active
}
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) shouldIncludeFolder(cwd string) bool {
cwdIncluded := segment.cwdIncluded(cwd)
cwdExcluded := segment.cwdExcluded(cwd)
return (cwdIncluded && !cwdExcluded)
}
func (segment *Segment) cwdIncluded(cwd string) bool {
value, ok := segment.Properties[IncludeFolders]
if !ok {
// IncludeFolders isn't specified, everything is included
return true
}
list := parseStringArray(value)
if len(list) == 0 {
// IncludeFolders is an empty array, everything is included
return true
}
return segment.cwdMatchesOneOf(cwd, list)
}
func (segment *Segment) cwdExcluded(cwd string) bool {
value, ok := segment.Properties[ExcludeFolders]
if !ok {
value = segment.Properties[IgnoreFolders]
}
list := parseStringArray(value)
return segment.cwdMatchesOneOf(cwd, list)
}
func (segment *Segment) cwdMatchesOneOf(cwd string, regexes []string) bool {
for _, element := range regexes {
pattern := fmt.Sprintf("^%s$", element)
matched := matchString(pattern, cwd)
if matched {
return true
}
}
return false
}
func (segment *Segment) getColor(templates []string, defaultColor string) string {
if len(templates) == 0 {
return defaultColor
}
txtTemplate := &textTemplate{
Context: segment.writer,
}
for _, template := range templates {
txtTemplate.Template = template
value, err := txtTemplate.render()
if err != nil || value == "" {
continue
}
return value
}
return defaultColor
}
func (segment *Segment) foreground() string {
color := segment.Foreground
if segment.props != nil {
color = segment.props.foreground
}
return segment.getColor(segment.ForegroundTemplates, color)
}
func (segment *Segment) background() string {
color := segment.Background
if segment.props != nil {
color = segment.props.background
}
return segment.getColor(segment.BackgroundTemplates, color)
}
func (segment *Segment) mapSegmentWithWriter(env environmentInfo) error {
functions := map[SegmentType]SegmentWriter{
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{},
Ruby: &ruby{},
Aws: &aws{},
Java: &java{},
PoshGit: &poshgit{},
AZFunc: &azfunc{},
Crystal: &crystal{},
Dart: &dart{},
}
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
}
return errors.New("unable to map writer")
}
func (segment *Segment) setStringValue(env environmentInfo, cwd string) {
defer func() {
err := recover()
if err == nil {
return
}
// display a message explaining omp failed(with the err)
message := fmt.Sprintf("oh-my-posh fatal error rendering %s segment:%s", segment.Type, err)
fmt.Println(message)
segment.stringValue = "error"
segment.active = true
}()
err := segment.mapSegmentWithWriter(env)
if err != nil || !segment.shouldIncludeFolder(cwd) {
return
}
if segment.enabled() {
segment.stringValue = segment.string()
}
}