2013-03-13 15:14:21 +01:00
|
|
|
// Copyright 2013 by Dobrosław Żybort. All rights reserved.
|
|
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
|
|
|
package slug
|
|
|
|
|
|
|
|
import (
|
|
|
|
"github.com/fiam/gounidecode/unidecode"
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2013-05-10 15:23:01 +02:00
|
|
|
var (
|
|
|
|
// Custom substitution map
|
2013-05-10 16:25:44 +02:00
|
|
|
CustomSub map[string]string
|
|
|
|
// Custom rune substitution map
|
|
|
|
CustomRuneSub map[rune]string
|
2013-05-10 19:21:18 +02:00
|
|
|
|
|
|
|
// Maximum slug length. It's smart so it will cat slug after full word.
|
|
|
|
// By default slugs aren't shortened.
|
|
|
|
// If MaxLength is smaller than length of the first word, then returned
|
2013-05-10 19:25:47 +02:00
|
|
|
// slug will contain only substring from the first word truncated
|
|
|
|
// after MaxLength.
|
2013-05-10 19:21:18 +02:00
|
|
|
MaxLength int
|
2013-05-10 15:23:01 +02:00
|
|
|
)
|
|
|
|
|
2013-03-13 15:14:21 +01:00
|
|
|
//=============================================================================
|
|
|
|
|
|
|
|
// Make returns slug generated from provided string. Will use "en" as language
|
|
|
|
// substitution.
|
|
|
|
func Make(s string) (slug string) {
|
|
|
|
return MakeLang(s, "en")
|
|
|
|
}
|
|
|
|
|
|
|
|
// MakeLang returns slug generated from provided string and will use provided
|
|
|
|
// language for chars substitution.
|
|
|
|
func MakeLang(s string, lang string) (slug string) {
|
|
|
|
slug = strings.TrimSpace(s)
|
|
|
|
|
2014-03-05 20:28:21 +01:00
|
|
|
// Custom substitutions
|
|
|
|
// Always substitute runes first
|
|
|
|
slug = SubstituteRune(slug, CustomRuneSub)
|
|
|
|
slug = Substitute(slug, CustomSub)
|
|
|
|
|
|
|
|
// Process string with selected substitution language
|
2013-03-13 15:14:21 +01:00
|
|
|
switch lang {
|
|
|
|
case "de":
|
2013-05-10 16:25:44 +02:00
|
|
|
slug = SubstituteRune(slug, deSub)
|
2013-03-13 15:14:21 +01:00
|
|
|
case "en":
|
2013-05-10 16:25:44 +02:00
|
|
|
slug = SubstituteRune(slug, enSub)
|
2013-03-13 15:14:21 +01:00
|
|
|
case "pl":
|
2013-05-10 16:25:44 +02:00
|
|
|
slug = SubstituteRune(slug, plSub)
|
2013-03-13 15:14:21 +01:00
|
|
|
default: // fallback to "en" if lang not found
|
2013-05-10 16:25:44 +02:00
|
|
|
slug = SubstituteRune(slug, enSub)
|
2013-03-13 15:14:21 +01:00
|
|
|
}
|
|
|
|
|
2013-05-10 16:25:44 +02:00
|
|
|
slug = SubstituteRune(slug, defaultSub)
|
|
|
|
|
2014-03-05 20:28:21 +01:00
|
|
|
// Process all non ASCII symbols
|
2013-03-13 15:14:21 +01:00
|
|
|
slug = unidecode.Unidecode(slug)
|
|
|
|
|
|
|
|
slug = strings.ToLower(slug)
|
|
|
|
|
2014-03-05 20:28:21 +01:00
|
|
|
// Process all remaining symbols
|
2013-03-13 15:14:21 +01:00
|
|
|
slug = regexp.MustCompile("[^a-z0-9-_]").ReplaceAllString(slug, "-")
|
|
|
|
slug = regexp.MustCompile("-+").ReplaceAllString(slug, "-")
|
|
|
|
slug = strings.Trim(slug, "-")
|
2013-05-10 19:21:18 +02:00
|
|
|
|
|
|
|
if MaxLength > 0 {
|
|
|
|
slug = smartTruncate(slug)
|
|
|
|
}
|
|
|
|
|
2013-03-13 15:14:21 +01:00
|
|
|
return slug
|
|
|
|
}
|
|
|
|
|
2013-05-10 16:25:44 +02:00
|
|
|
// Substitute returns string with superseded all substrings from
|
|
|
|
// provided substitution map.
|
|
|
|
func Substitute(s string, sub map[string]string) (buf string) {
|
|
|
|
buf = s
|
|
|
|
for key, val := range sub {
|
|
|
|
buf = strings.Replace(s, key, val, -1)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// SubstituteRune substitutes string chars with provided rune
|
|
|
|
// substitution map.
|
|
|
|
func SubstituteRune(s string, sub map[rune]string) (buf string) {
|
2013-03-13 15:14:21 +01:00
|
|
|
for _, c := range s {
|
|
|
|
if d, ok := sub[c]; ok {
|
|
|
|
buf += d
|
|
|
|
} else {
|
|
|
|
buf += string(c)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
2013-05-10 19:21:18 +02:00
|
|
|
|
|
|
|
func smartTruncate(text string) string {
|
|
|
|
if len(text) < MaxLength {
|
|
|
|
return text
|
|
|
|
}
|
|
|
|
|
|
|
|
var truncated string
|
|
|
|
words := strings.SplitAfter(text, "-")
|
|
|
|
// If MaxLength is smaller than length of the first word return word
|
|
|
|
// truncated after MaxLength.
|
|
|
|
if len(words[0]) > MaxLength {
|
|
|
|
return words[0][:MaxLength]
|
|
|
|
}
|
|
|
|
for _, word := range words {
|
|
|
|
if len(truncated)+len(word)-1 <= MaxLength {
|
|
|
|
truncated = truncated + word
|
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return strings.Trim(truncated, "-")
|
|
|
|
}
|