Você não pode selecionar mais de 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.

99 linhas
2.5KB

  1. package objects
  2. import (
  3. "github.com/deadsy/sdfx/sdf"
  4. "math"
  5. )
  6. type BoxParams struct {
  7. Size sdf.V3 // internal box size
  8. Wall float64 // wall thickness
  9. BottomTopRatio float64 // ratio between the height of the top and the bottom
  10. CornerRadius float64 // radius of rounded corners
  11. Taper float64 // angle of split
  12. ClosingHeight float64 // height of the closing
  13. Margin float64 // Margin for closing
  14. Text string
  15. }
  16. func Box3D(k *BoxParams) (top sdf.SDF3, bottom sdf.SDF3, err error) {
  17. top, err = TopBox3D(k)
  18. bottom, err = TopBox3D(k)
  19. if err != nil {
  20. return
  21. }
  22. return
  23. }
  24. func baseBox3D(k *BoxParams) (sdf.SDF3, error) {
  25. box, err := sdf.Box3D(k.Size.AddScalar(k.Wall*2), k.CornerRadius)
  26. if err != nil {
  27. return nil, err
  28. }
  29. return box, nil
  30. }
  31. func TopBox3D(k *BoxParams) (top sdf.SDF3, err error) {
  32. box, err := baseBox3D(k)
  33. if err != nil {
  34. return
  35. }
  36. a := sdf.V3{X: 0, Y: 0, Z: 0.5 - k.BottomTopRatio}.Mul(k.Size).Add(sdf.V3{X: 0, Y: 0, Z: k.ClosingHeight / 2})
  37. n := sdf.V3{X: 0, Y: 0, Z: 1}
  38. top = sdf.Cut3D(box, a, n)
  39. closing, err := InnerClosing(k)
  40. if err != nil {
  41. return
  42. }
  43. top = sdf.Union3D(top,closing)
  44. return
  45. }
  46. func BottomBox3D(k *BoxParams) (bottom sdf.SDF3, err error) {
  47. box, err := baseBox3D(k)
  48. if err != nil {
  49. return
  50. }
  51. a := sdf.V3{X: 0, Y: 0, Z: 0.5 - k.BottomTopRatio}.Mul(k.Size).Add(sdf.V3{X: 0, Y: 0, Z: k.ClosingHeight / 2})
  52. n := sdf.V3{X: 0, Y: 0, Z: -1}
  53. bottom = sdf.Cut3D(box, a, n)
  54. closing, err := InnerClosing(k)
  55. scale := k.Size.Add(sdf.V3{X: k.Margin, Y: k.Margin, Z: 0}).Div(k.Size)
  56. closing = sdf.Transform3D(closing, sdf.Scale3d(scale))
  57. if err != nil {
  58. return
  59. }
  60. bottom = sdf.Difference3D(bottom,closing)
  61. return
  62. }
  63. func InnerClosing(k *BoxParams) (closing sdf.SDF3, err error) {
  64. offset := math.Tan(k.Taper) * k.ClosingHeight / 2
  65. size := sdf.V2{X: k.Size.X, Y: k.Size.Y}
  66. s1 := sdf.Box2D(size.AddScalar(k.Wall-offset), k.CornerRadius)
  67. s2 := sdf.Box2D(size.AddScalar(k.Wall+offset), k.CornerRadius)
  68. closing, err = sdf.Loft3D(s1, s2, k.ClosingHeight, 0)
  69. if err != nil {
  70. return
  71. }
  72. closing = sdf.Transform3D(closing, sdf.Translate3d(sdf.V3{0,0,0.5-k.BottomTopRatio}.Mul(k.Size)))
  73. return
  74. }
  75. func OuterClosing(k *BoxParams) (closing sdf.SDF3, err error) {
  76. s0, err := InnerClosing(k)
  77. if err != nil {
  78. return
  79. }
  80. s1, err := baseBox3D(k)
  81. if err != nil {
  82. return
  83. }
  84. a := sdf.V3{0, 0, k.ClosingHeight / 2}
  85. n := sdf.V3{0, 0, -1}
  86. s1 = sdf.Cut3D(s1, a, n)
  87. s1 = sdf.Cut3D(s1, a.Neg(), n.Neg())
  88. closing = sdf.Difference3D(s1, s0)
  89. return
  90. }