playbook_ui 14.16.0.pre.rc.6 → 14.16.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 (80) hide show
  1. checksums.yaml +4 -4
  2. data/app/pb_kits/playbook/pb_bar_graph/_bar_graph.tsx +1 -1
  3. data/app/pb_kits/playbook/pb_circle_chart/_circle_chart.tsx +1 -1
  4. data/app/pb_kits/playbook/pb_draggable/_draggable.scss +115 -5
  5. data/app/pb_kits/playbook/pb_draggable/_draggable.tsx +6 -4
  6. data/app/pb_kits/playbook/pb_draggable/context/index.tsx +35 -9
  7. data/app/pb_kits/playbook/pb_draggable/context/types.ts +35 -28
  8. data/app/pb_kits/playbook/pb_draggable/docs/_draggable_drop_zones.jsx +184 -0
  9. data/app/pb_kits/playbook/pb_draggable/docs/_draggable_drop_zones.md +5 -0
  10. data/app/pb_kits/playbook/pb_draggable/docs/_draggable_drop_zones_colors.jsx +97 -0
  11. data/app/pb_kits/playbook/pb_draggable/docs/_draggable_drop_zones_colors.md +1 -0
  12. data/app/pb_kits/playbook/pb_draggable/subcomponents/DraggableContainer.tsx +11 -2
  13. data/app/pb_kits/playbook/pb_draggable/subcomponents/DraggableItem.tsx +65 -6
  14. data/app/pb_kits/playbook/pb_drawer/docs/_drawer_breakpoints.html.erb +3 -0
  15. data/app/pb_kits/playbook/pb_drawer/docs/example.yml +1 -0
  16. data/app/pb_kits/playbook/pb_filter/Filter/SortMenu.tsx +2 -1
  17. data/app/pb_kits/playbook/pb_fixed_confirmation_toast/_fixed_confirmation_toast.scss +2 -2
  18. data/app/pb_kits/playbook/pb_form/docs/_form_form_with.html.erb +67 -0
  19. data/app/pb_kits/playbook/pb_form/docs/_form_form_with_validate.html.erb +67 -0
  20. data/app/pb_kits/playbook/pb_gauge/_gauge.tsx +1 -1
  21. data/app/pb_kits/playbook/pb_icon/_icon.scss +8 -1
  22. data/app/pb_kits/playbook/pb_icon/docs/_icon_color.html.erb +10 -4
  23. data/app/pb_kits/playbook/pb_icon/docs/_icon_color.jsx +49 -24
  24. data/app/pb_kits/playbook/pb_icon_button/_icon_button.tsx +4 -1
  25. data/app/pb_kits/playbook/pb_icon_button/docs/_icon_button_click.jsx +13 -0
  26. data/app/pb_kits/playbook/pb_icon_button/docs/example.yml +1 -0
  27. data/app/pb_kits/playbook/pb_icon_button/docs/index.js +1 -0
  28. data/app/pb_kits/playbook/pb_lightbox/Carousel/Slide.tsx +1 -1
  29. data/app/pb_kits/playbook/pb_lightbox/Carousel/Slides.tsx +1 -1
  30. data/app/pb_kits/playbook/pb_lightbox/Carousel/Thumbnails.tsx +1 -1
  31. data/app/pb_kits/playbook/pb_line_graph/_line_graph.tsx +1 -1
  32. data/app/pb_kits/playbook/pb_multi_level_select/_multi_level_select.scss +23 -0
  33. data/app/pb_kits/playbook/pb_multi_level_select/_multi_level_select.tsx +26 -0
  34. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_error.html.erb +72 -0
  35. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_error.jsx +97 -0
  36. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_label.html.erb +71 -0
  37. data/app/pb_kits/playbook/pb_multi_level_select/docs/_multi_level_select_label.jsx +91 -0
  38. data/app/pb_kits/playbook/pb_multi_level_select/docs/example.yml +4 -0
  39. data/app/pb_kits/playbook/pb_multi_level_select/docs/index.js +3 -1
  40. data/app/pb_kits/playbook/pb_multi_level_select/index.js +105 -0
  41. data/app/pb_kits/playbook/pb_multi_level_select/multi_level_select.rb +10 -0
  42. data/app/pb_kits/playbook/pb_nav/_nav.scss +5 -0
  43. data/app/pb_kits/playbook/pb_treemap_chart/_treemap_chart.tsx +1 -1
  44. data/app/pb_kits/playbook/pb_typeahead/_typeahead.scss +4 -0
  45. data/app/pb_kits/playbook/pb_typeahead/_typeahead.tsx +3 -0
  46. data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_disabled.html.erb +19 -0
  47. data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_disabled.jsx +23 -0
  48. data/app/pb_kits/playbook/pb_typeahead/docs/example.yml +2 -0
  49. data/app/pb_kits/playbook/pb_typeahead/docs/index.js +1 -0
  50. data/app/pb_kits/playbook/pb_typeahead/typeahead.rb +3 -0
  51. data/app/pb_kits/playbook/pb_user/_user.tsx +78 -13
  52. data/app/pb_kits/playbook/pb_user/docs/_user_font_options.html.erb +22 -0
  53. data/app/pb_kits/playbook/pb_user/docs/_user_font_options.jsx +40 -0
  54. data/app/pb_kits/playbook/pb_user/docs/_user_font_options_rails.md +5 -0
  55. data/app/pb_kits/playbook/pb_user/docs/_user_font_options_react.md +5 -0
  56. data/app/pb_kits/playbook/pb_user/docs/example.yml +2 -0
  57. data/app/pb_kits/playbook/pb_user/docs/index.js +1 -0
  58. data/app/pb_kits/playbook/pb_user/user.html.erb +27 -6
  59. data/app/pb_kits/playbook/pb_user/user.rb +17 -1
  60. data/app/pb_kits/playbook/pb_user/user.test.js +182 -1
  61. data/app/pb_kits/playbook/tokens/_colors.scss +1 -4
  62. data/app/pb_kits/playbook/utilities/object.test.js +139 -1
  63. data/app/pb_kits/playbook/utilities/object.ts +86 -0
  64. data/app/pb_kits/playbook/utilities/text.ts +1 -1
  65. data/dist/chunks/_typeahead-BuTZG1Jn.js +22 -0
  66. data/dist/chunks/_weekday_stacked-oT22q75-.js +45 -0
  67. data/dist/chunks/lazysizes-DHz07jlL.js +1 -0
  68. data/dist/chunks/{lib-BeKPJYlk.js → lib-Co5y3V4K.js} +2 -2
  69. data/dist/chunks/{pb_form_validation-BvDxpfs-.js → pb_form_validation-DMajaRt3.js} +1 -1
  70. data/dist/chunks/vendor.js +1 -1
  71. data/dist/playbook-doc.js +1 -1
  72. data/dist/playbook-rails-react-bindings.js +1 -1
  73. data/dist/playbook-rails.js +1 -1
  74. data/dist/playbook.css +1 -1
  75. data/lib/playbook/forms/builder/multi_level_select_field.rb +2 -0
  76. data/lib/playbook/version.rb +1 -1
  77. metadata +23 -7
  78. data/dist/chunks/_typeahead-CRAPc8k-.js +0 -22
  79. data/dist/chunks/_weekday_stacked-T0kFfioG.js +0 -45
  80. data/dist/chunks/lazysizes-B7xYodB-.js +0 -1
@@ -22,24 +22,82 @@ type DraggableItemProps = {
22
22
  };
23
23
 
24
24
  const DraggableItem = (props: DraggableItemProps) => {
25
- const { aria = {}, children, className, container, data = {}, htmlOptions = {}, id, dragId, tag="div" } = props;
25
+ const {
26
+ aria = {},
27
+ children,
28
+ className,
29
+ container,
30
+ data = {},
31
+ htmlOptions = {},
32
+ id,
33
+ dragId,
34
+ tag="div"
35
+ } = props;
26
36
 
27
- const { isDragging, handleDragStart, handleDragEnter, handleDragEnd } =
28
- DraggableContext();
37
+ const {
38
+ isDragging,
39
+ handleDragStart,
40
+ handleDragEnter,
41
+ handleDragEnd,
42
+ dropZone = 'ghost',
43
+ dropZoneColor = 'neutral',
44
+ direction = 'horizontal'
45
+ } = DraggableContext();
46
+
47
+ const itemRef = React.useRef<HTMLElement>(null);
29
48
 
30
49
  const ariaProps = buildAriaProps(aria);
31
50
  const dataProps = buildDataProps(data);
32
51
  const htmlProps = buildHtmlProps(htmlOptions);
33
-
34
52
  const Tag: React.ReactElement | any = `${tag}`;
35
-
53
+
36
54
  const classes = classnames(
37
55
  buildCss("pb_draggable_item"),
38
56
  `${isDragging === dragId ? "is_dragging" : ""}`,
57
+ isDragging === dragId ? `drop_zone_${dropZone}` : "",
58
+ isDragging === dragId && dropZone !== 'ghost' ? `drop_zone_color_${dropZoneColor}` : '',
39
59
  globalProps(props),
40
60
  className
41
61
  );
42
62
 
63
+ // Enhanced drag start handler that preserves dimensions
64
+ const onDragStart = (e: React.DragEvent) => {
65
+ if (dropZone !== 'ghost' && itemRef.current) {
66
+ // Create a clone for the drag image
67
+ const clone = itemRef.current.cloneNode(true) as HTMLElement;
68
+
69
+ // Remove any classes that might affect appearance
70
+ clone.className = clone.className.replace(/drop_zone_[^ ]*/g, '');
71
+ clone.className = clone.className.replace(/is_dragging/g, '');
72
+
73
+ // Get the original dimensions
74
+ const rect = itemRef.current.getBoundingClientRect();
75
+
76
+ // Ensure it's styled properly but invisible on the page
77
+ clone.style.position = 'fixed';
78
+ clone.style.top = '-9999px';
79
+ clone.style.left = '-9999px';
80
+ clone.style.width = `${rect.width}px`; // Preserve width
81
+ clone.style.height = `${rect.height}px`; // Preserve height
82
+ clone.style.opacity = '1';
83
+ clone.style.pointerEvents = 'none';
84
+
85
+ // Add to document temporarily
86
+ document.body.appendChild(clone);
87
+
88
+ // Set as drag image
89
+ e.dataTransfer.setDragImage(clone, e.clientX - rect.left, e.clientY - rect.top);
90
+
91
+ // Remove after a short delay
92
+ setTimeout(() => {
93
+ document.body.removeChild(clone);
94
+ }, 0);
95
+ }
96
+
97
+ // Call the original handler
98
+ handleDragStart(dragId, container);
99
+ };
100
+
43
101
  return (
44
102
  <Tag
45
103
  {...ariaProps}
@@ -51,7 +109,8 @@ const DraggableItem = (props: DraggableItemProps) => {
51
109
  key={dragId}
52
110
  onDragEnd={() => handleDragEnd()}
53
111
  onDragEnter={() => handleDragEnter(dragId, container)}
54
- onDragStart={() => handleDragStart(dragId, container)}
112
+ onDragStart={onDragStart}
113
+ ref={itemRef}
55
114
  >
56
115
  {children}
57
116
  </Tag>
@@ -0,0 +1,3 @@
1
+
2
+ <%= pb_rails("body", props: { text: " The breakpoint prop determines when the Drawer is always visible. Above the specified breakpoint, the Drawer remains open on the page. Below it, only the trigger element is shown, allowing you to toggle the drawer open and closed. To see this in action, click the button below and resize your window.", padding_bottom: "md" }) %>
3
+ <%= pb_rails("button", props: { text: "Open Doc Example in New Tab", new_window: true, link: "/drawer_page", margin_right: "lg" }) %>
@@ -6,6 +6,7 @@ examples:
6
6
  - drawer_menu: Within Element
7
7
  - drawer_sizes: Sizes
8
8
  - drawer_overlay: Overlay
9
+ - drawer_breakpoints: Breakpoints
9
10
  - drawer_borders: Borders
10
11
 
11
12
 
@@ -1,5 +1,6 @@
1
1
  import React, { useState } from 'react'
2
- import { find, map, partial } from 'lodash'
2
+ import { map } from 'lodash'
3
+ import { find, partial } from '../../utilities/object'
3
4
 
4
5
  import Button from '../../pb_button/_button'
5
6
  import Icon from '../../pb_icon/_icon'
@@ -7,7 +7,7 @@ $pb_pill_height: 25px;
7
7
 
8
8
 
9
9
  $confirmation_toast_colors: (
10
- neutral: $neutral,
10
+ neutral: $text_lt_light,
11
11
  success: $success,
12
12
  error: $error,
13
13
  tip: transparent,
@@ -87,7 +87,7 @@ $confirmation_toast_colors: (
87
87
  overflow: hidden;
88
88
  display: -webkit-box;
89
89
  -webkit-line-clamp: 3;
90
- -webkit-box-orient: vertical;
90
+ -webkit-box-orient: vertical;
91
91
  white-space: normal;
92
92
  text-align: left;
93
93
  }
@@ -22,6 +22,72 @@
22
22
 
23
23
  %>
24
24
 
25
+ <% treeData = [{
26
+ label: "Power Home Remodeling",
27
+ value: "Power Home Remodeling",
28
+ id: "100",
29
+ expanded: true,
30
+ children: [
31
+ {
32
+ label: "People",
33
+ value: "People",
34
+ id: "101",
35
+ expanded: true,
36
+ children: [
37
+ {
38
+ label: "Talent Acquisition",
39
+ value: "Talent Acquisition",
40
+ id: "102",
41
+ },
42
+ {
43
+ label: "Business Affairs",
44
+ value: "Business Affairs",
45
+ id: "103",
46
+ children: [
47
+ {
48
+ label: "Initiatives",
49
+ value: "Initiatives",
50
+ id: "104",
51
+ },
52
+ {
53
+ label: "Learning & Development",
54
+ value: "Learning & Development",
55
+ id: "105",
56
+ },
57
+ ],
58
+ },
59
+ {
60
+ label: "People Experience",
61
+ value: "People Experience",
62
+ id: "106",
63
+ },
64
+ ],
65
+ },
66
+ {
67
+ label: "Contact Center",
68
+ value: "Contact Center",
69
+ id: "107",
70
+ children: [
71
+ {
72
+ label: "Appointment Management",
73
+ value: "Appointment Management",
74
+ id: "108",
75
+ },
76
+ {
77
+ label: "Customer Service",
78
+ value: "Customer Service",
79
+ id: "109",
80
+ },
81
+ {
82
+ label: "Energy",
83
+ value: "Energy",
84
+ id: "110",
85
+ },
86
+ ],
87
+ },
88
+ ],
89
+ }] %>
90
+
25
91
  <%= pb_form_with(scope: :example, url: "", method: :get) do |form| %>
26
92
  <%= form.typeahead :example_typeahead, props: { data: { typeahead_example1: true, user: {} }, label: true, placeholder: "Search for a user" } %>
27
93
  <%= form.text_field :example_text_field, props: { label: true } %>
@@ -47,6 +113,7 @@
47
113
  <%= form.date_picker :example_date_picker_1, props: { label: true } %>
48
114
  <%= form.star_rating_field :example_star_rating, props: { variant: "interactive", label: true } %>
49
115
  <%= form.time_zone_select_field :example_time_zone_select, ActiveSupport::TimeZone.us_zones, { default: "Eastern Time (US & Canada)" }, props: { label: true } %>
116
+ <%= form.multi_level_select :example_multi_level_select, props: { id: "multi-level-select-form-default", tree_data: treeData, margin_bottom: "sm", label: "Example Multi Level Select field" } %>
50
117
 
51
118
  <%= form.actions do |action| %>
52
119
  <%= action.submit %>
@@ -21,6 +21,72 @@
21
21
  ]
22
22
  %>
23
23
 
24
+ <% treeData = [{
25
+ label: "Power Home Remodeling",
26
+ value: "Power Home Remodeling",
27
+ id: "100",
28
+ expanded: true,
29
+ children: [
30
+ {
31
+ label: "People",
32
+ value: "People",
33
+ id: "101",
34
+ expanded: true,
35
+ children: [
36
+ {
37
+ label: "Talent Acquisition",
38
+ value: "Talent Acquisition",
39
+ id: "102",
40
+ },
41
+ {
42
+ label: "Business Affairs",
43
+ value: "Business Affairs",
44
+ id: "103",
45
+ children: [
46
+ {
47
+ label: "Initiatives",
48
+ value: "Initiatives",
49
+ id: "104",
50
+ },
51
+ {
52
+ label: "Learning & Development",
53
+ value: "Learning & Development",
54
+ id: "105",
55
+ },
56
+ ],
57
+ },
58
+ {
59
+ label: "People Experience",
60
+ value: "People Experience",
61
+ id: "106",
62
+ },
63
+ ],
64
+ },
65
+ {
66
+ label: "Contact Center",
67
+ value: "Contact Center",
68
+ id: "107",
69
+ children: [
70
+ {
71
+ label: "Appointment Management",
72
+ value: "Appointment Management",
73
+ id: "108",
74
+ },
75
+ {
76
+ label: "Customer Service",
77
+ value: "Customer Service",
78
+ id: "109",
79
+ },
80
+ {
81
+ label: "Energy",
82
+ value: "Energy",
83
+ id: "110",
84
+ },
85
+ ],
86
+ },
87
+ ],
88
+ }] %>
89
+
24
90
  <%= pb_form_with(scope: :example, method: :get, url: "", validate: true) do |form| %>
25
91
  <%= form.typeahead :example_typeahead_validation, props: { data: { typeahead_example2: true, user: {} }, label: true, placeholder: "Search for a user", required: true, validation: { message: "Please select a user." } } %>
26
92
  <%= form.text_field :example_text_field_validation, props: { label: true, required: true } %>
@@ -38,6 +104,7 @@
38
104
  <%= form.date_picker :example_date_picker_2, props: { label: true, required: true, validation_message: "Please, select a date.", allow_input: true } %>
39
105
  <%= form.star_rating_field :example_star_rating_validation, props: { variant: "interactive", label: true, required: true } %>
40
106
  <%= form.time_zone_select_field :example_time_zone_select, ActiveSupport::TimeZone.us_zones, { default: "Eastern Time (US & Canada)" }, props: { label: true, blank_selection: "Select a Time Zone...", required: true } %>
107
+ <%= form.multi_level_select :example_multi_level_select, props: { id: "multi-level-select-form", tree_data: treeData, margin_bottom: "sm", required: true, label: "Example Multi Level Select field" } %>
41
108
 
42
109
  <%= form.actions do |action| %>
43
110
  <%= action.submit %>
@@ -13,7 +13,7 @@ import typography from "../tokens/exports/_typography.module.scss";
13
13
  import { buildAriaProps, buildCss, buildDataProps, buildHtmlProps } from "../utilities/props";
14
14
  import { globalProps } from "../utilities/globalProps";
15
15
  import { GenericObject } from "../types";
16
- import { merge } from 'lodash'
16
+ import { merge } from '../utilities/object'
17
17
 
18
18
  type GaugeProps = {
19
19
  aria: { [key: string]: string };
@@ -1,10 +1,17 @@
1
1
  @import "../tokens/colors";
2
2
 
3
+ $additional_colors: (
4
+ "light": $text_lt_light,
5
+ "lighter": $text_lt_lighter,
6
+ "link": $primary
7
+ );
8
+
3
9
  // All the merges below create $icon_colors, a map of all color tokens in colors.scss
4
10
  $merge_kits1: map-merge($status_colors, $category_colors);
5
11
  $merge_kits2: map-merge($merge_kits1, $product_colors);
6
12
  $merge_kits3: map-merge($merge_kits2, $text_colors);
7
- $icon_colors: map-merge($merge_kits3, $data_colors);
13
+ $merge_kits4: map-merge($merge_kits3, $data_colors);
14
+ $icon_colors: map-merge($merge_kits4, $additional_colors);
8
15
 
9
16
  .pb_custom_icon, .pb_icon_kit {
10
17
  @each $color_name, $color_value in $icon_colors {
@@ -1,5 +1,11 @@
1
- <%= pb_rails("flex", props: {orientation: "column"}) do %>
2
- <%= pb_rails("icon", props: { icon: "user", fixed_width: true, color: "primary", padding_bottom: "sm", size: "2x" }) %>
3
- <%= pb_rails("icon", props: { icon: "recycle", fixed_width: true, color: "data_4", padding_bottom: "sm", size: "2x" }) %>
1
+ <%= pb_rails("flex", props: { margin_bottom: "sm" }) do %>
2
+ <%= pb_rails("icon", props: { icon: "user", fixed_width: true, color: "primary", size: "2x" }) %>
3
+ <%= pb_rails("icon", props: { icon: "recycle", fixed_width: true, color: "data_4", size: "2x" }) %>
4
4
  <%= pb_rails("icon", props: { icon: "roofing", fixed_width: true, color: "product_5_background", size: "2x" }) %>
5
- <% end %>
5
+ <% end %>
6
+
7
+ <%= pb_rails("flex", props: {}) do %>
8
+ <%= pb_rails("icon", props: { icon: "user", fixed_width: true, color: "light", size: "2x" }) %>
9
+ <%= pb_rails("icon", props: { icon: "recycle", fixed_width: true, color: "lighter", size: "2x" }) %>
10
+ <%= pb_rails("icon", props: { icon: "roofing", fixed_width: true, color: "link", size: "2x" }) %>
11
+ <% end %>
@@ -1,32 +1,57 @@
1
1
  import React from "react"
2
2
  import Icon from "../_icon"
3
+ import Flex from "../../pb_flex/_flex"
3
4
 
4
5
  const IconDefault = (props) => {
5
6
  return (
6
- <div style={{ display: "flex", flexDirection: "column"}}>
7
- <Icon
8
- color="primary"
9
- fixedWidth
10
- icon="user"
11
- paddingBottom="sm"
12
- size="2x"
13
- {...props}
14
- />
15
- <Icon
16
- color="data_4"
17
- fixedWidth
18
- icon="recycle"
19
- paddingBottom="sm"
20
- size="2x"
21
- {...props}
22
- />
23
- <Icon
24
- color="product_5_background"
25
- fixedWidth
26
- icon="product-roofing"
27
- size="2x"
28
- {...props}
29
- />
7
+ <div>
8
+ <Flex marginBottom="sm">
9
+ <Icon
10
+ color="primary"
11
+ fixedWidth
12
+ icon="user"
13
+ size="2x"
14
+ {...props}
15
+ />
16
+ <Icon
17
+ color="data_4"
18
+ fixedWidth
19
+ icon="recycle"
20
+ size="2x"
21
+ {...props}
22
+ />
23
+ <Icon
24
+ color="product_5_background"
25
+ fixedWidth
26
+ icon="product-roofing"
27
+ size="2x"
28
+ {...props}
29
+ />
30
+ </Flex>
31
+
32
+ <Flex>
33
+ <Icon
34
+ color="light"
35
+ fixedWidth
36
+ icon="user"
37
+ size="2x"
38
+ {...props}
39
+ />
40
+ <Icon
41
+ color="lighter"
42
+ fixedWidth
43
+ icon="recycle"
44
+ size="2x"
45
+ {...props}
46
+ />
47
+ <Icon
48
+ color="link"
49
+ fixedWidth
50
+ icon="product-roofing"
51
+ size="2x"
52
+ {...props}
53
+ />
54
+ </Flex>
30
55
  </div>
31
56
  )
32
57
  }
@@ -1,7 +1,7 @@
1
1
 
2
2
  import React from 'react'
3
3
  import classnames from 'classnames'
4
- import { buildAriaProps, buildCss, buildDataProps } from '../utilities/props'
4
+ import { buildAriaProps, buildCss, buildDataProps, noop } from '../utilities/props'
5
5
  import { globalProps } from '../utilities/globalProps'
6
6
 
7
7
  import Button from '../pb_button/_button'
@@ -17,6 +17,7 @@ type IconButtonProps = {
17
17
  id?: string,
18
18
  link?: string,
19
19
  newWindow?: boolean,
20
+ onClick?: React.MouseEventHandler<HTMLElement>,
20
21
  size?: IconSizes,
21
22
  target?: string,
22
23
  variant?: 'default' | 'link',
@@ -32,6 +33,7 @@ const IconButton = (props: IconButtonProps) => {
32
33
  id,
33
34
  link,
34
35
  newWindow = false,
36
+ onClick = noop,
35
37
  size = "2x",
36
38
  target,
37
39
  variant = "default",
@@ -53,6 +55,7 @@ const IconButton = (props: IconButtonProps) => {
53
55
  htmlType={htmlType}
54
56
  link={link}
55
57
  newWindow={newWindow}
58
+ onClick={onClick}
56
59
  target={target}
57
60
  >
58
61
  <Icon
@@ -0,0 +1,13 @@
1
+ import React from 'react'
2
+ import IconButton from '../../pb_icon_button/_icon_button'
3
+
4
+ const IconButtonClick = (props) => (
5
+ <div>
6
+ <IconButton
7
+ {...props}
8
+ onClick={() => alert('Click!')}
9
+ />
10
+ </div>
11
+ )
12
+
13
+ export default IconButtonClick
@@ -7,3 +7,4 @@ examples:
7
7
  react:
8
8
  - icon_button_default: Default
9
9
  - icon_button_sizes: Sizes
10
+ - icon_button_click: Click Handler
@@ -1,2 +1,3 @@
1
1
  export { default as IconButtonDefault } from './_icon_button_default.jsx'
2
2
  export { default as IconButtonSizes } from './_icon_button_sizes.jsx'
3
+ export { default as IconButtonClick } from './_icon_button_click.jsx'
@@ -1,5 +1,5 @@
1
1
  import React from 'react'
2
- import { noop } from 'lodash'
2
+ import { noop } from '../../utilities/object'
3
3
  import { TransformComponent, TransformWrapper } from 'react-zoom-pan-pinch'
4
4
  import Image from '../../pb_image/_image'
5
5
 
@@ -1,4 +1,4 @@
1
- import { noop } from 'lodash'
1
+ import { noop } from '../../utilities/object'
2
2
  import React, { useState } from 'react'
3
3
  import CircleIconButton from '../../pb_circle_icon_button/_circle_icon_button'
4
4
 
@@ -1,4 +1,4 @@
1
- import { noop } from 'lodash'
1
+ import { noop } from '../../utilities/object'
2
2
  import classnames from 'classnames'
3
3
  import React, { useLayoutEffect } from 'react'
4
4
  import { useWindowSize } from '../hooks/useWindowSize'
@@ -8,7 +8,7 @@ import Highcharts from "highcharts";
8
8
  import { highchartsTheme } from "../pb_dashboard/pbChartsLightTheme";
9
9
  import { highchartsDarkTheme } from "../pb_dashboard/pbChartsDarkTheme";
10
10
  import mapColors from "../pb_dashboard/pbChartsColorsHelper";
11
- import { merge } from 'lodash'
11
+ import { merge } from '../utilities/object'
12
12
 
13
13
  type LineGraphProps = {
14
14
  align?: "left" | "right" | "center";
@@ -10,6 +10,16 @@
10
10
  .pb_multi_level_select {
11
11
  font-family: $font-family-base;
12
12
 
13
+ &.error {
14
+ .wrapper {
15
+ .input_wrapper {
16
+ border-color: $error;
17
+ }
18
+ }
19
+ [class*=pb_body_kit_negative] {
20
+ margin-top: $space_xxs;
21
+ }
22
+ }
13
23
  .wrapper {
14
24
  position: relative;
15
25
 
@@ -92,4 +102,17 @@
92
102
  .open {
93
103
  display: block;
94
104
  }
105
+
106
+ &.dark {
107
+ &.error {
108
+ .wrapper {
109
+ .input_wrapper {
110
+ border-color: $error_dark;
111
+ }
112
+ }
113
+ [class*=pb_body_kit_negative] {
114
+ color: $error_dark;
115
+ }
116
+ }
117
+ }
95
118
  }
@@ -9,6 +9,8 @@ import {
9
9
  } from "../utilities/props";
10
10
  import Icon from "../pb_icon/_icon";
11
11
  import FormPill from "../pb_form_pill/_form_pill";
12
+ import Body from "../pb_body/_body";
13
+ import Caption from "../pb_caption/_caption";
12
14
  import { cloneDeep } from "lodash";
13
15
  import MultiLevelSelectOptions from "./multi_level_select_options";
14
16
  import MultiLevelSelectContext from "./context";
@@ -35,11 +37,14 @@ type MultiLevelSelectProps = {
35
37
  className?: string
36
38
  data?: { [key: string]: string }
37
39
  disabled?: boolean
40
+ error?: string
38
41
  htmlOptions?: {[key: string]: string | number | boolean | (() => void)},
39
42
  id?: string
40
43
  inputDisplay?: "pills" | "none"
41
44
  inputName?: string
45
+ label?: string
42
46
  name?: string
47
+ required?: boolean
43
48
  returnAllSelected?: boolean
44
49
  treeData?: { [key: string]: string; }[] | any
45
50
  onChange?: (event: { target: { name?: string; value: any } }) => void
@@ -56,11 +61,14 @@ const MultiLevelSelect = forwardRef<HTMLInputElement, MultiLevelSelectProps>((pr
56
61
  className,
57
62
  data = {},
58
63
  disabled = false,
64
+ error,
59
65
  htmlOptions = {},
60
66
  id,
61
67
  inputDisplay = "pills",
62
68
  inputName,
63
69
  name,
70
+ label,
71
+ required = false,
64
72
  returnAllSelected = false,
65
73
  treeData,
66
74
  onChange = () => null,
@@ -77,6 +85,7 @@ const MultiLevelSelect = forwardRef<HTMLInputElement, MultiLevelSelectProps>((pr
77
85
  const htmlProps = buildHtmlProps(htmlOptions);
78
86
  const classes = classnames(
79
87
  buildCss("pb_multi_level_select"),
88
+ error && "error",
80
89
  globalProps(props),
81
90
  className
82
91
  );
@@ -439,6 +448,12 @@ const MultiLevelSelect = forwardRef<HTMLInputElement, MultiLevelSelectProps>((pr
439
448
  className={classes}
440
449
  id={id}
441
450
  >
451
+ {label &&
452
+ <Caption
453
+ marginBottom="xs"
454
+ text={label}
455
+ />
456
+ }
442
457
  <MultiLevelSelectContext.Provider value={{
443
458
  variant,
444
459
  inputName,
@@ -462,6 +477,7 @@ const MultiLevelSelect = forwardRef<HTMLInputElement, MultiLevelSelectProps>((pr
462
477
  disabled={disabled}
463
478
  key={selectedItem.id}
464
479
  name={`${name}[]`}
480
+ required={required}
465
481
  type="hidden"
466
482
  value={selectedItem.id}
467
483
  />
@@ -476,6 +492,7 @@ const MultiLevelSelect = forwardRef<HTMLInputElement, MultiLevelSelectProps>((pr
476
492
  disabled={disabled}
477
493
  key={item.id}
478
494
  name={`${name}[]`}
495
+ required={required}
479
496
  type="hidden"
480
497
  value={item.id}
481
498
  />
@@ -488,6 +505,7 @@ const MultiLevelSelect = forwardRef<HTMLInputElement, MultiLevelSelectProps>((pr
488
505
  disabled={disabled}
489
506
  key={item.id}
490
507
  name={`${name}[]`}
508
+ required={required}
491
509
  type="hidden"
492
510
  value={item.id}
493
511
  />
@@ -548,6 +566,7 @@ const MultiLevelSelect = forwardRef<HTMLInputElement, MultiLevelSelectProps>((pr
548
566
  } selected`
549
567
  : "Start typing..."
550
568
  }
569
+ required={required}
551
570
  value={singleSelectedItem.value || filterItem}
552
571
  />
553
572
  </div>
@@ -580,6 +599,13 @@ const MultiLevelSelect = forwardRef<HTMLInputElement, MultiLevelSelectProps>((pr
580
599
  </div>
581
600
  </div>
582
601
  </MultiLevelSelectContext.Provider>
602
+ {error &&
603
+ <Body
604
+ dark={props.dark}
605
+ status="negative"
606
+ text={error}
607
+ />
608
+ }
583
609
  </div>
584
610
  );
585
611
  }) as MultiLevelSelectComponent;