You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

64 lines
2.0 KiB

  1. /*
  2. * The MIT License (MIT)
  3. *
  4. * Copyright (c) 2015 Ian Coleman
  5. *
  6. * Permission is hereby granted, free of charge, to any person obtaining a copy
  7. * of this software and associated documentation files (the "Software"), to deal
  8. * in the Software without restriction, including without limitation the rights
  9. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  10. * copies of the Software, and to permit persons to whom the Software is
  11. * furnished to do so, Subject to the following conditions:
  12. *
  13. * The above copyright notice and this permission notice shall be included in all
  14. * copies or Substantial portions of the Software.
  15. *
  16. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  17. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  18. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  19. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  20. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  21. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  22. * SOFTWARE.
  23. */
  24. // Package strcase converts strings to snake_case or CamelCase
  25. package strcase
  26. import (
  27. "strings"
  28. )
  29. // Converts a string to snake_case
  30. func ToSnake(s string) string {
  31. s = addWordBoundariesToNumbers(s)
  32. s = strings.Trim(s, " ")
  33. n := ""
  34. for i, v := range s {
  35. // treat acronyms as words, eg for JSONData -> JSON is a whole word
  36. nextCaseIsChanged := false
  37. if i+1 < len(s) {
  38. next := s[i+1]
  39. if (v >= 'A' && v <= 'Z' && next >= 'a' && next <= 'z') || (v >= 'a' && v <= 'z' && next >= 'A' && next <= 'Z') {
  40. nextCaseIsChanged = true
  41. }
  42. }
  43. if i > 0 && n[len(n)-1] != '_' && nextCaseIsChanged {
  44. // add underscore if next letter case type is changed
  45. if v >= 'A' && v <= 'Z' {
  46. n += "_" + string(v)
  47. } else if v >= 'a' && v <= 'z' {
  48. n += string(v) + "_"
  49. }
  50. } else if v == ' ' {
  51. // replace spaces with underscores
  52. n += "_"
  53. } else {
  54. n = n + string(v)
  55. }
  56. }
  57. n = strings.ToLower(n)
  58. return n
  59. }