mirror of
https://gitee.com/johng/gf.git
synced 2024-11-29 18:57:44 +08:00
288 lines
7.2 KiB
Go
288 lines
7.2 KiB
Go
// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
|
|
//
|
|
// This Source Code Form is subject to the terms of the MIT License.
|
|
// If a copy of the MIT was not distributed with this file,
|
|
// You can obtain one at https://github.com/gogf/gf.
|
|
//
|
|
// | Function | Result |
|
|
// |-----------------------------------|--------------------|
|
|
// | CaseSnake(s) | any_kind_of_string |
|
|
// | CaseSnakeScreaming(s) | ANY_KIND_OF_STRING |
|
|
// | CaseSnakeFirstUpper("RGBCodeMd5") | rgb_code_md5 |
|
|
// | CaseKebab(s) | any-kind-of-string |
|
|
// | CaseKebabScreaming(s) | ANY-KIND-OF-STRING |
|
|
// | CaseDelimited(s, '.') | any.kind.of.string |
|
|
// | CaseDelimitedScreaming(s, '.') | ANY.KIND.OF.STRING |
|
|
// | CaseCamel(s) | AnyKindOfString |
|
|
// | CaseCamelLower(s) | anyKindOfString |
|
|
|
|
package gstr
|
|
|
|
import (
|
|
"regexp"
|
|
"strings"
|
|
)
|
|
|
|
// CaseType is the type for Case.
|
|
type CaseType string
|
|
|
|
// The case type constants.
|
|
const (
|
|
Camel CaseType = "Camel"
|
|
CamelLower CaseType = "CamelLower"
|
|
Snake CaseType = "Snake"
|
|
SnakeFirstUpper CaseType = "SnakeFirstUpper"
|
|
SnakeScreaming CaseType = "SnakeScreaming"
|
|
Kebab CaseType = "Kebab"
|
|
KebabScreaming CaseType = "KebabScreaming"
|
|
Lower CaseType = "Lower"
|
|
)
|
|
|
|
var (
|
|
numberSequence = regexp.MustCompile(`([a-zA-Z]{0,1})(\d+)([a-zA-Z]{0,1})`)
|
|
firstCamelCaseStart = regexp.MustCompile(`([A-Z]+)([A-Z]?[_a-z\d]+)|$`)
|
|
firstCamelCaseEnd = regexp.MustCompile(`([\w\W]*?)([_]?[A-Z]+)$`)
|
|
)
|
|
|
|
// CaseTypeMatch matches the case type from string.
|
|
func CaseTypeMatch(caseStr string) CaseType {
|
|
caseTypes := []CaseType{
|
|
Camel,
|
|
CamelLower,
|
|
Snake,
|
|
SnakeFirstUpper,
|
|
SnakeScreaming,
|
|
Kebab,
|
|
KebabScreaming,
|
|
Lower,
|
|
}
|
|
|
|
for _, caseType := range caseTypes {
|
|
if Equal(caseStr, string(caseType)) {
|
|
return caseType
|
|
}
|
|
}
|
|
|
|
return CaseType(caseStr)
|
|
}
|
|
|
|
// CaseConvert converts a string to the specified naming convention.
|
|
// Use CaseTypeMatch to match the case type from string.
|
|
func CaseConvert(s string, caseType CaseType) string {
|
|
if s == "" || caseType == "" {
|
|
return s
|
|
}
|
|
|
|
switch caseType {
|
|
case Camel:
|
|
return CaseCamel(s)
|
|
|
|
case CamelLower:
|
|
return CaseCamelLower(s)
|
|
|
|
case Kebab:
|
|
return CaseKebab(s)
|
|
|
|
case KebabScreaming:
|
|
return CaseKebabScreaming(s)
|
|
|
|
case Snake:
|
|
return CaseSnake(s)
|
|
|
|
case SnakeFirstUpper:
|
|
return CaseSnakeFirstUpper(s)
|
|
|
|
case SnakeScreaming:
|
|
return CaseSnakeScreaming(s)
|
|
|
|
case Lower:
|
|
return ToLower(s)
|
|
|
|
default:
|
|
return s
|
|
}
|
|
}
|
|
|
|
// CaseCamel converts a string to CamelCase.
|
|
//
|
|
// Example:
|
|
// CaseCamel("any_kind_of_string") -> AnyKindOfString
|
|
// CaseCamel("anyKindOfString") -> AnyKindOfString
|
|
func CaseCamel(s string) string {
|
|
return toCamelInitCase(s, true)
|
|
}
|
|
|
|
// CaseCamelLower converts a string to lowerCamelCase.
|
|
//
|
|
// Example:
|
|
// CaseCamelLower("any_kind_of_string") -> anyKindOfString
|
|
// CaseCamelLower("AnyKindOfString") -> anyKindOfString
|
|
func CaseCamelLower(s string) string {
|
|
if s == "" {
|
|
return s
|
|
}
|
|
if r := rune(s[0]); r >= 'A' && r <= 'Z' {
|
|
s = strings.ToLower(string(r)) + s[1:]
|
|
}
|
|
return toCamelInitCase(s, false)
|
|
}
|
|
|
|
// CaseSnake converts a string to snake_case.
|
|
//
|
|
// Example:
|
|
// CaseSnake("AnyKindOfString") -> any_kind_of_string
|
|
func CaseSnake(s string) string {
|
|
return CaseDelimited(s, '_')
|
|
}
|
|
|
|
// CaseSnakeScreaming converts a string to SNAKE_CASE_SCREAMING.
|
|
//
|
|
// Example:
|
|
// CaseSnakeScreaming("AnyKindOfString") -> ANY_KIND_OF_STRING
|
|
func CaseSnakeScreaming(s string) string {
|
|
return CaseDelimitedScreaming(s, '_', true)
|
|
}
|
|
|
|
// CaseSnakeFirstUpper converts a string like "RGBCodeMd5" to "rgb_code_md5".
|
|
// TODO for efficiency should change regexp to traversing string in future.
|
|
//
|
|
// Example:
|
|
// CaseSnakeFirstUpper("RGBCodeMd5") -> rgb_code_md5
|
|
func CaseSnakeFirstUpper(word string, underscore ...string) string {
|
|
replace := "_"
|
|
if len(underscore) > 0 {
|
|
replace = underscore[0]
|
|
}
|
|
|
|
m := firstCamelCaseEnd.FindAllStringSubmatch(word, 1)
|
|
if len(m) > 0 {
|
|
word = m[0][1] + replace + TrimLeft(ToLower(m[0][2]), replace)
|
|
}
|
|
|
|
for {
|
|
m = firstCamelCaseStart.FindAllStringSubmatch(word, 1)
|
|
if len(m) > 0 && m[0][1] != "" {
|
|
w := strings.ToLower(m[0][1])
|
|
w = w[:len(w)-1] + replace + string(w[len(w)-1])
|
|
|
|
word = strings.Replace(word, m[0][1], w, 1)
|
|
} else {
|
|
break
|
|
}
|
|
}
|
|
|
|
return TrimLeft(word, replace)
|
|
}
|
|
|
|
// CaseKebab converts a string to kebab-case.
|
|
//
|
|
// Example:
|
|
// CaseKebab("AnyKindOfString") -> any-kind-of-string
|
|
func CaseKebab(s string) string {
|
|
return CaseDelimited(s, '-')
|
|
}
|
|
|
|
// CaseKebabScreaming converts a string to KEBAB-CASE-SCREAMING.
|
|
//
|
|
// Example:
|
|
// CaseKebab("AnyKindOfString") -> ANY-KIND-OF-STRING
|
|
func CaseKebabScreaming(s string) string {
|
|
return CaseDelimitedScreaming(s, '-', true)
|
|
}
|
|
|
|
// CaseDelimited converts a string to snake.case.delimited.
|
|
//
|
|
// Example:
|
|
// CaseDelimited("AnyKindOfString", '.') -> any.kind.of.string
|
|
func CaseDelimited(s string, del byte) string {
|
|
return CaseDelimitedScreaming(s, del, false)
|
|
}
|
|
|
|
// CaseDelimitedScreaming converts a string to DELIMITED.SCREAMING.CASE or delimited.screaming.case.
|
|
//
|
|
// Example:
|
|
// CaseDelimitedScreaming("AnyKindOfString", '.') -> ANY.KIND.OF.STRING
|
|
func CaseDelimitedScreaming(s string, del uint8, screaming bool) string {
|
|
s = addWordBoundariesToNumbers(s)
|
|
s = strings.Trim(s, " ")
|
|
n := ""
|
|
for i, v := range s {
|
|
// treat acronyms as words, eg for JSONData -> JSON is a whole word
|
|
nextCaseIsChanged := false
|
|
if i+1 < len(s) {
|
|
next := s[i+1]
|
|
if (v >= 'A' && v <= 'Z' && next >= 'a' && next <= 'z') || (v >= 'a' && v <= 'z' && next >= 'A' && next <= 'Z') {
|
|
nextCaseIsChanged = true
|
|
}
|
|
}
|
|
|
|
if i > 0 && n[len(n)-1] != del && nextCaseIsChanged {
|
|
// add underscore if next letter case type is changed
|
|
if v >= 'A' && v <= 'Z' {
|
|
n += string(del) + string(v)
|
|
} else if v >= 'a' && v <= 'z' {
|
|
n += string(v) + string(del)
|
|
}
|
|
} else if v == ' ' || v == '_' || v == '-' || v == '.' {
|
|
// replace spaces/underscores with delimiters
|
|
n += string(del)
|
|
} else {
|
|
n = n + string(v)
|
|
}
|
|
}
|
|
|
|
if screaming {
|
|
n = strings.ToUpper(n)
|
|
} else {
|
|
n = strings.ToLower(n)
|
|
}
|
|
return n
|
|
}
|
|
|
|
func addWordBoundariesToNumbers(s string) string {
|
|
r := numberSequence.ReplaceAllFunc([]byte(s), func(bytes []byte) []byte {
|
|
var result []byte
|
|
match := numberSequence.FindSubmatch(bytes)
|
|
if len(match[1]) > 0 {
|
|
result = append(result, match[1]...)
|
|
result = append(result, []byte(" ")...)
|
|
}
|
|
result = append(result, match[2]...)
|
|
if len(match[3]) > 0 {
|
|
result = append(result, []byte(" ")...)
|
|
result = append(result, match[3]...)
|
|
}
|
|
return result
|
|
})
|
|
return string(r)
|
|
}
|
|
|
|
// Converts a string to CamelCase
|
|
func toCamelInitCase(s string, initCase bool) string {
|
|
s = addWordBoundariesToNumbers(s)
|
|
s = strings.Trim(s, " ")
|
|
n := ""
|
|
capNext := initCase
|
|
for _, v := range s {
|
|
if v >= 'A' && v <= 'Z' {
|
|
n += string(v)
|
|
}
|
|
if v >= '0' && v <= '9' {
|
|
n += string(v)
|
|
}
|
|
if v >= 'a' && v <= 'z' {
|
|
if capNext {
|
|
n += strings.ToUpper(string(v))
|
|
} else {
|
|
n += string(v)
|
|
}
|
|
}
|
|
if v == '_' || v == ' ' || v == '-' || v == '.' {
|
|
capNext = true
|
|
} else {
|
|
capNext = false
|
|
}
|
|
}
|
|
return n
|
|
}
|