package main import ( "fmt" "path/filepath" "sort" "strings" ) type path struct { props *properties env environmentInfo } const ( // FolderSeparatorIcon the path which is split will be separated by this icon FolderSeparatorIcon Property = "folder_separator_icon" // HomeIcon indicates the $HOME location HomeIcon Property = "home_icon" // FolderIcon identifies one folder FolderIcon Property = "folder_icon" // WindowsRegistryIcon indicates the registry location on Windows WindowsRegistryIcon Property = "windows_registry_icon" // Agnoster displays a short path with separator icon, this the default style Agnoster string = "agnoster" // AgnosterFull displays all the folder names with the folder_separator_icon AgnosterFull string = "agnoster_full" // AgnosterShort displays the folder names with one folder_separator_icon, regardless of depth AgnosterShort string = "agnoster_short" // Short displays a shorter path Short string = "short" // Full displays the full path Full string = "full" // Folder displays the current folder Folder string = "folder" // Mixed like agnoster, but if the path is short it displays it Mixed string = "mixed" // MixedThreshold the threshold of the length of the path Mixed will display MixedThreshold Property = "mixed_threshold" // MappedLocations allows overriding certain location with an icon MappedLocations Property = "mapped_locations" // MappedLocationsEnabled enables overriding certain locations with an icon MappedLocationsEnabled Property = "mapped_locations_enabled" ) func (pt *path) enabled() bool { return true } func (pt *path) string() string { cwd := pt.env.getcwd() var formattedPath string switch style := pt.props.getString(Style, Agnoster); style { case Agnoster: formattedPath = pt.getAgnosterPath() case AgnosterFull: formattedPath = pt.getAgnosterFullPath() case AgnosterShort: formattedPath = pt.getAgnosterShortPath() case Mixed: formattedPath = pt.getMixedPath() case Short: // "short" is a duplicate of "full", just here for backwards compatibility fallthrough case Full: formattedPath = pt.getFullPath() case Folder: formattedPath = pt.getFolderPath() default: return fmt.Sprintf("Path style: %s is not available", style) } formattedPath = pt.formatWindowsDrive(formattedPath) if pt.props.getBool(EnableHyperlink, false) { // wsl check if pt.env.isWsl() { cwd, _ = pt.env.runCommand("wslpath", "-m", cwd) } return fmt.Sprintf("[%s](file://%s)", formattedPath, cwd) } return formattedPath } func (pt *path) formatWindowsDrive(pwd string) string { if pt.env.getRuntimeGOOS() != windowsPlatform || !strings.HasSuffix(pwd, ":") { return pwd } return pwd + "\\" } func (pt *path) init(props *properties, env environmentInfo) { pt.props = props pt.env = env } func (pt *path) getMixedPath() string { var buffer strings.Builder pwd := pt.getPwd() splitted := strings.Split(pwd, pt.env.getPathSeperator()) threshold := int(pt.props.getFloat64(MixedThreshold, 4)) for i, part := range splitted { if part == "" { continue } folder := part if len(part) > threshold && i != 0 && i != len(splitted)-1 { folder = pt.props.getString(FolderIcon, "..") } separator := pt.props.getString(FolderSeparatorIcon, pt.env.getPathSeperator()) if i == 0 { separator = "" } buffer.WriteString(fmt.Sprintf("%s%s", separator, folder)) } return buffer.String() } func (pt *path) getAgnosterPath() string { var buffer strings.Builder pwd := pt.getPwd() buffer.WriteString(pt.rootLocation()) pathDepth := pt.pathDepth(pwd) for i := 1; i < pathDepth; i++ { buffer.WriteString(fmt.Sprintf("%s%s", pt.props.getString(FolderSeparatorIcon, pt.env.getPathSeperator()), pt.props.getString(FolderIcon, ".."))) } if pathDepth > 0 { buffer.WriteString(fmt.Sprintf("%s%s", pt.props.getString(FolderSeparatorIcon, pt.env.getPathSeperator()), base(pwd, pt.env))) } return buffer.String() } func (pt *path) getAgnosterFullPath() string { pwd := pt.getPwd() if string(pwd[0]) == pt.env.getPathSeperator() { pwd = pwd[1:] } return pt.replaceFolderSeparators(pwd) } func (pt *path) getAgnosterShortPath() string { pathSeparator := pt.env.getPathSeperator() folderSeparator := pt.props.getString(FolderSeparatorIcon, pathSeparator) folderIcon := pt.props.getString(FolderIcon, "..") root := pt.rootLocation() pwd := pt.getPwd() base := base(pwd, pt.env) pathDepth := pt.pathDepth(pwd) if pathDepth <= 0 { return root } if pathDepth == 1 { return fmt.Sprintf("%s%s%s", root, folderSeparator, base) } return fmt.Sprintf("%s%s%s%s%s", root, folderSeparator, folderIcon, folderSeparator, base) } func (pt *path) getFullPath() string { pwd := pt.getPwd() return pt.replaceFolderSeparators(pwd) } func (pt *path) getFolderPath() string { pwd := pt.getPwd() pwd = base(pwd, pt.env) return pt.replaceFolderSeparators(pwd) } func (pt *path) getPwd() string { pwd := *pt.env.getArgs().PSWD if pwd == "" { pwd = pt.env.getcwd() } if pt.props.getBool(MappedLocationsEnabled, true) { pwd = pt.replaceMappedLocations(pwd) } return pwd } func (pt *path) replaceMappedLocations(pwd string) string { if strings.HasPrefix(pwd, "Microsoft.PowerShell.Core\\FileSystem::") { pwd = strings.Replace(pwd, "Microsoft.PowerShell.Core\\FileSystem::", "", 1) } mappedLocations := map[string]string{ "HKCU:": pt.props.getString(WindowsRegistryIcon, "\uF013"), "HKLM:": pt.props.getString(WindowsRegistryIcon, "\uF013"), pt.env.homeDir(): pt.props.getString(HomeIcon, "~"), } // merge custom locations with mapped locations // mapped locations can override predefined locations keyValues := pt.props.getKeyValueMap(MappedLocations, make(map[string]string)) for key, val := range keyValues { mappedLocations[key] = val } // sort map keys in reverse order // fixes case when a subfoder and its parent are mapped // ex /users/test and /users/test/dev keys := make([]string, len(mappedLocations)) i := 0 for k := range mappedLocations { keys[i] = k i++ } sort.Sort(sort.Reverse(sort.StringSlice(keys))) for _, value := range keys { if strings.HasPrefix(pwd, value) { return strings.Replace(pwd, value, mappedLocations[value], 1) } } return pwd } func (pt *path) replaceFolderSeparators(pwd string) string { defaultSeparator := pt.env.getPathSeperator() if pwd == defaultSeparator { return pwd } folderSeparator := pt.props.getString(FolderSeparatorIcon, defaultSeparator) if folderSeparator == defaultSeparator { return pwd } pwd = strings.ReplaceAll(pwd, defaultSeparator, folderSeparator) return pwd } func (pt *path) inHomeDir(pwd string) bool { return strings.HasPrefix(pwd, pt.env.homeDir()) } func (pt *path) rootLocation() string { pwd := pt.getPwd() pwd = strings.TrimPrefix(pwd, pt.env.getPathSeperator()) splitted := strings.Split(pwd, pt.env.getPathSeperator()) rootLocation := splitted[0] return rootLocation } func (pt *path) pathDepth(pwd string) int { splitted := strings.Split(pwd, pt.env.getPathSeperator()) depth := 0 for _, part := range splitted { if part != "" { depth++ } } return depth - 1 } // Base returns the last element of path. // Trailing path separators are removed before extracting the last element. // If the path consists entirely of separators, Base returns a single separator. func base(path string, env environmentInfo) string { if path == "/" { return path } volumeName := filepath.VolumeName(path) // Strip trailing slashes. for len(path) > 0 && string(path[len(path)-1]) == env.getPathSeperator() { path = path[0 : len(path)-1] } if volumeName == path { return path } // Throw away volume name path = path[len(filepath.VolumeName(path)):] // Find the last element i := len(path) - 1 for i >= 0 && string(path[i]) != env.getPathSeperator() { i-- } if i >= 0 { path = path[i+1:] } // If empty now, it had only slashes. if path == "" { return env.getPathSeperator() } return path }