playbook_ui 13.24.0.pre.alpha.play1305drycontenttag2689 → 13.25.0.pre.alpha.PBNTR272Dropdownkitv42769

Sign up to get free protection for your applications and to get access to all the features.
Files changed (173) hide show
  1. checksums.yaml +4 -4
  2. data/app/pb_kits/playbook/_playbook.scss +2 -0
  3. data/app/pb_kits/playbook/index.js +2 -0
  4. data/app/pb_kits/playbook/pb_advanced_table/advanced_table.html.erb +1 -1
  5. data/app/pb_kits/playbook/pb_advanced_table/table_body.html.erb +1 -1
  6. data/app/pb_kits/playbook/pb_advanced_table/table_header.html.erb +1 -1
  7. data/app/pb_kits/playbook/pb_advanced_table/table_row.html.erb +1 -1
  8. data/app/pb_kits/playbook/pb_avatar_action_button/avatar_action_button.html.erb +1 -1
  9. data/app/pb_kits/playbook/pb_background/_background.tsx +7 -5
  10. data/app/pb_kits/playbook/pb_background/background.html.erb +0 -1
  11. data/app/pb_kits/playbook/pb_badge/badge.html.erb +1 -1
  12. data/app/pb_kits/playbook/pb_body/_body.tsx +1 -1
  13. data/app/pb_kits/playbook/pb_body/body.html.erb +1 -1
  14. data/app/pb_kits/playbook/pb_bread_crumbs/bread_crumb_item.html.erb +1 -1
  15. data/app/pb_kits/playbook/pb_bread_crumbs/bread_crumbs.html.erb +1 -1
  16. data/app/pb_kits/playbook/pb_button/_button.scss +1 -1
  17. data/app/pb_kits/playbook/pb_button/button.html.erb +1 -2
  18. data/app/pb_kits/playbook/pb_button_toolbar/button_toolbar.html.erb +1 -1
  19. data/app/pb_kits/playbook/pb_caption/caption.html.erb +1 -1
  20. data/app/pb_kits/playbook/pb_card/card.html.erb +1 -1
  21. data/app/pb_kits/playbook/pb_card/card_body.html.erb +1 -1
  22. data/app/pb_kits/playbook/pb_card/card_header.html.erb +1 -1
  23. data/app/pb_kits/playbook/pb_checkbox/checkbox.html.erb +1 -1
  24. data/app/pb_kits/playbook/pb_circle_icon_button/circle_icon_button.html.erb +1 -1
  25. data/app/pb_kits/playbook/pb_collapsible/collapsible.html.erb +1 -1
  26. data/app/pb_kits/playbook/pb_collapsible/collapsible_content.html.erb +1 -1
  27. data/app/pb_kits/playbook/pb_collapsible/collapsible_main.html.erb +1 -2
  28. data/app/pb_kits/playbook/pb_contact/contact.html.erb +1 -1
  29. data/app/pb_kits/playbook/pb_currency/currency.html.erb +1 -1
  30. data/app/pb_kits/playbook/pb_currency/docs/_currency_alignment_swift.md +43 -0
  31. data/app/pb_kits/playbook/pb_currency/docs/_currency_props_swift.md +12 -0
  32. data/app/pb_kits/playbook/pb_currency/docs/_currency_size_swift.md +31 -0
  33. data/app/pb_kits/playbook/pb_currency/docs/example.yml +5 -0
  34. data/app/pb_kits/playbook/pb_dashboard_value/dashboard_value.html.erb +1 -1
  35. data/app/pb_kits/playbook/pb_date/date.html.erb +1 -1
  36. data/app/pb_kits/playbook/pb_date_picker/date_picker.html.erb +1 -2
  37. data/app/pb_kits/playbook/pb_date_range_inline/date_range_inline.html.erb +1 -1
  38. data/app/pb_kits/playbook/pb_date_range_stacked/date_range_stacked.html.erb +1 -1
  39. data/app/pb_kits/playbook/pb_date_range_stacked/docs/_date_range_stacked_default_swift.md +14 -0
  40. data/app/pb_kits/playbook/pb_date_range_stacked/docs/_date_range_stacked_props_swift.md +9 -0
  41. data/app/pb_kits/playbook/pb_date_range_stacked/docs/example.yml +4 -0
  42. data/app/pb_kits/playbook/pb_date_stacked/date_stacked.html.erb +1 -1
  43. data/app/pb_kits/playbook/pb_date_time/date_time.html.erb +1 -1
  44. data/app/pb_kits/playbook/pb_date_time_stacked/date_time_stacked.html.erb +1 -1
  45. data/app/pb_kits/playbook/pb_date_year_stacked/date_year_stacked.html.erb +1 -1
  46. data/app/pb_kits/playbook/pb_detail/detail.html.erb +1 -1
  47. data/app/pb_kits/playbook/pb_dialog/_dialog.scss +4 -2
  48. data/app/pb_kits/playbook/pb_dialog/dialog.html.erb +1 -1
  49. data/app/pb_kits/playbook/pb_dialog/dialog_body.html.erb +1 -1
  50. data/app/pb_kits/playbook/pb_dialog/dialog_footer.html.erb +1 -1
  51. data/app/pb_kits/playbook/pb_dialog/dialog_header.html.erb +1 -2
  52. data/app/pb_kits/playbook/pb_dialog/docs/example.yml +1 -1
  53. data/app/pb_kits/playbook/pb_dropdown/_dropdown.scss +181 -0
  54. data/app/pb_kits/playbook/pb_dropdown/_dropdown.tsx +266 -0
  55. data/app/pb_kits/playbook/pb_dropdown/context/index.tsx +5 -0
  56. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_default.jsx +38 -0
  57. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_default.md +1 -0
  58. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_autocomplete.jsx +87 -0
  59. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_autocomplete.md +1 -0
  60. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_autocomplete_and_custom_display.jsx +102 -0
  61. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_autocomplete_and_custom_display.md +1 -0
  62. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_custom_display.jsx +104 -0
  63. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_custom_display.md +5 -0
  64. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_custom_options.jsx +63 -0
  65. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_custom_options.md +1 -0
  66. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_custom_padding.jsx +48 -0
  67. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_custom_padding.md +1 -0
  68. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_custom_trigger.jsx +77 -0
  69. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_custom_trigger.md +1 -0
  70. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_external_control.jsx +62 -0
  71. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_hook.jsx +75 -0
  72. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_label.jsx +39 -0
  73. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_label.md +1 -0
  74. data/app/pb_kits/playbook/pb_dropdown/docs/example.yml +15 -0
  75. data/app/pb_kits/playbook/pb_dropdown/docs/index.js +10 -0
  76. data/app/pb_kits/playbook/pb_dropdown/dropdown.test.jsx +207 -0
  77. data/app/pb_kits/playbook/pb_dropdown/hooks/useDropdown.tsx +17 -0
  78. data/app/pb_kits/playbook/pb_dropdown/hooks/useHandleOnKeydown.tsx +61 -0
  79. data/app/pb_kits/playbook/pb_dropdown/subcomponents/DropdownContainer.tsx +109 -0
  80. data/app/pb_kits/playbook/pb_dropdown/subcomponents/DropdownOption.tsx +116 -0
  81. data/app/pb_kits/playbook/pb_dropdown/subcomponents/DropdownTrigger.tsx +190 -0
  82. data/app/pb_kits/playbook/pb_dropdown/utilities/subComponentHelper.tsx +61 -0
  83. data/app/pb_kits/playbook/pb_file_upload/file_upload.html.erb +1 -1
  84. data/app/pb_kits/playbook/pb_filter/filter.html.erb +1 -1
  85. data/app/pb_kits/playbook/pb_fixed_confirmation_toast/fixed_confirmation_toast.html.erb +1 -1
  86. data/app/pb_kits/playbook/pb_flex/flex.html.erb +1 -1
  87. data/app/pb_kits/playbook/pb_flex/flex_item.html.erb +1 -2
  88. data/app/pb_kits/playbook/pb_form_group/form_group.html.erb +1 -1
  89. data/app/pb_kits/playbook/pb_form_pill/form_pill.html.erb +1 -1
  90. data/app/pb_kits/playbook/pb_hashtag/hashtag.html.erb +1 -1
  91. data/app/pb_kits/playbook/pb_highlight/highlight.html.erb +1 -1
  92. data/app/pb_kits/playbook/pb_home_address_street/home_address_street.html.erb +1 -1
  93. data/app/pb_kits/playbook/pb_icon_circle/_icon_circle.scss +1 -1
  94. data/app/pb_kits/playbook/pb_icon_circle/_icon_circle.tsx +1 -1
  95. data/app/pb_kits/playbook/pb_icon_circle/icon_circle.html.erb +1 -1
  96. data/app/pb_kits/playbook/pb_icon_circle/icon_circle.rb +1 -1
  97. data/app/pb_kits/playbook/pb_icon_circle/icon_circle.test.js +3 -3
  98. data/app/pb_kits/playbook/pb_icon_stat_value/icon_stat_value.html.erb +1 -1
  99. data/app/pb_kits/playbook/pb_icon_value/icon_value.html.erb +1 -1
  100. data/app/pb_kits/playbook/pb_label_pill/label_pill.html.erb +1 -1
  101. data/app/pb_kits/playbook/pb_label_value/label_value.html.erb +1 -1
  102. data/app/pb_kits/playbook/pb_layout/body.html.erb +1 -1
  103. data/app/pb_kits/playbook/pb_layout/footer.html.erb +1 -1
  104. data/app/pb_kits/playbook/pb_layout/header.html.erb +1 -1
  105. data/app/pb_kits/playbook/pb_layout/item.html.erb +1 -1
  106. data/app/pb_kits/playbook/pb_layout/layout.html.erb +1 -1
  107. data/app/pb_kits/playbook/pb_layout/sidebar.html.erb +1 -1
  108. data/app/pb_kits/playbook/pb_list/_list_item.tsx +2 -2
  109. data/app/pb_kits/playbook/pb_list/item.html.erb +1 -2
  110. data/app/pb_kits/playbook/pb_list/list.html.erb +1 -3
  111. data/app/pb_kits/playbook/pb_loading_inline/loading_inline.html.erb +1 -1
  112. data/app/pb_kits/playbook/pb_message/message.html.erb +1 -1
  113. data/app/pb_kits/playbook/pb_message/message_mention.html.erb +1 -1
  114. data/app/pb_kits/playbook/pb_multi_level_select/multi_level_select.html.erb +1 -1
  115. data/app/pb_kits/playbook/pb_multiple_users/multiple_users.html.erb +1 -1
  116. data/app/pb_kits/playbook/pb_multiple_users_stacked/multiple_users_stacked.html.erb +1 -1
  117. data/app/pb_kits/playbook/pb_nav/item.html.erb +1 -4
  118. data/app/pb_kits/playbook/pb_nav/nav.html.erb +1 -1
  119. data/app/pb_kits/playbook/pb_online_status/online_status.html.erb +1 -2
  120. data/app/pb_kits/playbook/pb_pagination/pagination.html.erb +1 -1
  121. data/app/pb_kits/playbook/pb_passphrase/passphrase.html.erb +1 -1
  122. data/app/pb_kits/playbook/pb_person/person.html.erb +1 -1
  123. data/app/pb_kits/playbook/pb_person_contact/person_contact.html.erb +1 -1
  124. data/app/pb_kits/playbook/pb_pill/pill.html.erb +1 -1
  125. data/app/pb_kits/playbook/pb_popover/popover.html.erb +1 -1
  126. data/app/pb_kits/playbook/pb_progress_pills/progress_pills.html.erb +1 -2
  127. data/app/pb_kits/playbook/pb_progress_simple/progress_simple.html.erb +1 -2
  128. data/app/pb_kits/playbook/pb_progress_step/progress_step.html.erb +1 -1
  129. data/app/pb_kits/playbook/pb_progress_step/progress_step_item.html.erb +1 -1
  130. data/app/pb_kits/playbook/pb_radio/radio.html.erb +1 -2
  131. data/app/pb_kits/playbook/pb_section_separator/_section_separator.scss +6 -2
  132. data/app/pb_kits/playbook/pb_section_separator/_section_separator_mixin.scss +11 -1
  133. data/app/pb_kits/playbook/pb_section_separator/section_separator.html.erb +1 -1
  134. data/app/pb_kits/playbook/pb_select/select.html.erb +1 -1
  135. data/app/pb_kits/playbook/pb_selectable_card/selectable_card.html.erb +1 -1
  136. data/app/pb_kits/playbook/pb_selectable_card_icon/selectable_card_icon.html.erb +1 -1
  137. data/app/pb_kits/playbook/pb_selectable_icon/selectable_icon.html.erb +1 -1
  138. data/app/pb_kits/playbook/pb_selectable_list/selectable_list.html.erb +1 -1
  139. data/app/pb_kits/playbook/pb_selectable_list/selectable_list_item.html.erb +1 -1
  140. data/app/pb_kits/playbook/pb_source/source.html.erb +1 -1
  141. data/app/pb_kits/playbook/pb_source/source.test.js +2 -2
  142. data/app/pb_kits/playbook/pb_star_rating/star_rating.html.erb +1 -1
  143. data/app/pb_kits/playbook/pb_stat_change/stat_change.html.erb +1 -1
  144. data/app/pb_kits/playbook/pb_stat_value/stat_value.html.erb +1 -1
  145. data/app/pb_kits/playbook/pb_table/table.html.erb +2 -8
  146. data/app/pb_kits/playbook/pb_table/table_body.html.erb +2 -2
  147. data/app/pb_kits/playbook/pb_table/table_cell.html.erb +2 -2
  148. data/app/pb_kits/playbook/pb_table/table_head.html.erb +2 -2
  149. data/app/pb_kits/playbook/pb_table/table_header.html.erb +2 -3
  150. data/app/pb_kits/playbook/pb_table/table_row.html.erb +2 -2
  151. data/app/pb_kits/playbook/pb_text_input/text_input.html.erb +5 -1
  152. data/app/pb_kits/playbook/pb_textarea/textarea.html.erb +1 -1
  153. data/app/pb_kits/playbook/pb_time/time.html.erb +1 -1
  154. data/app/pb_kits/playbook/pb_time_range_inline/time_range_inline.html.erb +1 -1
  155. data/app/pb_kits/playbook/pb_time_stacked/time_stacked.html.erb +1 -1
  156. data/app/pb_kits/playbook/pb_timeline/item.html.erb +1 -1
  157. data/app/pb_kits/playbook/pb_timeline/timeline.html.erb +1 -1
  158. data/app/pb_kits/playbook/pb_timestamp/timestamp.html.erb +1 -1
  159. data/app/pb_kits/playbook/pb_title/title.html.erb +1 -3
  160. data/app/pb_kits/playbook/pb_title_count/title_count.html.erb +1 -1
  161. data/app/pb_kits/playbook/pb_title_detail/title_detail.html.erb +1 -1
  162. data/app/pb_kits/playbook/pb_toggle/toggle.html.erb +1 -1
  163. data/app/pb_kits/playbook/pb_tooltip/tooltip.html.erb +1 -1
  164. data/app/pb_kits/playbook/pb_user/user.html.erb +1 -1
  165. data/app/pb_kits/playbook/pb_user_badge/user_badge.html.erb +1 -1
  166. data/app/pb_kits/playbook/pb_weekday_stacked/weekday_stacked.html.erb +1 -1
  167. data/app/pb_kits/playbook/playbook-doc.js +2 -0
  168. data/dist/menu.yml +5 -1
  169. data/dist/playbook-rails.js +5 -5
  170. data/lib/playbook/kit_base.rb +1 -1
  171. data/lib/playbook/version.rb +2 -2
  172. metadata +38 -3
  173. /data/app/pb_kits/playbook/pb_dialog/docs/{_dialog_props_table.md → _dialog_props_swift.md} +0 -0
@@ -0,0 +1,181 @@
1
+ @import "../tokens/colors";
2
+ @import "../tokens/spacing";
3
+ @import "../tokens/typography";
4
+ @import "../tokens/border_radius";
5
+ @import "../tokens/shadows";
6
+ @import "../tokens/positioning";
7
+ @import "../pb_body/body_mixins";
8
+ @import "../pb_textarea/textarea_mixin";
9
+
10
+ @keyframes fadeIn {
11
+ from {
12
+ opacity: 0;
13
+ }
14
+
15
+ to {
16
+ opacity: 1;
17
+ }
18
+ }
19
+ @keyframes fadeOut {
20
+ from {
21
+ opacity: 1;
22
+ }
23
+
24
+ to {
25
+ opacity: 0;
26
+ }
27
+ }
28
+
29
+ .pb_dropdown {
30
+ .dropdown_wrapper {
31
+ [class*=dropdown_trigger_wrapper] {
32
+ @include pb_body;
33
+ border: 1px solid $border_light;
34
+ background-color: $white;
35
+ height: 45px;
36
+ @media (hover: hover) {
37
+ &:hover,
38
+ &:active,
39
+ &:focus {
40
+ background-color: $focus_input_light;
41
+ input {
42
+ background-color: $focus_input_light;
43
+ }
44
+ }
45
+ }
46
+
47
+ .dropdown_input {
48
+ @include pb_body;
49
+ border: unset;
50
+ border-radius: $border_rad_heavier;
51
+ padding: 0;
52
+ background-color: $white;
53
+ &:focus-visible {
54
+ outline: none;
55
+ }
56
+ }
57
+ &:focus {
58
+ box-shadow: 0px 0px 0 1px $primary;
59
+ outline: unset;
60
+ transition: box-shadow 0.15s ease-in-out;
61
+ }
62
+
63
+ &[class*=_select_only] {
64
+ box-shadow: inset 0 -11px 20px rgba($primary, 0.05);
65
+ }
66
+
67
+ &[class*=_focus] {
68
+ box-shadow: 0px 0px 0 1px $primary;
69
+ transition: box-shadow 0.1s ease-in-out;
70
+ }
71
+ }
72
+
73
+ .pb_dropdown_container {
74
+ background-color: $white;
75
+ overflow: hidden;
76
+ box-shadow: $shadow_deep;
77
+ border-radius: $border_rad_heavier;
78
+ border: 1px solid $border_light;
79
+ margin-top: $space_xs;
80
+ z-index: $z_1;
81
+ width: 100%;
82
+
83
+ .pb_dropdown_option {
84
+ cursor: pointer;
85
+ &:hover {
86
+ background-color: $border_light;
87
+ }
88
+ }
89
+
90
+ .dropdown_option_focused {
91
+ background-color: $border_light;
92
+ }
93
+
94
+ .dropdown_option {
95
+ width: 100%;
96
+ }
97
+
98
+ .dropdown_option_list {
99
+ border-bottom: 1px solid $border_light;
100
+ }
101
+ .dropdown_option_selected {
102
+ background-color: $primary;
103
+ [class^="pb_body"],
104
+ [class^="pb_title_kit"] {
105
+ color: $white !important;
106
+ }
107
+ &:hover {
108
+ background-color: $primary !important;
109
+ }
110
+ }
111
+ }
112
+ .close {
113
+ display: none;
114
+ animation-name: fadeOut;
115
+ animation-duration: 150ms;
116
+ animation-timing-function: linear;
117
+ animation-fill-mode: forwards;
118
+ }
119
+
120
+ .open {
121
+ display: block;
122
+ animation-name: fadeIn;
123
+ animation-duration: 150ms;
124
+ animation-timing-function: linear;
125
+ animation-fill-mode: forwards;
126
+ }
127
+ }
128
+
129
+ &.dark {
130
+ .dropdown_wrapper {
131
+ [class*=dropdown_trigger_wrapper] {
132
+ @include pb_body_light_dark;
133
+ background-color: rgba($white, 0.1) !important;
134
+ background: none;
135
+ border-color: rgba($white, 0.15);
136
+ @media (hover: hover) {
137
+ &:hover,
138
+ &:active,
139
+ &:focus {
140
+ background-color: rgba($white, 0.05) !important;
141
+ }
142
+ }
143
+ [class^="pb_body"] {
144
+ color: $white;
145
+ }
146
+ &[class*=_select_only] {
147
+ box-shadow: inset 0 -11px 20px rgba($white, 0.05) !important;
148
+ }
149
+
150
+ .dropdown_input {
151
+ background-color: unset;
152
+ color: $white;
153
+ }
154
+ }
155
+ }
156
+ .pb_dropdown_container {
157
+ background-color: $bg_dark !important;
158
+ border-color: rgba($white, 0.15);
159
+ color: $white;
160
+ .dropdown_option_list {
161
+ border-color: rgba($white, 0.15);
162
+ }
163
+ [class^="pb_body"],
164
+ [class^="pb_title_kit"] {
165
+ color: $white !important;
166
+ }
167
+
168
+ .pb_dropdown_option {
169
+ &:hover {
170
+ background-color: $hover_dark;
171
+ }
172
+ }
173
+ .dropdown_option_focused {
174
+ background-color: $hover_dark;
175
+ }
176
+ .dropdown_option_selected {
177
+ background-color: $primary;
178
+ }
179
+ }
180
+ }
181
+ }
@@ -0,0 +1,266 @@
1
+ import React, { useState, useRef, useEffect, ReactElement } from "react";
2
+ import classnames from "classnames";
3
+ import { buildAriaProps, buildCss, buildDataProps, buildHtmlProps } from "../utilities/props";
4
+ import { globalProps } from "../utilities/globalProps";
5
+
6
+ import Body from "../pb_body/_body";
7
+ import Caption from "../pb_caption/_caption";
8
+
9
+ import DropdownContainer from "./subcomponents/DropdownContainer";
10
+ import DropdownOption from "./subcomponents/DropdownOption";
11
+ import DropdownTrigger from "./subcomponents/DropdownTrigger";
12
+ import DropdownContext from "./context";
13
+ import useDropdown from "./hooks/useDropdown";
14
+
15
+ import {
16
+ separateChildComponents,
17
+ prepareSubcomponents,
18
+ } from "./utilities/subComponentHelper";
19
+ import { GenericObject } from "../types";
20
+
21
+ type DropdownProps = {
22
+ aria?: { [key: string]: string };
23
+ autocomplete?: boolean;
24
+ children?: React.ReactChild[] | React.ReactChild | ReactElement[];
25
+ className?: string;
26
+ dark?: boolean;
27
+ data?: { [key: string]: string };
28
+ htmlOptions?: {[key: string]: string | number | boolean | (() => void)},
29
+ id?: string;
30
+ isClosed?: boolean;
31
+ label?: string;
32
+ onSelect?: (arg: GenericObject) => null;
33
+ options: GenericObject;
34
+ triggerRef?: any;
35
+ };
36
+
37
+ const Dropdown = (props: DropdownProps) => {
38
+ const {
39
+ aria = {},
40
+ autocomplete = false,
41
+ children,
42
+ className,
43
+ dark = false,
44
+ data = {},
45
+ htmlOptions = {},
46
+ id,
47
+ isClosed = true,
48
+ label,
49
+ onSelect,
50
+ options,
51
+ triggerRef
52
+ } = props;
53
+
54
+ const ariaProps = buildAriaProps(aria);
55
+ const dataProps = buildDataProps(data);
56
+ const htmlProps = buildHtmlProps(htmlOptions);
57
+ const classes = classnames(
58
+ buildCss("pb_dropdown"),
59
+ globalProps(props),
60
+ className
61
+ );
62
+
63
+ const [isDropDownClosed, setIsDropDownClosed, toggleDropdown] = useDropdown(isClosed);
64
+
65
+ const [filterItem, setFilterItem] = useState("");
66
+ const [selected, setSelected] = useState<GenericObject>({});
67
+ const [isInputFocused, setIsInputFocused] = useState(false);
68
+ const [hasTriggerSubcomponent, setHasTriggerSubcomponent] = useState(true);
69
+ const [hasContainerSubcomponent, setHasContainerSubcomponent] =
70
+ useState(true);
71
+ //state for keyboard events
72
+ const [focusedOptionIndex, setFocusedOptionIndex] = useState(-1);
73
+
74
+ const dropdownRef = useRef(null);
75
+ const inputRef = useRef(null);
76
+ const inputWrapperRef = useRef(null);
77
+ const dropdownContainerRef = useRef(null);
78
+
79
+ const { trigger, container, otherChildren } =
80
+ separateChildComponents(children);
81
+
82
+ useEffect(() => {
83
+ // Set the parent element of the trigger to relative to allow for absolute positioning of the dropdown
84
+ //Only needed for when useDropdown hook used with external trigger
85
+ if (triggerRef?.current) {
86
+ const parentElement = triggerRef.current.parentNode;
87
+ if (parentElement) {
88
+ parentElement.style.position = 'relative';
89
+ }
90
+ }
91
+ // Handle clicks outside the dropdown
92
+ const handleClickOutside = (e: MouseEvent) => {
93
+ let targetElement = e.target as HTMLElement;
94
+ let shouldClose = true;
95
+
96
+ while (targetElement && shouldClose) {
97
+ //Only needed for when useDropdown hook used with external trigger
98
+ if (targetElement.getAttribute('data-dropdown') === 'pb-dropdown-trigger') {
99
+ shouldClose = false;
100
+ }
101
+ targetElement = targetElement.parentElement as HTMLElement;
102
+ }
103
+ if (
104
+ inputWrapperRef.current && !inputWrapperRef.current.contains(e.target) &&
105
+ (dropdownContainerRef.current && !dropdownContainerRef.current.contains(e.target)) &&
106
+ shouldClose
107
+ ) {
108
+ setIsDropDownClosed(true);
109
+ setFocusedOptionIndex(-1);
110
+ setIsInputFocused(false);
111
+ }
112
+ };
113
+
114
+ window.addEventListener("click", handleClickOutside);
115
+ return () => {
116
+ window.removeEventListener("click", handleClickOutside);
117
+ };
118
+ }, []);
119
+
120
+ useEffect(() => {
121
+ setHasTriggerSubcomponent(!!trigger);
122
+ setHasContainerSubcomponent(!!container);
123
+ }, []);
124
+
125
+ // dropdown to toggle with external control
126
+ useEffect(()=> {
127
+ setIsDropDownClosed(isClosed)
128
+ },[isClosed])
129
+
130
+ const filteredOptions = options?.filter((option: GenericObject) =>
131
+ option.label.toLowerCase().includes(filterItem.toLowerCase())
132
+ );
133
+
134
+ // For keyboard accessibility: Set focus within dropdown to selected item if it exists
135
+ useEffect(() => {
136
+ if (!isDropDownClosed) {
137
+ let newIndex = 0;
138
+ if (selected && selected?.label) {
139
+ const selectedIndex = filteredOptions.findIndex((option: GenericObject) => option.label === selected.label);
140
+ if (selectedIndex >= 0) {
141
+ newIndex = selectedIndex;
142
+ }
143
+ }
144
+ setFocusedOptionIndex(newIndex);
145
+ }
146
+ }, [isDropDownClosed]);
147
+
148
+
149
+ const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => {
150
+ setFilterItem(e.target.value);
151
+ setIsDropDownClosed(false);
152
+ };
153
+
154
+ const handleOptionClick = (selectedItem: GenericObject) => {
155
+ setSelected(selectedItem);
156
+ setFilterItem("");
157
+ setIsDropDownClosed(true);
158
+ onSelect && onSelect(selectedItem);
159
+ };
160
+
161
+ const handleWrapperClick = () => {
162
+ autocomplete && inputRef.current.focus();
163
+ toggleDropdown();
164
+ };
165
+
166
+ const handleBackspace = () => {
167
+ setSelected({});
168
+ onSelect && onSelect(null);
169
+ setFocusedOptionIndex(-1);
170
+ };
171
+
172
+ const componentsToRender = prepareSubcomponents({
173
+ children,
174
+ hasTriggerSubcomponent,
175
+ hasContainerSubcomponent,
176
+ trigger,
177
+ container,
178
+ otherChildren,
179
+ dark
180
+ });
181
+
182
+
183
+ return (
184
+ <div {...ariaProps}
185
+ {...dataProps}
186
+ {...htmlProps}
187
+ className={classes}
188
+ id={id}
189
+ style={triggerRef ? { position: "absolute"} : { position: "relative"}}
190
+ >
191
+ <DropdownContext.Provider
192
+ value={{
193
+ autocomplete,
194
+ dropdownContainerRef,
195
+ filteredOptions,
196
+ filterItem,
197
+ focusedOptionIndex,
198
+ handleBackspace,
199
+ handleChange,
200
+ handleOptionClick,
201
+ handleWrapperClick,
202
+ inputRef,
203
+ inputWrapperRef,
204
+ isDropDownClosed,
205
+ isInputFocused,
206
+ options,
207
+ selected,
208
+ setFocusedOptionIndex,
209
+ setIsDropDownClosed,
210
+ setIsInputFocused,
211
+ setSelected,
212
+ toggleDropdown,
213
+ triggerRef
214
+ }}
215
+ >
216
+ {label &&
217
+ <Caption
218
+ dark={dark}
219
+ marginBottom="xs"
220
+ text={label}
221
+ />
222
+ }
223
+ <div className="dropdown_wrapper"
224
+ onBlur={() => {
225
+ // Debounce to delay the execution to prevent jumpiness in Focus state
226
+ setTimeout(() => {
227
+ if (!dropdownRef.current.contains(document.activeElement)) {
228
+ setIsInputFocused(false);
229
+ }
230
+ }, 0);
231
+ }}
232
+ onFocus={() => setIsInputFocused(true)}
233
+ ref={dropdownRef}
234
+ >
235
+ {children ? (
236
+ <>
237
+ {componentsToRender.map((component, index) => (
238
+ <React.Fragment key={index}>{component}</React.Fragment>
239
+ ))}
240
+ </>
241
+ ) : (
242
+ <>
243
+ <DropdownTrigger />
244
+ <DropdownContainer>
245
+ {options &&
246
+ options?.map((option: GenericObject) => (
247
+ <Dropdown.Option key={option.id}
248
+ option={option}
249
+ >
250
+ <Body text={option.label} />
251
+ </Dropdown.Option>
252
+ ))}
253
+ </DropdownContainer>
254
+ </>
255
+ )}
256
+ </div>
257
+ </DropdownContext.Provider>
258
+ </div>
259
+ )
260
+ };
261
+
262
+ Dropdown.Option = DropdownOption;
263
+ Dropdown.Trigger = DropdownTrigger;
264
+ Dropdown.Container = DropdownContainer;
265
+
266
+ export default Dropdown;
@@ -0,0 +1,5 @@
1
+ import { createContext } from "react";
2
+
3
+ const DropdownContext = createContext<any>({});
4
+
5
+ export default DropdownContext;
@@ -0,0 +1,38 @@
1
+ import React from 'react'
2
+ import { Dropdown } from '../../'
3
+
4
+ const DropdownDefault = (props) => {
5
+
6
+ const options = [
7
+ {
8
+ label: "United States",
9
+ value: "United States",
10
+ },
11
+ {
12
+ label: "Canada",
13
+ value: "Canada",
14
+ },
15
+ {
16
+ label: "Pakistan",
17
+ value: "Pakistan",
18
+ }
19
+ ];
20
+
21
+
22
+ return (
23
+ <div>
24
+ <Dropdown
25
+ options={options}
26
+ {...props}
27
+ >
28
+ {options.map((option) => (
29
+ <Dropdown.Option key={option.id}
30
+ option={option}
31
+ />
32
+ ))}
33
+ </Dropdown>
34
+ </div>
35
+ )
36
+ }
37
+
38
+ export default DropdownDefault
@@ -0,0 +1 @@
1
+ `options` for the Dropdown and `option` for the Dropdown.Option are the only required props for this kit. `options` must be an array of objects. Each object can contain any key/value pairs needed but 'label' and 'value' are required.
@@ -0,0 +1,87 @@
1
+ import React, { useState } from 'react'
2
+ import { Dropdown, User, Badge, FlexItem } from '../..'
3
+
4
+ const DropdownWithAutocomplete = (props) => {
5
+ // eslint-disable-next-line no-unused-vars
6
+ const [selectedOption, setSelectedOption] = useState();
7
+
8
+ const options = [
9
+ {
10
+ label: "Jasper Furniss",
11
+ value: "Jasper Furniss",
12
+ territory: "PHL",
13
+ title: "Senior UX Engineer",
14
+ id: "jasper-furniss",
15
+ status: "Offline"
16
+ },
17
+ {
18
+ label: "Ramon Ruiz",
19
+ value: "Ramon Ruiz",
20
+ territory: "PHL",
21
+ title: "Senior UX Desinger",
22
+ id: "ramon-ruiz",
23
+ status: "Away"
24
+ },
25
+ {
26
+ label: "Jason Cypret",
27
+ value: "Jason Cypret",
28
+ territory: "PHL",
29
+ title: "VP of User Experience",
30
+ id: "jason-cypret",
31
+ status: "Online"
32
+ },
33
+ {
34
+ label: "Courtney Long",
35
+ value: "Courtney Long",
36
+ territory: "PHL",
37
+ title: "UX Design Mentor",
38
+ id: "courtney-long",
39
+ status: "Online"
40
+ }
41
+ ];
42
+
43
+
44
+ return (
45
+ <div>
46
+ <Dropdown autocomplete
47
+ onSelect={(selectedItem) => setSelectedOption(selectedItem)}
48
+ options={options}
49
+ {...props}
50
+ >
51
+ {options.map((option) => (
52
+ <Dropdown.Option key={option.id}
53
+ option={option}
54
+ >
55
+ <>
56
+ <FlexItem>
57
+ <User
58
+ align="left"
59
+ avatar
60
+ name={option.label}
61
+ orientation="horizontal"
62
+ territory={option.territory}
63
+ title={option.title}
64
+ />
65
+ </FlexItem>
66
+ <FlexItem>
67
+ <Badge
68
+ rounded
69
+ text={option.status}
70
+ variant={`${
71
+ option.status === "Offline"
72
+ ? "neutral"
73
+ : option.status === "Online"
74
+ ? "success"
75
+ : "warning"
76
+ }`}
77
+ />
78
+ </FlexItem>
79
+ </>
80
+ </Dropdown.Option>
81
+ ))}
82
+ </Dropdown>
83
+ </div>
84
+ )
85
+ }
86
+
87
+ export default DropdownWithAutocomplete
@@ -0,0 +1 @@
1
+ The `autocomplete` prop can be used to add autocomplete or typeahead functionality to the Dropdown's default Trigger. This prop is set to 'false' by default.
@@ -0,0 +1,102 @@
1
+ import React, { useState } from 'react'
2
+ import { Dropdown, User, Badge, FlexItem, Avatar } from '../..'
3
+
4
+ const DropdownWithAutocompleteAndCustomDisplay = (props) => {
5
+ // eslint-disable-next-line no-unused-vars
6
+ const [selectedOption, setSelectedOption] = useState();
7
+
8
+ const options = [
9
+ {
10
+ label: "Jasper Furniss",
11
+ value: "Jasper Furniss",
12
+ territory: "PHL",
13
+ title: "Senior UX Engineer",
14
+ id: "jasper-furniss",
15
+ status: "Offline"
16
+ },
17
+ {
18
+ label: "Ramon Ruiz",
19
+ value: "Ramon Ruiz",
20
+ territory: "PHL",
21
+ title: "Senior UX Desinger",
22
+ id: "ramon-ruiz",
23
+ status: "Away"
24
+ },
25
+ {
26
+ label: "Jason Cypret",
27
+ value: "Jason Cypret",
28
+ territory: "PHL",
29
+ title: "VP of User Experience",
30
+ id: "jason-cypret",
31
+ status: "Online"
32
+ },
33
+ {
34
+ label: "Courtney Long",
35
+ value: "Courtney Long",
36
+ territory: "PHL",
37
+ title: "UX Design Mentor",
38
+ id: "courtney-long",
39
+ status: "Online"
40
+ }
41
+ ];
42
+
43
+ const CustomDisplay = () => {
44
+ return (
45
+ <>
46
+ {
47
+ selectedOption && (
48
+ <Avatar
49
+ name={selectedOption.label}
50
+ size="xs"
51
+ />
52
+ )
53
+ }
54
+ </>
55
+ )
56
+ };
57
+
58
+ return (
59
+ <div>
60
+ <Dropdown autocomplete
61
+ onSelect={(selectedItem) => setSelectedOption(selectedItem)}
62
+ options={options}
63
+ {...props}
64
+ >
65
+ <Dropdown.Trigger customDisplay={<CustomDisplay/>} />
66
+ {options.map((option) => (
67
+ <Dropdown.Option key={option.id}
68
+ option={option}
69
+ >
70
+ <>
71
+ <FlexItem>
72
+ <User
73
+ align="left"
74
+ avatar
75
+ name={option.label}
76
+ orientation="horizontal"
77
+ territory={option.territory}
78
+ title={option.title}
79
+ />
80
+ </FlexItem>
81
+ <FlexItem>
82
+ <Badge
83
+ rounded
84
+ text={option.status}
85
+ variant={`${
86
+ option.status === "Offline"
87
+ ? "neutral"
88
+ : option.status === "Online"
89
+ ? "success"
90
+ : "warning"
91
+ }`}
92
+ />
93
+ </FlexItem>
94
+ </>
95
+ </Dropdown.Option>
96
+ ))}
97
+ </Dropdown>
98
+ </div>
99
+ )
100
+ }
101
+
102
+ export default DropdownWithAutocompleteAndCustomDisplay
@@ -0,0 +1 @@
1
+ `autocomplete` prop can also be used in conjunction with the `customDisplay` prop.