2019-03-13 04:14:30 -07:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"path/filepath"
|
2020-11-20 10:39:07 -08:00
|
|
|
"sort"
|
2019-03-13 04:14:30 -07:00
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
type path struct {
|
|
|
|
props *properties
|
|
|
|
env environmentInfo
|
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
2020-11-12 00:43:32 -08:00
|
|
|
// FolderSeparatorIcon the path which is split will be separated by this icon
|
2019-03-13 04:14:30 -07:00
|
|
|
FolderSeparatorIcon Property = "folder_separator_icon"
|
2020-11-12 00:43:32 -08:00
|
|
|
// HomeIcon indicates the $HOME location
|
2019-03-13 04:14:30 -07:00
|
|
|
HomeIcon Property = "home_icon"
|
2020-11-12 00:43:32 -08:00
|
|
|
// FolderIcon identifies one folder
|
2019-03-13 04:14:30 -07:00
|
|
|
FolderIcon Property = "folder_icon"
|
2020-11-12 00:43:32 -08:00
|
|
|
// WindowsRegistryIcon indicates the registry location on Windows
|
2019-03-13 04:14:30 -07:00
|
|
|
WindowsRegistryIcon Property = "windows_registry_icon"
|
2020-11-12 00:43:32 -08:00
|
|
|
// Agnoster displays a short path with separator icon, this the default style
|
2019-03-13 04:14:30 -07:00
|
|
|
Agnoster string = "agnoster"
|
2020-11-12 00:43:32 -08:00
|
|
|
// AgnosterFull displays all the folder names with the folder_separator_icon
|
2020-11-11 04:53:53 -08:00
|
|
|
AgnosterFull string = "agnoster_full"
|
2020-12-17 03:10:01 -08:00
|
|
|
// AgnosterShort displays the folder names with one folder_separator_icon, regardless of depth
|
|
|
|
AgnosterShort string = "agnoster_short"
|
2020-11-12 00:43:32 -08:00
|
|
|
// Short displays a shorter path
|
2019-03-13 04:14:30 -07:00
|
|
|
Short string = "short"
|
2020-11-12 00:43:32 -08:00
|
|
|
// Full displays the full path
|
2019-03-13 04:14:30 -07:00
|
|
|
Full string = "full"
|
2020-11-12 00:43:32 -08:00
|
|
|
// Folder displays the current folder
|
2019-03-13 04:14:30 -07:00
|
|
|
Folder string = "folder"
|
2020-11-27 11:10:19 -08:00
|
|
|
// MappedLocations allows overriding certain location with an icon
|
|
|
|
MappedLocations Property = "mapped_locations"
|
2020-12-24 13:17:00 -08:00
|
|
|
// MappedLocationsEnabled enables overriding certain locations with an icon
|
|
|
|
MappedLocationsEnabled Property = "mapped_locations_enabled"
|
2019-03-13 04:14:30 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
func (pt *path) enabled() bool {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pt *path) string() string {
|
|
|
|
switch style := pt.props.getString(Style, Agnoster); style {
|
|
|
|
case Agnoster:
|
|
|
|
return pt.getAgnosterPath()
|
2020-11-11 04:53:53 -08:00
|
|
|
case AgnosterFull:
|
|
|
|
return pt.getAgnosterFullPath()
|
2020-12-17 03:10:01 -08:00
|
|
|
case AgnosterShort:
|
|
|
|
return pt.getAgnosterShortPath()
|
2019-03-13 04:14:30 -07:00
|
|
|
case Short:
|
2020-12-24 13:17:00 -08:00
|
|
|
// "short" is a duplicate of "full", just here for backwards compatibility
|
|
|
|
fallthrough
|
2019-03-13 04:14:30 -07:00
|
|
|
case Full:
|
2020-12-24 13:17:00 -08:00
|
|
|
return pt.getFullPath()
|
2019-03-13 04:14:30 -07:00
|
|
|
case Folder:
|
2020-12-24 06:19:15 -08:00
|
|
|
return pt.getFolderPath()
|
2019-03-13 04:14:30 -07:00
|
|
|
default:
|
|
|
|
return fmt.Sprintf("Path style: %s is not available", style)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pt *path) init(props *properties, env environmentInfo) {
|
|
|
|
pt.props = props
|
|
|
|
pt.env = env
|
|
|
|
}
|
|
|
|
|
|
|
|
func (pt *path) getAgnosterPath() string {
|
|
|
|
buffer := new(bytes.Buffer)
|
2020-12-24 13:17:00 -08:00
|
|
|
pwd := pt.getPwd()
|
2020-11-17 10:22:56 -08:00
|
|
|
buffer.WriteString(pt.rootLocation())
|
2019-03-13 04:14:30 -07:00
|
|
|
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()
|
|
|
|
}
|
|
|
|
|
2020-11-11 04:53:53 -08:00
|
|
|
func (pt *path) getAgnosterFullPath() string {
|
2020-12-24 13:17:00 -08:00
|
|
|
pwd := pt.getPwd()
|
2020-11-11 04:53:53 -08:00
|
|
|
pathSeparator := pt.env.getPathSeperator()
|
|
|
|
folderSeparator := pt.props.getString(FolderSeparatorIcon, pathSeparator)
|
|
|
|
if string(pwd[0]) == pathSeparator {
|
|
|
|
pwd = pwd[1:]
|
|
|
|
}
|
2020-11-12 00:43:32 -08:00
|
|
|
return strings.ReplaceAll(pwd, pathSeparator, folderSeparator)
|
2020-11-11 04:53:53 -08:00
|
|
|
}
|
|
|
|
|
2020-12-17 03:10:01 -08:00
|
|
|
func (pt *path) getAgnosterShortPath() string {
|
|
|
|
pathSeparator := pt.env.getPathSeperator()
|
|
|
|
folderSeparator := pt.props.getString(FolderSeparatorIcon, pathSeparator)
|
|
|
|
folderIcon := pt.props.getString(FolderIcon, "..")
|
|
|
|
root := pt.rootLocation()
|
2020-12-24 13:17:00 -08:00
|
|
|
pwd := pt.getPwd()
|
|
|
|
base := base(pwd, pt.env)
|
|
|
|
pathDepth := pt.pathDepth(pwd)
|
2020-12-24 08:00:10 -08:00
|
|
|
if pathDepth <= 0 {
|
|
|
|
return root
|
|
|
|
}
|
2020-12-17 03:10:01 -08:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
2020-12-24 13:17:00 -08:00
|
|
|
func (pt *path) getFullPath() string {
|
|
|
|
return pt.getPwd()
|
|
|
|
}
|
|
|
|
|
2020-12-24 06:19:15 -08:00
|
|
|
func (pt *path) getFolderPath() string {
|
2020-12-24 13:17:00 -08:00
|
|
|
pwd := pt.getPwd()
|
2020-12-24 06:19:15 -08:00
|
|
|
return base(pwd, pt.env)
|
|
|
|
}
|
|
|
|
|
2020-12-24 13:17:00 -08:00
|
|
|
func (pt *path) getPwd() string {
|
|
|
|
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, "\uE0B1"),
|
|
|
|
"HKLM:": pt.props.getString(WindowsRegistryIcon, "\uE0B1"),
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2019-03-13 04:14:30 -07:00
|
|
|
func (pt *path) inHomeDir(pwd string) bool {
|
2020-10-02 12:50:13 -07:00
|
|
|
return strings.HasPrefix(pwd, pt.env.homeDir())
|
2019-03-13 04:14:30 -07:00
|
|
|
}
|
|
|
|
|
2020-11-17 10:22:56 -08:00
|
|
|
func (pt *path) rootLocation() string {
|
2020-12-24 13:17:00 -08:00
|
|
|
pwd := pt.getPwd()
|
2019-03-13 04:14:30 -07:00
|
|
|
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())
|
|
|
|
var validParts []string
|
|
|
|
for _, part := range splitted {
|
|
|
|
if part != "" {
|
|
|
|
validParts = append(validParts, part)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
depth := len(validParts)
|
|
|
|
return depth - 1
|
|
|
|
}
|
|
|
|
|
|
|
|
// Base returns the last element of path.
|
|
|
|
// Trailing path separators are removed before extracting the last element.
|
|
|
|
// If the path is empty, Base returns ".".
|
|
|
|
// If the path consists entirely of separators, Base returns a single separator.
|
|
|
|
func base(path string, env environmentInfo) string {
|
|
|
|
if path == "" {
|
|
|
|
return "."
|
|
|
|
}
|
|
|
|
// Strip trailing slashes.
|
|
|
|
for len(path) > 0 && string(path[len(path)-1]) == env.getPathSeperator() {
|
|
|
|
path = path[0 : len(path)-1]
|
|
|
|
}
|
|
|
|
// 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 == "" {
|
2020-11-12 00:43:32 -08:00
|
|
|
return env.getPathSeperator()
|
2019-03-13 04:14:30 -07:00
|
|
|
}
|
|
|
|
return path
|
|
|
|
}
|