components.go 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. package discordgo
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. )
  6. // ComponentType is type of component.
  7. type ComponentType uint
  8. // MessageComponent types.
  9. const (
  10. ActionsRowComponent ComponentType = 1
  11. ButtonComponent ComponentType = 2
  12. SelectMenuComponent ComponentType = 3
  13. TextInputComponent ComponentType = 4
  14. )
  15. // MessageComponent is a base interface for all message components.
  16. type MessageComponent interface {
  17. json.Marshaler
  18. Type() ComponentType
  19. }
  20. type unmarshalableMessageComponent struct {
  21. MessageComponent
  22. }
  23. // UnmarshalJSON is a helper function to unmarshal MessageComponent object.
  24. func (umc *unmarshalableMessageComponent) UnmarshalJSON(src []byte) error {
  25. var v struct {
  26. Type ComponentType `json:"type"`
  27. }
  28. err := json.Unmarshal(src, &v)
  29. if err != nil {
  30. return err
  31. }
  32. switch v.Type {
  33. case ActionsRowComponent:
  34. umc.MessageComponent = &ActionsRow{}
  35. case ButtonComponent:
  36. umc.MessageComponent = &Button{}
  37. case SelectMenuComponent:
  38. umc.MessageComponent = &SelectMenu{}
  39. case TextInputComponent:
  40. umc.MessageComponent = &TextInput{}
  41. default:
  42. return fmt.Errorf("unknown component type: %d", v.Type)
  43. }
  44. return json.Unmarshal(src, umc.MessageComponent)
  45. }
  46. func MessageComponentFromJSON(b []byte) (MessageComponent, error) {
  47. var u unmarshalableMessageComponent
  48. err := u.UnmarshalJSON(b)
  49. if err != nil {
  50. return nil, fmt.Errorf("failed to unmarshal into MessageComponent: %w", err)
  51. }
  52. return u.MessageComponent, nil
  53. }
  54. // ActionsRow is a container for components within one row.
  55. type ActionsRow struct {
  56. Components []MessageComponent `json:"components"`
  57. }
  58. // MarshalJSON is a method for marshaling ActionsRow to a JSON object.
  59. func (r ActionsRow) MarshalJSON() ([]byte, error) {
  60. type actionsRow ActionsRow
  61. return json.Marshal(struct {
  62. actionsRow
  63. Type ComponentType `json:"type"`
  64. }{
  65. actionsRow: actionsRow(r),
  66. Type: r.Type(),
  67. })
  68. }
  69. // UnmarshalJSON is a helper function to unmarshal Actions Row.
  70. func (r *ActionsRow) UnmarshalJSON(data []byte) error {
  71. var v struct {
  72. RawComponents []unmarshalableMessageComponent `json:"components"`
  73. }
  74. err := json.Unmarshal(data, &v)
  75. if err != nil {
  76. return err
  77. }
  78. r.Components = make([]MessageComponent, len(v.RawComponents))
  79. for i, v := range v.RawComponents {
  80. r.Components[i] = v.MessageComponent
  81. }
  82. return err
  83. }
  84. // Type is a method to get the type of a component.
  85. func (r ActionsRow) Type() ComponentType {
  86. return ActionsRowComponent
  87. }
  88. // ButtonStyle is style of button.
  89. type ButtonStyle uint
  90. // Button styles.
  91. const (
  92. // PrimaryButton is a button with blurple color.
  93. PrimaryButton ButtonStyle = 1
  94. // SecondaryButton is a button with grey color.
  95. SecondaryButton ButtonStyle = 2
  96. // SuccessButton is a button with green color.
  97. SuccessButton ButtonStyle = 3
  98. // DangerButton is a button with red color.
  99. DangerButton ButtonStyle = 4
  100. // LinkButton is a special type of button which navigates to a URL. Has grey color.
  101. LinkButton ButtonStyle = 5
  102. )
  103. // ComponentEmoji represents button emoji, if it does have one.
  104. type ComponentEmoji struct {
  105. Name string `json:"name,omitempty"`
  106. ID string `json:"id,omitempty"`
  107. Animated bool `json:"animated,omitempty"`
  108. }
  109. // Button represents button component.
  110. type Button struct {
  111. Label string `json:"label"`
  112. Style ButtonStyle `json:"style"`
  113. Disabled bool `json:"disabled"`
  114. Emoji ComponentEmoji `json:"emoji"`
  115. // NOTE: Only button with LinkButton style can have link. Also, URL is mutually exclusive with CustomID.
  116. URL string `json:"url,omitempty"`
  117. CustomID string `json:"custom_id,omitempty"`
  118. }
  119. // MarshalJSON is a method for marshaling Button to a JSON object.
  120. func (b Button) MarshalJSON() ([]byte, error) {
  121. type button Button
  122. if b.Style == 0 {
  123. b.Style = PrimaryButton
  124. }
  125. return json.Marshal(struct {
  126. button
  127. Type ComponentType `json:"type"`
  128. }{
  129. button: button(b),
  130. Type: b.Type(),
  131. })
  132. }
  133. // Type is a method to get the type of a component.
  134. func (Button) Type() ComponentType {
  135. return ButtonComponent
  136. }
  137. // SelectMenuOption represents an option for a select menu.
  138. type SelectMenuOption struct {
  139. Label string `json:"label,omitempty"`
  140. Value string `json:"value"`
  141. Description string `json:"description"`
  142. Emoji ComponentEmoji `json:"emoji"`
  143. // Determines whenever option is selected by default or not.
  144. Default bool `json:"default"`
  145. }
  146. // SelectMenu represents select menu component.
  147. type SelectMenu struct {
  148. CustomID string `json:"custom_id,omitempty"`
  149. // The text which will be shown in the menu if there's no default options or all options was deselected and component was closed.
  150. Placeholder string `json:"placeholder"`
  151. // This value determines the minimal amount of selected items in the menu.
  152. MinValues int `json:"min_values,omitempty"`
  153. // This value determines the maximal amount of selected items in the menu.
  154. // If MaxValues or MinValues are greater than one then the user can select multiple items in the component.
  155. MaxValues int `json:"max_values,omitempty"`
  156. Options []SelectMenuOption `json:"options"`
  157. }
  158. // Type is a method to get the type of a component.
  159. func (SelectMenu) Type() ComponentType {
  160. return SelectMenuComponent
  161. }
  162. // MarshalJSON is a method for marshaling SelectMenu to a JSON object.
  163. func (m SelectMenu) MarshalJSON() ([]byte, error) {
  164. type selectMenu SelectMenu
  165. return json.Marshal(struct {
  166. selectMenu
  167. Type ComponentType `json:"type"`
  168. }{
  169. selectMenu: selectMenu(m),
  170. Type: m.Type(),
  171. })
  172. }
  173. // TextInput represents text input component.
  174. type TextInput struct {
  175. CustomID string `json:"custom_id"`
  176. Label string `json:"label"`
  177. Style TextInputStyle `json:"style"`
  178. Placeholder string `json:"placeholder,omitempty"`
  179. Value string `json:"value,omitempty"`
  180. Required bool `json:"required,omitempty"`
  181. MinLength int `json:"min_length,omitempty"`
  182. MaxLength int `json:"max_length,omitempty"`
  183. }
  184. // Type is a method to get the type of a component.
  185. func (TextInput) Type() ComponentType {
  186. return TextInputComponent
  187. }
  188. // MarshalJSON is a method for marshaling TextInput to a JSON object.
  189. func (m TextInput) MarshalJSON() ([]byte, error) {
  190. type inputText TextInput
  191. return json.Marshal(struct {
  192. inputText
  193. Type ComponentType `json:"type"`
  194. }{
  195. inputText: inputText(m),
  196. Type: m.Type(),
  197. })
  198. }
  199. // TextInputStyle is style of text in TextInput component.
  200. type TextInputStyle uint
  201. // Text styles
  202. const (
  203. TextInputShort TextInputStyle = 1
  204. TextInputParagraph TextInputStyle = 2
  205. )