playbook_ui 14.17.0.pre.rc.0 → 14.17.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (78) hide show
  1. checksums.yaml +4 -4
  2. data/app/pb_kits/playbook/pb_advanced_table/Utilities/types.ts +1 -1
  3. data/app/pb_kits/playbook/pb_advanced_table/_advanced_table.scss +70 -0
  4. data/app/pb_kits/playbook/pb_advanced_table/_advanced_table.tsx +83 -2
  5. data/app/pb_kits/playbook/pb_advanced_table/advanced_table.test.jsx +20 -7
  6. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_fullscreen.jsx +90 -0
  7. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_fullscreen.md +3 -0
  8. data/app/pb_kits/playbook/pb_advanced_table/docs/example.yml +2 -1
  9. data/app/pb_kits/playbook/pb_advanced_table/docs/index.js +2 -1
  10. data/app/pb_kits/playbook/pb_collapsible/__snapshots__/collapsible.test.js.snap +14 -7
  11. data/app/pb_kits/playbook/pb_contact/contact.test.js +7 -7
  12. data/app/pb_kits/playbook/pb_date_picker/_date_picker.tsx +34 -34
  13. data/app/pb_kits/playbook/pb_date_picker/date_picker.html.erb +2 -2
  14. data/app/pb_kits/playbook/pb_date_picker/date_picker_helper.ts +16 -0
  15. data/app/pb_kits/playbook/pb_date_picker/docs/_date_picker_inline.html.erb +0 -11
  16. data/app/pb_kits/playbook/pb_date_picker/docs/_date_picker_inline.jsx +0 -7
  17. data/app/pb_kits/playbook/pb_date_picker/sass_partials/_inline_styles.scss +28 -24
  18. data/app/pb_kits/playbook/pb_date_range_inline/date_range_inline.test.js +2 -2
  19. data/app/pb_kits/playbook/pb_date_range_stacked/date_range_stacked.test.js +1 -1
  20. data/app/pb_kits/playbook/pb_draggable/context/index.tsx +58 -17
  21. data/app/pb_kits/playbook/pb_draggable/draggable.test.jsx +3 -3
  22. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_autocomplete.jsx +6 -6
  23. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_autocomplete_and_custom_display.jsx +6 -6
  24. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_custom_display.jsx +6 -6
  25. data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_custom_display_rails.html.erb +8 -8
  26. data/app/pb_kits/playbook/pb_dropdown/dropdown.test.jsx +3 -3
  27. data/app/pb_kits/playbook/pb_filter/Filter/CurrentFilters.tsx +3 -4
  28. data/app/pb_kits/playbook/pb_filter/Filter/SortMenu.tsx +2 -3
  29. data/app/pb_kits/playbook/pb_form/pb_form_validation.js +1 -1
  30. data/app/pb_kits/playbook/pb_form_group/_form_group.scss +22 -0
  31. data/app/pb_kits/playbook/pb_icon/icon.test.js +9 -9
  32. data/app/pb_kits/playbook/pb_icon_circle/icon_circle.test.js +1 -1
  33. data/app/pb_kits/playbook/pb_icon_stat_value/icon_stat_value.test.js +1 -1
  34. data/app/pb_kits/playbook/pb_icon_value/icon_value.test.js +1 -1
  35. data/app/pb_kits/playbook/pb_label_value/label_value.test.js +1 -1
  36. data/app/pb_kits/playbook/pb_layout/_layout.scss +58 -0
  37. data/app/pb_kits/playbook/pb_layout/_layout.tsx +20 -7
  38. data/app/pb_kits/playbook/pb_layout/docs/_layout_bracket.jsx +190 -0
  39. data/app/pb_kits/playbook/pb_layout/docs/_layout_bracket.md +5 -0
  40. data/app/pb_kits/playbook/pb_layout/docs/example.yml +1 -0
  41. data/app/pb_kits/playbook/pb_layout/docs/index.js +1 -0
  42. data/app/pb_kits/playbook/pb_layout/layout.test.js +4 -0
  43. data/app/pb_kits/playbook/pb_layout/subcomponents/_game.tsx +90 -0
  44. data/app/pb_kits/playbook/pb_layout/subcomponents/_round.tsx +57 -0
  45. data/app/pb_kits/playbook/pb_lightbox/hooks/useVisibility.js +1 -1
  46. data/app/pb_kits/playbook/pb_link/link.test.jsx +2 -2
  47. data/app/pb_kits/playbook/pb_multi_level_select/_multi_level_select.tsx +2 -1
  48. data/app/pb_kits/playbook/pb_nav/_nav_item.test.js +5 -3
  49. data/app/pb_kits/playbook/pb_section_separator/docs/_section_separator_vertical.md +1 -0
  50. data/app/pb_kits/playbook/pb_stat_change/stat_change.test.js +8 -4
  51. data/app/pb_kits/playbook/pb_table/styles/_striped.scss +3 -3
  52. data/app/pb_kits/playbook/pb_tooltip/index.js +183 -56
  53. data/app/pb_kits/playbook/pb_tooltip/tooltip.html.erb +2 -5
  54. data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_with_highlight.jsx +4 -4
  55. data/app/pb_kits/playbook/pb_typeahead/index.ts +2 -2
  56. data/app/pb_kits/playbook/pb_typeahead/typeahead.html.erb +2 -5
  57. data/app/pb_kits/playbook/pb_user/user.html.erb +1 -6
  58. data/app/pb_kits/playbook/pb_user_badge/user_badge.html.erb +1 -6
  59. data/app/pb_kits/playbook/utilities/globalProps.ts +1 -1
  60. data/app/pb_kits/playbook/utilities/object.test.js +149 -1
  61. data/app/pb_kits/playbook/utilities/object.ts +124 -42
  62. data/dist/chunks/{_typeahead-N-EFroAX.js → _typeahead-CPM091Hj.js} +3 -3
  63. data/dist/chunks/_weekday_stacked-BzIANIYX.js +45 -0
  64. data/dist/chunks/lazysizes-DHz07jlL.js +1 -0
  65. data/dist/chunks/lib-Bg2KFzpz.js +29 -0
  66. data/dist/chunks/{pb_form_validation-DMajaRt3.js → pb_form_validation-BiHyZedy.js} +1 -1
  67. data/dist/chunks/vendor.js +1 -1
  68. data/dist/playbook-doc.js +1 -1
  69. data/dist/playbook-rails-react-bindings.js +1 -1
  70. data/dist/playbook-rails.js +1 -1
  71. data/dist/playbook.css +1 -1
  72. data/lib/playbook/kit_base.rb +4 -4
  73. data/lib/playbook/version.rb +1 -1
  74. metadata +14 -8
  75. data/app/pb_kits/playbook/pb_tooltip/floating_ui.js +0 -282
  76. data/dist/chunks/_weekday_stacked-Db780bKR.js +0 -45
  77. data/dist/chunks/lazysizes-B7xYodB-.js +0 -1
  78. data/dist/chunks/lib-Co5y3V4K.js +0 -29
@@ -16,7 +16,7 @@ describe("Icon Kit", () => {
16
16
  )
17
17
 
18
18
  const kit = screen.getByTestId(testId)
19
- expect(kit).toHaveClass("fa-user pb_icon_kit fa-fw far")
19
+ expect(kit).toHaveClass("pb_custom_icon svg-inline--fa svg_fw")
20
20
  })
21
21
 
22
22
  test("renders rotate prop", () => {[
@@ -31,7 +31,7 @@ describe("Icon Kit", () => {
31
31
  )
32
32
 
33
33
  const kit = screen.getByTestId(testId)
34
- expect(kit).toHaveClass(`fa-user pb_icon_kit fa-fw fa-rotate-${rotateProp} far`)
34
+ expect(kit).toHaveClass(`pb_custom_icon svg-inline--fa rotate_${rotateProp} svg_fw`)
35
35
 
36
36
  cleanup()
37
37
  })
@@ -48,7 +48,7 @@ describe("Icon Kit", () => {
48
48
  )
49
49
 
50
50
  const kit = screen.getByTestId(testId)
51
- expect(kit).toHaveClass("fa-user pb_icon_kit fa-fw fa-flip-horizontal far")
51
+ expect(kit).toHaveClass("pb_custom_icon svg-inline--fa flip_horizontal svg_fw")
52
52
  })
53
53
 
54
54
 
@@ -63,7 +63,7 @@ describe("Icon Kit", () => {
63
63
  )
64
64
 
65
65
  const kit = screen.getByTestId(testId)
66
- expect(kit).toHaveClass("fa-spinner pb_icon_kit fa-fw fa-spin far")
66
+ expect(kit).toHaveClass("pb_custom_icon svg-inline--fa spin svg_fw")
67
67
  })
68
68
 
69
69
  test("renders pull icon", () => {
@@ -77,7 +77,7 @@ describe("Icon Kit", () => {
77
77
  )
78
78
 
79
79
  const kit = screen.getByTestId(testId)
80
- expect(kit).toHaveClass("fa-arrow-left pb_icon_kit fa-fw fa-pull-left far")
80
+ expect(kit).toHaveClass("pb_custom_icon svg-inline--fa svg_fw pull_left")
81
81
  })
82
82
 
83
83
  test("renders pull icon", () => {
@@ -91,7 +91,7 @@ describe("Icon Kit", () => {
91
91
  )
92
92
 
93
93
  const kit = screen.getByTestId(testId)
94
- expect(kit).toHaveClass("fa-arrow-left pb_icon_kit fa-fw fa-pull-left far")
94
+ expect(kit).toHaveClass("pb_custom_icon svg-inline--fa svg_fw pull_left")
95
95
  })
96
96
 
97
97
  test("renders border around icon", () => {
@@ -105,7 +105,7 @@ describe("Icon Kit", () => {
105
105
  )
106
106
 
107
107
  const kit = screen.getByTestId(testId)
108
- expect(kit).toHaveClass("fa-user pb_icon_kit fa-border fa-fw far")
108
+ expect(kit).toHaveClass("pb_custom_icon svg-inline--fa svg_border svg_fw")
109
109
  })
110
110
 
111
111
  test("renders size prop", () => {
@@ -132,7 +132,7 @@ describe("Icon Kit", () => {
132
132
  )
133
133
 
134
134
  const kit = screen.getByTestId(testId)
135
- expect(kit).toHaveClass(`pb_icon_kit fa-user fa-fw fa-${sizeProp} far`)
135
+ expect(kit).toHaveClass(`pb_custom_icon svg-inline--fa svg_${sizeProp} svg_fw`)
136
136
 
137
137
  cleanup()
138
138
  })
@@ -149,7 +149,7 @@ describe("Icon Kit", () => {
149
149
  )
150
150
 
151
151
  const kit = screen.getByTestId(testId)
152
- expect(kit).toHaveClass("fa-user pb_icon_kit fa-fw fas")
152
+ expect(kit).toHaveClass("pb_custom_icon svg-inline--fa svg_fw")
153
153
  })
154
154
 
155
155
  test("renders with color prop", () => {
@@ -29,7 +29,7 @@ describe("IconCircle Kit", () => {
29
29
  )
30
30
 
31
31
  const kit = screen.getByTestId(testId)
32
- const icon = kit.querySelector('.pb_icon_kit')
32
+ const icon = kit.querySelector('.pb_custom_icon')
33
33
  expect(icon).toBeInTheDocument()
34
34
  })
35
35
 
@@ -33,7 +33,7 @@ describe("IconStatValue Kit", () => {
33
33
  )
34
34
 
35
35
  const kit = screen.getByTestId(testId)
36
- const icon = kit.querySelector(".fa-lightbulb-on.pb_icon_kit.fa-fw")
36
+ const icon = kit.querySelector(".pb_custom_icon")
37
37
  expect(icon).toBeInTheDocument()
38
38
  })
39
39
 
@@ -29,7 +29,7 @@ describe("IconValue Kit", () => {
29
29
  )
30
30
 
31
31
  const kit = screen.getByTestId(testId)
32
- const icon = kit.querySelector(".fa-clipboard.pb_icon_kit.fa-fw")
32
+ const icon = kit.querySelector(".pb_custom_icon")
33
33
  expect(icon).toBeInTheDocument()
34
34
  })
35
35
 
@@ -56,7 +56,7 @@ describe("LabelValue Kit", () => {
56
56
  />
57
57
  )
58
58
  const kit = screen.getByTestId(testId)
59
- const icon = kit.querySelector(".fa-truck.pb_icon_kit.fa-fw")
59
+ const icon = kit.querySelector(".pb_custom_icon")
60
60
  expect(icon).toBeInTheDocument()
61
61
  })
62
62
 
@@ -10,6 +10,8 @@ $list-header-height: $space_lg;
10
10
  $list-footer-height: $space_lg;
11
11
  $list-border-radius: $border_rad_lighter;
12
12
  $card-border-radius: $border_rad_lightest;
13
+ $bracket-connector-width: 45px;
14
+ $bracket-border-width: 4px;
13
15
 
14
16
 
15
17
  [class^=pb_layout_kit] {
@@ -168,6 +170,62 @@ $card-border-radius: $border_rad_lightest;
168
170
  }
169
171
  }
170
172
 
173
+ &[class*=_bracket]{
174
+ display: flex;
175
+ overflow-x: scroll;
176
+
177
+ div.layout_round {
178
+ display: flex;
179
+ flex-direction: column;
180
+ justify-content: space-around;
181
+ flex-grow: 1;
182
+ }
183
+
184
+ .connector_container {
185
+ display: flex;
186
+ flex-direction: column;
187
+ justify-content: space-around;
188
+ }
189
+ .right_connector {
190
+ border-top: $bracket-border-width solid $border_light;
191
+ width: $bracket-connector-width;
192
+ margin-left: $bracket-connector-width;
193
+ }
194
+
195
+ .layout_round .layout_game {
196
+ position: relative;
197
+ }
198
+
199
+ .half_box {
200
+ content: '';
201
+ position: absolute;
202
+ top: calc(50% - $bracket-border-width / 2);
203
+ height: calc(100% + $bracket-border-width);
204
+ width: $bracket-connector-width;
205
+ right: -$bracket-connector-width;
206
+ border-top-right-radius: $border_radius_lg;
207
+ border-bottom-right-radius: $border_radius_lg;
208
+ border-top: $bracket-border-width solid $border_light;
209
+ border-bottom: $bracket-border-width solid $border_light;
210
+ border-right: $bracket-border-width solid $border_light;
211
+ }
212
+
213
+ .layout_round_label {
214
+ display: none;
215
+ }
216
+
217
+ @media (max-width: $screen-md-max) {
218
+ flex-direction: column;
219
+ .layout_round_label {
220
+ display: block;
221
+ }
222
+ .layout_round .layout_game::after,
223
+ .connector_container,
224
+ .half_box {
225
+ display: none !important;
226
+ }
227
+ }
228
+ }
171
229
 
172
230
  &[class*=_sidebar]{
173
231
  $layout_sizes: (
@@ -4,6 +4,9 @@ import { buildAriaProps, buildCss, buildDataProps, buildHtmlProps } from '../uti
4
4
 
5
5
  import { GlobalProps, globalProps, globalInlineProps } from '../utilities/globalProps'
6
6
 
7
+ import { Round, RoundLabel } from "./subcomponents/_round";
8
+ import Game from "./subcomponents/_game";
9
+
7
10
  type LayoutPropTypes = {
8
11
  aria?: {[key: string]: string},
9
12
  children?: React.ReactChild[] | React.ReactChild,
@@ -18,7 +21,7 @@ type LayoutPropTypes = {
18
21
  size?: "xs" | "sm" | "md" | "base" | "lg" | "xl",
19
22
  variant?: "light" | "dark" | "gradient",
20
23
  transparent?: boolean,
21
- layout?: "sidebar" | "collection" | "kanban" | "content" | "masonry",
24
+ layout?: "sidebar" | "collection" | "kanban" | "content" | "masonry" | "bracket",
22
25
  } & GlobalProps
23
26
 
24
27
  type LayoutSideProps = {
@@ -140,7 +143,7 @@ const Layout = (props: LayoutPropTypes) => {
140
143
  const htmlProps = buildHtmlProps(htmlOptions)
141
144
 
142
145
  const layoutCss =
143
- layout == 'collection'
146
+ (layout == 'collection' || layout == 'bracket')
144
147
  ? `pb_layout_kit_${layout}`
145
148
  : layout == 'kanban'
146
149
  ? `pb_layout_kit_${layout}${responsiveClass}`
@@ -151,11 +154,9 @@ const Layout = (props: LayoutPropTypes) => {
151
154
  })
152
155
 
153
156
  const layoutCollapseCss =
154
- layout == 'collection'
157
+ (layout == 'collection' || layout == 'kanban' || layout == 'bracket')
155
158
  ? ''
156
- : layout == 'kanban'
157
- ? ''
158
- : buildCss('layout', position, 'collapse', collapse)
159
+ : buildCss('layout', position, 'collapse', collapse)
159
160
 
160
161
  const layoutChildren = React.Children.toArray(children)
161
162
 
@@ -175,6 +176,15 @@ const Layout = (props: LayoutPropTypes) => {
175
176
  (child: React.ReactElement & {type: {displayName: string}}) => child.type?.displayName !== 'Side'
176
177
  )
177
178
 
179
+ const numberOfRounds = Array.isArray(nonSideChildren) ? React.Children.toArray(children).filter(
180
+ (child) => {
181
+ return (child as React.ReactElement).type === Layout.Round;
182
+ }
183
+ ).length : 0
184
+ const bracketChildren = nonSideChildren.map(child =>
185
+ React.isValidElement(child) ? React.cloneElement(child, { numberOfRounds }) : child
186
+ )
187
+
178
188
  const filteredProps = {...props}
179
189
  delete filteredProps?.position
180
190
 
@@ -196,7 +206,7 @@ const Layout = (props: LayoutPropTypes) => {
196
206
  style={dynamicInlineProps}
197
207
  >
198
208
  {subComponentTags('Side')}
199
- {nonSideChildren}
209
+ {layout === 'bracket' ? bracketChildren : nonSideChildren}
200
210
  </div>
201
211
  )
202
212
  }
@@ -206,5 +216,8 @@ Layout.Body = Body
206
216
  Layout.Item = Item
207
217
  Layout.Header = Header
208
218
  Layout.Footer = Footer
219
+ Layout.Round = Round
220
+ Layout.Game = Game
221
+ Layout.RoundLabel = RoundLabel
209
222
 
210
223
  export default Layout
@@ -0,0 +1,190 @@
1
+ import React from 'react'
2
+
3
+ import Layout from '../../pb_layout/_layout'
4
+ import Flex from '../../pb_flex/_flex'
5
+ import Body from '../../pb_body/_body'
6
+ import Caption from '../../pb_caption/_caption'
7
+ import SectionSeparator from '../../pb_section_separator/_section_separator'
8
+
9
+ const LayoutBracket = () => {
10
+ return (
11
+ <div>
12
+ <Layout
13
+ layout="bracket"
14
+ >
15
+ <Layout.RoundLabel>
16
+ <Caption>Wild Card</Caption>
17
+ <SectionSeparator marginY="sm"/>
18
+ </Layout.RoundLabel>
19
+ <Layout.Round marginBottom={{ xs: "md", sm: "md" }}>
20
+ <Layout.Game>
21
+ <Flex justify="between">
22
+ <Body>Packers</Body>
23
+ <Body>10</Body>
24
+ </Flex>
25
+ <Flex justify="between">
26
+ <Body><strong>Eagles</strong></Body>
27
+ <Body>22</Body>
28
+ </Flex>
29
+ </Layout.Game>
30
+ <Layout.Game>
31
+ <Flex justify="between">
32
+ <Body>Vikings</Body>
33
+ <Body>9</Body>
34
+ </Flex>
35
+ <Flex justify="between">
36
+ <Body><strong>Rams</strong></Body>
37
+ <Body>27</Body>
38
+ </Flex>
39
+ </Layout.Game>
40
+ <Layout.Game>
41
+ <Flex justify="between">
42
+ <Body><strong>Lions</strong></Body>
43
+ </Flex>
44
+ <Flex justify="between">
45
+ <Body>BYE</Body>
46
+ </Flex>
47
+ </Layout.Game>
48
+ <Layout.Game>
49
+ <Flex justify="between">
50
+ <Body><strong>Commanders</strong></Body>
51
+ <Body>23</Body>
52
+ </Flex>
53
+ <Flex justify="between">
54
+ <Body>Buccaneers</Body>
55
+ <Body>20</Body>
56
+ </Flex>
57
+ </Layout.Game>
58
+ <Layout.Game>
59
+ <Flex justify="between">
60
+ <Body><strong>Chiefs</strong></Body>
61
+ </Flex>
62
+ <Flex justify="between">
63
+ <Body>BYE</Body>
64
+ </Flex>
65
+ </Layout.Game>
66
+ <Layout.Game>
67
+ <Flex justify="between">
68
+ <Body>Chargers</Body>
69
+ <Body>12</Body>
70
+ </Flex>
71
+ <Flex justify="between">
72
+ <Body><strong>Texans</strong></Body>
73
+ <Body>32</Body>
74
+ </Flex>
75
+ </Layout.Game>
76
+ <Layout.Game>
77
+ <Flex justify="between">
78
+ <Body>Broncos</Body>
79
+ <Body>7</Body>
80
+ </Flex>
81
+ <Flex justify="between">
82
+ <Body><strong>Bills</strong></Body>
83
+ <Body>31</Body>
84
+ </Flex>
85
+ </Layout.Game>
86
+ <Layout.Game>
87
+ <Flex justify="between">
88
+ <Body>Steelers</Body>
89
+ <Body>14</Body>
90
+ </Flex>
91
+ <Flex justify="between">
92
+ <Body><strong>Ravens</strong></Body>
93
+ <Body>28</Body>
94
+ </Flex>
95
+ </Layout.Game>
96
+ </Layout.Round>
97
+ <Layout.RoundLabel>
98
+ <Caption>Divisional</Caption>
99
+ <SectionSeparator marginY="sm"/>
100
+ </Layout.RoundLabel>
101
+ <Layout.Round marginBottom={{ xs: "md", sm: "md" }}>
102
+ <Layout.Game>
103
+ <Flex justify="between">
104
+ <Body>Rams</Body>
105
+ <Body>22</Body>
106
+ </Flex>
107
+ <Flex justify="between">
108
+ <Body><strong>Eagles</strong></Body>
109
+ <Body>28</Body>
110
+ </Flex>
111
+ </Layout.Game>
112
+ <Layout.Game>
113
+ <Flex justify="between">
114
+ <Body><strong>Commanders</strong></Body>
115
+ <Body>45</Body>
116
+ </Flex>
117
+ <Flex justify="between">
118
+ <Body>Lions</Body>
119
+ <Body>31</Body>
120
+ </Flex>
121
+ </Layout.Game>
122
+ <Layout.Game>
123
+ <Flex justify="between">
124
+ <Body>Texans</Body>
125
+ <Body>14</Body>
126
+ </Flex>
127
+ <Flex justify="between">
128
+ <Body><strong>Chiefs</strong></Body>
129
+ <Body>23</Body>
130
+ </Flex>
131
+ </Layout.Game>
132
+ <Layout.Game>
133
+ <Flex justify="between">
134
+ <Body>Ravens</Body>
135
+ <Body>25</Body>
136
+ </Flex>
137
+ <Flex justify="between">
138
+ <Body><strong>Bills</strong></Body>
139
+ <Body>27</Body>
140
+ </Flex>
141
+ </Layout.Game>
142
+ </Layout.Round>
143
+ <Layout.RoundLabel>
144
+ <Caption>Conference</Caption>
145
+ <SectionSeparator marginY="sm"/>
146
+ </Layout.RoundLabel>
147
+ <Layout.Round marginBottom={{ xs: "md", sm: "md" }}>
148
+ <Layout.Game>
149
+ <Flex justify="between">
150
+ <Body>Commanders</Body>
151
+ <Body>23</Body>
152
+ </Flex>
153
+ <Flex justify="between">
154
+ <Body><strong>Eagles</strong></Body>
155
+ <Body>55</Body>
156
+ </Flex>
157
+ </Layout.Game>
158
+ <Layout.Game>
159
+ <Flex justify="between">
160
+ <Body>Bills</Body>
161
+ <Body>29</Body>
162
+ </Flex>
163
+ <Flex justify="between">
164
+ <Body><strong>Chiefs</strong></Body>
165
+ <Body>32</Body>
166
+ </Flex>
167
+ </Layout.Game>
168
+ </Layout.Round>
169
+ <Layout.RoundLabel>
170
+ <Caption>Super Bowl</Caption>
171
+ <SectionSeparator marginY="sm"/>
172
+ </Layout.RoundLabel>
173
+ <Layout.Round>
174
+ <Layout.Game>
175
+ <Flex justify="between">
176
+ <Body>Chiefs</Body>
177
+ <Body>22</Body>
178
+ </Flex>
179
+ <Flex justify="between">
180
+ <Body><strong>Eagles</strong></Body>
181
+ <Body>40</Body>
182
+ </Flex>
183
+ </Layout.Game>
184
+ </Layout.Round>
185
+ </Layout>
186
+ </div>
187
+ )
188
+ }
189
+
190
+ export default LayoutBracket
@@ -0,0 +1,5 @@
1
+ Use `<Layout.RoundLabel>`, `<Layout.Round>`, and `<Layout.Game>` to make a bracket layout.
2
+
3
+ On mobile devices, `<Layout.RoundLabel>` will display (on desktop these components are hidden) and the bracket will be in one column.
4
+
5
+ Ensure that each `<Layout.Game>` maintains a consistent height for the bracket lines to lay out properly.
@@ -18,4 +18,5 @@ examples:
18
18
  - layout_kanban: Kanban Layout
19
19
  - layout_content: Content Layout
20
20
  - layout_masonry: Masonry Layout
21
+ - layout_bracket: Bracket Layout
21
22
 
@@ -7,4 +7,5 @@ export { default as LayoutKanbanResponsive } from './_layout_kanban_responsive.j
7
7
  export { default as LayoutCollectionDetail } from './_layout_collection_detail.jsx'
8
8
  export { default as LayoutContent } from './_layout_content.jsx'
9
9
  export { default as LayoutMasonry } from './_layout_masonry.jsx'
10
+ export { default as LayoutBracket } from './_layout_bracket.jsx'
10
11
 
@@ -82,6 +82,10 @@ test("render all layout variants", () => {
82
82
  layout: "masonry",
83
83
  expected: "pb_layout_kit_masonry_size_md_left_light",
84
84
  },
85
+ {
86
+ layout: "bracket",
87
+ expected: "pb_layout_kit_bracket",
88
+ },
85
89
  ]
86
90
 
87
91
  testValues.forEach(({ layout, expected }) => {
@@ -0,0 +1,90 @@
1
+ import React from 'react'
2
+ import classnames from 'classnames'
3
+
4
+ import { GlobalProps, globalProps, globalInlineProps } from '../../utilities/globalProps'
5
+
6
+ import Card from '../../pb_card/_card'
7
+ import SectionSeparator from '../../pb_section_separator/_section_separator'
8
+
9
+ type LayoutGameProps = {
10
+ children: React.ReactNode[] | React.ReactNode,
11
+ className?: string,
12
+ numberOfRounds: number,
13
+ numberOfGames: number,
14
+ isOdd: boolean,
15
+ } & GlobalProps
16
+
17
+ // Game component (modeled after Item)
18
+ const Game = (props: LayoutGameProps) => {
19
+ const { children, className, numberOfRounds, numberOfGames, isOdd } = props
20
+ const dynamicInlineProps = globalInlineProps(props)
21
+
22
+ const numberOfChildren = Array.isArray(children) ? children.length : 0
23
+
24
+ const isMultiple = Array.isArray(children)
25
+
26
+ let ratio = 0
27
+ let exponent
28
+ if (numberOfGames > 1) {
29
+ exponent = (numberOfRounds) - Math.log2(numberOfGames) - 1
30
+ ratio = 2 ** exponent
31
+ }
32
+
33
+ if (numberOfChildren === 2) {
34
+ const [firstChild, secondChild] = React.Children.toArray(children)
35
+
36
+ if (React.isValidElement(firstChild) && React.isValidElement(secondChild)) {
37
+ return (
38
+ <div
39
+ className={classnames('layout_game', globalProps(props), className)}
40
+ style={dynamicInlineProps}
41
+ >
42
+ <Card
43
+ marginY="xs"
44
+ padding="none"
45
+ shadow="deep"
46
+ >
47
+ <Card.Body padding="xs">{firstChild}</Card.Body>
48
+ <SectionSeparator />
49
+ <Card.Body padding="xs">{secondChild}</Card.Body>
50
+ </Card>
51
+ {isOdd && numberOfGames > 1 &&
52
+ <div
53
+ className="half_box"
54
+ style={{ height: `calc(${ratio} * 100% + 4px)` }}
55
+ />
56
+ }
57
+ </div>
58
+ )
59
+ }
60
+ }
61
+
62
+ return (
63
+ <div
64
+ className={classnames('layout_game', globalProps(props), className)}
65
+ style={dynamicInlineProps}
66
+ >
67
+ {((!isMultiple && children) || numberOfChildren >= 1 )? (
68
+ children
69
+ ) : (
70
+ <Card
71
+ marginY="xs"
72
+ padding="none"
73
+ shadow="deep"
74
+ >
75
+ <Card.Body padding="xs">To be determined...</Card.Body>
76
+ <SectionSeparator />
77
+ <Card.Body padding="xs">To be determined...</Card.Body>
78
+ </Card>
79
+ )}
80
+ {isOdd && numberOfGames > 1 &&
81
+ <div
82
+ className="half_box"
83
+ style={{ height: `calc(${ratio} * 100% + 4px)` }}
84
+ />
85
+ }
86
+ </div>
87
+ )
88
+ }
89
+
90
+ export default Game
@@ -0,0 +1,57 @@
1
+ import React from 'react'
2
+ import classnames from 'classnames'
3
+
4
+ import { GlobalProps, globalProps, globalInlineProps } from '../../utilities/globalProps'
5
+
6
+ type LayoutRoundLabelProps = {
7
+ children: React.ReactNode[] | React.ReactNode,
8
+ className?: string,
9
+ } & GlobalProps
10
+
11
+ export const RoundLabel = (props: LayoutRoundLabelProps) => {
12
+ const { children, className } = props
13
+ const dynamicInlineProps = globalInlineProps(props)
14
+ return (
15
+ <div
16
+ className={classnames('layout_round_label', className)}
17
+ style={dynamicInlineProps}
18
+ >
19
+ {children}
20
+ </div>
21
+ )
22
+ }
23
+
24
+ type LayoutRoundProps = {
25
+ children: React.ReactNode[] | React.ReactNode,
26
+ className?: string,
27
+ numberOfRounds: number,
28
+ } & GlobalProps
29
+
30
+ export const Round = (props: LayoutRoundProps) => {
31
+ const { children, className, numberOfRounds } = props
32
+ const dynamicInlineProps = globalInlineProps(props)
33
+ const numberOfChildren = Array.isArray(children) ? children.length : 0
34
+
35
+ const childrenWithProps = Array.isArray(children) ? children.map((child, index) =>
36
+ React.isValidElement(child) ? React.cloneElement(child, { numberOfRounds, numberOfGames: numberOfChildren, isOdd: index % 2 === 0, key: `child_${index}` }) : child
37
+ ) : children
38
+
39
+ const rightConnectors = Array.from({ length: numberOfChildren / 2 }, (_, index) => (
40
+ <div
41
+ className="right_connector"
42
+ key={`right_connector_${index}`}
43
+ />
44
+ ))
45
+
46
+ return (
47
+ <>
48
+ <div
49
+ className={classnames('layout_round', globalProps(props), className)}
50
+ style={dynamicInlineProps}
51
+ >
52
+ {childrenWithProps}
53
+ </div>
54
+ <div className="connector_container">{rightConnectors}</div>
55
+ </>
56
+ )
57
+ }
@@ -1,4 +1,4 @@
1
- import { debounce } from 'lodash'
1
+ import { debounce } from "../../utilities/object"
2
2
  import { useCallback, useMemo, useState } from 'react'
3
3
 
4
4
  export default function useVisibility(initialState = false) {
@@ -73,7 +73,7 @@ test('adds icon', () => {
73
73
 
74
74
  const kit = screen.getByTestId('icon-test')
75
75
 
76
- const icon = kit.querySelector('.pb_icon_kit')
76
+ const icon = kit.querySelector('.pb_custom_icon')
77
77
  expect(icon).toBeInTheDocument();
78
78
  })
79
79
 
@@ -87,7 +87,7 @@ test('adds icon right', () => {
87
87
 
88
88
  const kit = screen.getByTestId('icon-right-test')
89
89
 
90
- const icon = kit.querySelector('.pb_icon_kit')
90
+ const icon = kit.querySelector('.pb_custom_icon')
91
91
  expect(icon).toBeInTheDocument();
92
92
  })
93
93
 
@@ -7,11 +7,12 @@ import {
7
7
  buildDataProps,
8
8
  buildHtmlProps,
9
9
  } from "../utilities/props";
10
+ import { cloneDeep } from "../utilities/object";
11
+
10
12
  import Icon from "../pb_icon/_icon";
11
13
  import FormPill from "../pb_form_pill/_form_pill";
12
14
  import Body from "../pb_body/_body";
13
15
  import Caption from "../pb_caption/_caption";
14
- import { cloneDeep } from "lodash";
15
16
  import MultiLevelSelectOptions from "./multi_level_select_options";
16
17
  import MultiLevelSelectContext from "./context";
17
18