// segment.  MakeSegment is similar to MakePath but disallows the '/' and
 // '#' characters because of their reserved meaning in URIs.
 func (p *PathSpec) MakeSegment(s string) string {
-       s = p.MakePathSanitized(strings.Trim(segmentReplacer.Replace(s), "- "))
+       return p.MakePathSanitized(segmentReplacer.Replace(s))
 
-       var pos int
-       var last byte
-       b := make([]byte, len(s))
-
-       for i := 0; i < len(s); i++ {
-               // consolidate dashes
-               if s[i] == '-' && last == '-' {
-                       continue
-               }
-
-               b[pos], last = s[i], s[i]
-               pos++
-       }
-
-       if p.DisablePathToLower {
-               return string(b[:pos])
-       }
-       return strings.ToLower(string(b[:pos]))
 }
 
 // MakePath takes a string with any characters and replace it
 // whilst preserving the original casing of the string.
 // E.g. Social Media -> Social-Media
 func (p *PathSpec) MakePath(s string) string {
-       return p.UnicodeSanitize(strings.Replace(strings.TrimSpace(s), " ", "-", -1))
+       return p.UnicodeSanitize(s)
 }
 
 // MakePathSanitized creates a Unicode-sanitized string, with the spaces replaced
 // a predefined set of special Unicode characters.
 // If RemovePathAccents configuration flag is enabled, Uniccode accents
 // are also removed.
+// Spaces will be replaced with a single hyphen, and sequential hyphens will be reduced to one.
 func (p *PathSpec) UnicodeSanitize(s string) string {
        source := []rune(s)
        target := make([]rune, 0, len(source))
+       var prependHyphen bool
 
        for i, r := range source {
-               if r == '%' && i+2 < len(source) && ishex(source[i+1]) && ishex(source[i+2]) {
-                       target = append(target, r)
-               } else if unicode.IsLetter(r) || unicode.IsDigit(r) || unicode.IsMark(r) || r == '.' || r == '/' || r == '\\' || r == '_' || r == '-' || r == '#' || r == '+' || r == '~' {
+               isAllowed := r == '.' || r == '/' || r == '\\' || r == '_' || r == '#' || r == '+' || r == '~'
+               isAllowed = isAllowed || unicode.IsLetter(r) || unicode.IsDigit(r) || unicode.IsMark(r)
+               isAllowed = isAllowed || (r == '%' && i+2 < len(source) && ishex(source[i+1]) && ishex(source[i+2]))
+
+               if isAllowed {
+                       if prependHyphen {
+                               target = append(target, '-')
+                               prependHyphen = false
+                       }
                        target = append(target, r)
+               } else if len(target) > 0 && (r == '-' || unicode.IsSpace(r)) {
+                       prependHyphen = true
                }
        }
 
 
                {"Your #1 Fan", "your-1-fan"},
                {"Red & Blue", "red-blue"},
                {"double//slash", "double-slash"},
-               {"My // Taxonomy", "my-taxonomy"},
+               {"triple///slash", "triple-slash"},
+               {"-my/way-", "my-way"},
        }
 
        for _, test := range tests {
 
 
 // pageToPermalinkTitle returns the URL-safe form of the title
 func pageToPermalinkTitle(p *Page, _ string) (string, error) {
-       if p.Kind == "taxonomy" {
+       if p.Kind == KindTaxonomy {
                // Taxonomies are allowed to have '/' characters, so don't normalize
                // them with MakeSegment.
-               return p.s.PathSpec.URLize(p.title), nil
+               return p.s.PathSpec.MakePathSanitized(p.title), nil
        }
 
        return p.s.PathSpec.MakeSegment(p.title), nil