|  | // Copyright 2018 The Fuchsia Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | package fidlgen | 
|  |  | 
|  | import ( | 
|  | "strings" | 
|  | "unicode" | 
|  | "unicode/utf8" | 
|  | ) | 
|  |  | 
|  | // The functions in this file convert names between snake_case, ALL_CAPS_SNAKE, | 
|  | // UpperCamelCase, lowerCamelCase, kCamelCase, and "friendly case". They accept | 
|  | // inputs in any style, split into parts, and recombine with the desired style. | 
|  | // See "RFC-0040: Identifier Uniqueness" for details on how name-part boundaries | 
|  | // are calculated. | 
|  | // | 
|  | // TODO(fxbug.dev/95218): Fix any discrepancies with RFC-0040. In particular, | 
|  | // ensure that all these functions are idempotent. | 
|  |  | 
|  | // nameParts breaks an identifier into parts recognized according to RFC-0040 | 
|  | // which can be recombined into identifiers in different case systems. | 
|  | // | 
|  | // Splits the string according to what is recognized as boundaries for both | 
|  | // snake case and camel case, allowing it to be transformed into either case | 
|  | // system regardless of the original case. Note that runs of multiple uppercase | 
|  | // characters are treated as a single part, so all-caps abbreviations will be | 
|  | // grouped. For example, "HTTPExample" will be split into "HTTP" and "Example". | 
|  | func nameParts(name string) []string { | 
|  | var parts []string | 
|  | for _, namePart := range strings.Split(name, "_") { | 
|  | partStart := 0 | 
|  | lastRune, _ := utf8.DecodeRuneInString(namePart) | 
|  | lastRuneStart := 0 | 
|  | for i, curRune := range namePart { | 
|  | if i == 0 { | 
|  | continue | 
|  | } | 
|  | if unicode.IsUpper(curRune) && !unicode.IsUpper(lastRune) { | 
|  | parts = append(parts, namePart[partStart:i]) | 
|  | partStart = i | 
|  | } | 
|  | if !(unicode.IsUpper(curRune) || unicode.IsDigit(curRune)) && unicode.IsUpper(lastRune) && partStart != lastRuneStart { | 
|  | parts = append(parts, namePart[partStart:lastRuneStart]) | 
|  | partStart = lastRuneStart | 
|  | } | 
|  | lastRuneStart = i | 
|  | lastRune = curRune | 
|  | } | 
|  | parts = append(parts, namePart[partStart:]) | 
|  | } | 
|  | return parts | 
|  | } | 
|  |  | 
|  | // ToSnakeCase converts an identifier to RFC-0040 canonical snake_case style. | 
|  | // Works independent of which case the identifier is originally in. | 
|  | func ToSnakeCase(name string) string { | 
|  | parts := nameParts(name) | 
|  | for i := range parts { | 
|  | parts[i] = strings.ToLower(parts[i]) | 
|  | } | 
|  | return strings.Join(parts, "_") | 
|  | } | 
|  |  | 
|  | // ToUpperCamelCase converts an identifier to RFC-0040 UpperCamelCase style. | 
|  | // Works independent of which case the identifier is originally in. Note that | 
|  | // canonical identifiers use title case for abbreviations, so e.g. HTTPExample | 
|  | // will become HttpExample. | 
|  | func ToUpperCamelCase(name string) string { | 
|  | parts := nameParts(name) | 
|  | for i := range parts { | 
|  | parts[i] = strings.Title(strings.ToLower(parts[i])) | 
|  | if parts[i] == "" { | 
|  | parts[i] = "_" | 
|  | } | 
|  | } | 
|  | return strings.Join(parts, "") | 
|  | } | 
|  |  | 
|  | // ToLowerCamelCase converts an identifier to RFC-0040 lowerCamelCase style. | 
|  | // Works independent of which case the identifier is originally in. Note that | 
|  | // canonical identifiers use title case for abbreviations, so e.g. ExampleHTTP | 
|  | // will become ExampleHttp. | 
|  | func ToLowerCamelCase(name string) string { | 
|  | parts := nameParts(name) | 
|  | for i := range parts { | 
|  | if i == 0 { | 
|  | parts[i] = strings.ToLower(parts[i]) | 
|  | } else { | 
|  | parts[i] = strings.Title(strings.ToLower(parts[i])) | 
|  | } | 
|  | if parts[i] == "" { | 
|  | parts[i] = "_" | 
|  | } | 
|  | } | 
|  | return strings.Join(parts, "") | 
|  | } | 
|  |  | 
|  | // ToFriendlyCase converts an identifier to RFC-0040 "friendly case" style (like | 
|  | // snake case, but with spaces). Works independent of which case the identifier | 
|  | // is originally in. | 
|  | func ToFriendlyCase(name string) string { | 
|  | parts := nameParts(name) | 
|  | for i := range parts { | 
|  | parts[i] = strings.ToLower(parts[i]) | 
|  | } | 
|  | return strings.Join(parts, " ") | 
|  | } | 
|  |  | 
|  | // ConstNameToAllCapsSnake converts a const name from kCamelCase to | 
|  | // ALL_CAPS_SNAKE style | 
|  | func ConstNameToAllCapsSnake(name string) string { | 
|  | parts := nameParts(RemoveLeadingK(name)) | 
|  | for i := range parts { | 
|  | parts[i] = strings.ToUpper(parts[i]) | 
|  | } | 
|  | return strings.Join(parts, "_") | 
|  | } | 
|  |  | 
|  | // ConstNameToKCamelCase converts a const name to kCamelCase style | 
|  | func ConstNameToKCamelCase(name string) string { | 
|  | return "k" + ToUpperCamelCase(RemoveLeadingK(name)) | 
|  | } | 
|  |  | 
|  | // RemoveLeadingK removes a leading 'k' if the second character is upper-case, | 
|  | // otherwise returns the argument | 
|  | func RemoveLeadingK(name string) string { | 
|  | if len(name) >= 2 && name[0] == 'k' { | 
|  | secondRune, _ := utf8.DecodeRuneInString(name[1:]) | 
|  | if unicode.IsUpper(secondRune) { | 
|  | name = name[1:] | 
|  | } | 
|  | } | 
|  | return name | 
|  | } |