playbook_ui 14.12.0.pre.alpha.PBNTR720railscarddraggable5649 → 14.12.0.pre.alpha.PBNTR779railsdraggablecrosscontainer5863

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 (101) hide show
  1. checksums.yaml +4 -4
  2. data/app/pb_kits/playbook/_playbook.scss +1 -0
  3. data/app/pb_kits/playbook/pb_advanced_table/Components/CustomCell.tsx +1 -1
  4. data/app/pb_kits/playbook/pb_advanced_table/Components/TableHeaderCell.tsx +2 -2
  5. data/app/pb_kits/playbook/pb_advanced_table/_advanced_table.scss +3 -4
  6. data/app/pb_kits/playbook/pb_advanced_table/advanced_table.html.erb +3 -3
  7. data/app/pb_kits/playbook/pb_advanced_table/advanced_table.rb +3 -1
  8. data/app/pb_kits/playbook/pb_advanced_table/advanced_table.test.jsx +1 -1
  9. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_loading.html.erb +33 -0
  10. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_loading_rails.md +1 -0
  11. data/app/pb_kits/playbook/pb_advanced_table/docs/{_advanced_table_loading.md → _advanced_table_loading_react.md} +2 -2
  12. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_responsive.html.erb +38 -0
  13. data/app/pb_kits/playbook/pb_advanced_table/docs/example.yml +2 -0
  14. data/app/pb_kits/playbook/pb_advanced_table/index.js +9 -6
  15. data/app/pb_kits/playbook/pb_advanced_table/table_body.rb +17 -3
  16. data/app/pb_kits/playbook/pb_advanced_table/table_header.html.erb +15 -11
  17. data/app/pb_kits/playbook/pb_advanced_table/table_header.rb +14 -3
  18. data/app/pb_kits/playbook/pb_advanced_table/table_row.html.erb +10 -7
  19. data/app/pb_kits/playbook/pb_advanced_table/table_row.rb +9 -1
  20. data/app/pb_kits/playbook/pb_advanced_table/table_subrow_header.html.erb +1 -1
  21. data/app/pb_kits/playbook/pb_advanced_table/table_subrow_header.rb +9 -0
  22. data/app/pb_kits/playbook/pb_avatar/_avatar.scss +14 -0
  23. data/app/pb_kits/playbook/pb_avatar/_avatar.tsx +11 -7
  24. data/app/pb_kits/playbook/pb_avatar/avatar.html.erb +6 -7
  25. data/app/pb_kits/playbook/pb_avatar/docs/_avatar_badge_component_overlay.jsx +9 -3
  26. data/app/pb_kits/playbook/pb_avatar/docs/_avatar_circle_icon_component_overlay.jsx +6 -2
  27. data/app/pb_kits/playbook/pb_button/button.rb +1 -1
  28. data/app/pb_kits/playbook/pb_copy_button/_copy_button.scss +3 -0
  29. data/app/pb_kits/playbook/pb_copy_button/_copy_button.tsx +92 -0
  30. data/app/pb_kits/playbook/pb_copy_button/copy_button.test.jsx +64 -0
  31. data/app/pb_kits/playbook/pb_copy_button/docs/_copy_button_default.jsx +21 -0
  32. data/app/pb_kits/playbook/pb_copy_button/docs/_copy_button_from.jsx +45 -0
  33. data/app/pb_kits/playbook/pb_copy_button/docs/_copy_button_from.md +1 -0
  34. data/app/pb_kits/playbook/pb_copy_button/docs/example.yml +8 -0
  35. data/app/pb_kits/playbook/pb_copy_button/docs/index.js +2 -0
  36. data/app/pb_kits/playbook/pb_date/_date.tsx +14 -4
  37. data/app/pb_kits/playbook/pb_date/docs/_date_default.jsx +2 -1
  38. data/app/pb_kits/playbook/pb_date/docs/_date_unstyled.jsx +13 -5
  39. data/app/pb_kits/playbook/pb_draggable/docs/_draggable_multiple_containers_rails.html.erb +193 -0
  40. data/app/pb_kits/playbook/pb_draggable/docs/_draggable_multiple_containers_rails.md +1 -0
  41. data/app/pb_kits/playbook/pb_draggable/docs/example.yml +1 -5
  42. data/app/pb_kits/playbook/pb_draggable/draggable_container.html.erb +2 -2
  43. data/app/pb_kits/playbook/pb_draggable/draggable_container.rb +3 -0
  44. data/app/pb_kits/playbook/pb_draggable/draggable_item.rb +2 -0
  45. data/app/pb_kits/playbook/pb_draggable/index.js +88 -16
  46. data/app/pb_kits/playbook/pb_file_upload/file_upload.html.erb +1 -6
  47. data/app/pb_kits/playbook/pb_filter/filter.html.erb +1 -5
  48. data/app/pb_kits/playbook/pb_fixed_confirmation_toast/docs/_fixed_confirmation_toast_auto_close.html.erb +58 -0
  49. data/app/pb_kits/playbook/pb_fixed_confirmation_toast/docs/_fixed_confirmation_toast_auto_close_rails.md +3 -0
  50. data/app/pb_kits/playbook/pb_fixed_confirmation_toast/docs/example.yml +1 -0
  51. data/app/pb_kits/playbook/pb_fixed_confirmation_toast/index.js +7 -5
  52. data/app/pb_kits/playbook/pb_form_group/form_group.html.erb +1 -6
  53. data/app/pb_kits/playbook/pb_form_pill/form_pill.html.erb +1 -1
  54. data/app/pb_kits/playbook/pb_home_address_street/_home_address_street.tsx +11 -7
  55. data/app/pb_kits/playbook/pb_home_address_street/docs/_home_address_street_formatting.html.erb +11 -0
  56. data/app/pb_kits/playbook/pb_home_address_street/docs/_home_address_street_formatting.jsx +22 -0
  57. data/app/pb_kits/playbook/pb_home_address_street/docs/_home_address_street_formatting_rails.md +1 -0
  58. data/app/pb_kits/playbook/pb_home_address_street/docs/_home_address_street_formatting_react.md +1 -0
  59. data/app/pb_kits/playbook/pb_home_address_street/docs/example.yml +2 -0
  60. data/app/pb_kits/playbook/pb_home_address_street/docs/index.js +1 -0
  61. data/app/pb_kits/playbook/pb_home_address_street/home_address_street.rb +11 -2
  62. data/app/pb_kits/playbook/pb_table/docs/_table_with_background_kit.html.erb +6 -9
  63. data/app/pb_kits/playbook/pb_table/docs/_table_with_background_kit.jsx +6 -9
  64. data/app/pb_kits/playbook/pb_table/docs/_table_with_collapsible_with_custom_content.jsx +12 -8
  65. data/app/pb_kits/playbook/pb_table/docs/_table_with_collapsible_with_custom_content_rails.html.erb +52 -0
  66. data/app/pb_kits/playbook/pb_table/docs/_table_with_collapsible_with_custom_content_rails.md +0 -0
  67. data/app/pb_kits/playbook/pb_table/docs/_table_with_collapsible_with_nested_rows_rails.html.erb +52 -0
  68. data/app/pb_kits/playbook/pb_table/docs/_table_with_collapsible_with_nested_rows_rails.md +3 -0
  69. data/app/pb_kits/playbook/pb_table/docs/_table_with_collapsible_with_nested_table_rails.html.erb +80 -0
  70. data/app/pb_kits/playbook/pb_table/docs/_table_with_collapsible_with_nested_table_rails.md +1 -0
  71. data/app/pb_kits/playbook/pb_table/docs/example.yml +3 -0
  72. data/app/pb_kits/playbook/pb_table/styles/_desktop_collapse.scss +26 -0
  73. data/app/pb_kits/playbook/pb_table/styles/_mobile.scss +0 -1
  74. data/app/pb_kits/playbook/pb_table/styles/_mobile_collapse.scss +25 -0
  75. data/app/pb_kits/playbook/pb_table/styles/_tablet_collapse.scss +25 -0
  76. data/app/pb_kits/playbook/pb_table/table_row.rb +1 -1
  77. data/app/pb_kits/playbook/pb_tooltip/_tooltip.tsx +3 -1
  78. data/app/pb_kits/playbook/pb_user/_user.tsx +3 -0
  79. data/app/pb_kits/playbook/pb_user/docs/_user_light_weight.html.erb +42 -0
  80. data/app/pb_kits/playbook/pb_user/docs/_user_light_weight.jsx +59 -0
  81. data/app/pb_kits/playbook/pb_user/docs/_user_light_weight.md +2 -0
  82. data/app/pb_kits/playbook/pb_user/docs/example.yml +2 -0
  83. data/app/pb_kits/playbook/pb_user/docs/index.js +1 -0
  84. data/app/pb_kits/playbook/pb_user/user.html.erb +1 -1
  85. data/app/pb_kits/playbook/pb_user/user.rb +1 -0
  86. data/app/pb_kits/playbook/pb_user/user.test.js +14 -0
  87. data/dist/chunks/_typeahead-W0hatdPs.js +36 -0
  88. data/dist/chunks/_weekday_stacked-C98LOqgG.js +45 -0
  89. data/dist/chunks/vendor.js +1 -1
  90. data/dist/menu.yml +6 -0
  91. data/dist/playbook-doc.js +1 -1
  92. data/dist/playbook-rails-react-bindings.js +1 -1
  93. data/dist/playbook-rails.js +1 -1
  94. data/dist/playbook.css +1 -1
  95. data/lib/playbook/pb_forms_global_props_helper.rb +136 -0
  96. data/lib/playbook/pb_forms_helper.rb +13 -4
  97. data/lib/playbook/version.rb +1 -1
  98. metadata +35 -6
  99. data/dist/chunks/_typeahead-BWwaAo_0.js +0 -36
  100. data/dist/chunks/_weekday_stacked-zyBCd1s8.js +0 -45
  101. /data/app/pb_kits/playbook/pb_fixed_confirmation_toast/docs/{_fixed_confirmation_toast_auto_close.md → _fixed_confirmation_toast_auto_close_react.md} +0 -0
@@ -1,18 +1,19 @@
1
1
  import React from "react";
2
2
  import { Avatar } from 'playbook-ui'
3
3
 
4
- const AvatarBadgeComponentOverlay = () => {
4
+ const AvatarBadgeComponentOverlay = (props) => {
5
5
  return (
6
6
  <div>
7
7
  <Avatar
8
8
  componentOverlay={{
9
9
  component: "badge",
10
10
  placement: "bottom-right",
11
- text: "12"
11
+ text: "12",
12
12
  }}
13
13
  imageUrl="https://randomuser.me/api/portraits/men/44.jpg"
14
14
  marginBottom="sm"
15
15
  size="sm"
16
+ {...props}
16
17
  />
17
18
 
18
19
  <Avatar
@@ -24,6 +25,8 @@ const AvatarBadgeComponentOverlay = () => {
24
25
  imageUrl="https://randomuser.me/api/portraits/men/44.jpg"
25
26
  marginBottom="sm"
26
27
  size="md"
28
+ {...props}
29
+
27
30
  />
28
31
 
29
32
  <Avatar
@@ -36,6 +39,8 @@ const AvatarBadgeComponentOverlay = () => {
36
39
  imageUrl="https://randomuser.me/api/portraits/men/44.jpg"
37
40
  marginBottom="sm"
38
41
  size="lg"
42
+ {...props}
43
+
39
44
  />
40
45
 
41
46
  <Avatar
@@ -48,7 +53,8 @@ const AvatarBadgeComponentOverlay = () => {
48
53
  imageUrl="https://randomuser.me/api/portraits/men/44.jpg"
49
54
  marginBottom="sm"
50
55
  size="xl"
51
- />
56
+ {...props}
57
+ />
52
58
  </div>
53
59
  )
54
60
  }
@@ -1,7 +1,7 @@
1
1
  import React from "react";
2
2
  import { Avatar } from 'playbook-ui'
3
3
 
4
- const AvatarCircleIconComponentOverlay = () => {
4
+ const AvatarCircleIconComponentOverlay = (props) => {
5
5
  return (
6
6
  <div>
7
7
  <Avatar
@@ -14,6 +14,7 @@ const AvatarCircleIconComponentOverlay = () => {
14
14
  imageUrl="https://randomuser.me/api/portraits/men/44.jpg"
15
15
  marginBottom="sm"
16
16
  size="sm"
17
+ {...props}
17
18
  />
18
19
 
19
20
  <Avatar
@@ -26,6 +27,7 @@ const AvatarCircleIconComponentOverlay = () => {
26
27
  imageUrl="https://randomuser.me/api/portraits/men/44.jpg"
27
28
  marginBottom="sm"
28
29
  size="md"
30
+ {...props}
29
31
  />
30
32
 
31
33
  <Avatar
@@ -38,6 +40,7 @@ const AvatarCircleIconComponentOverlay = () => {
38
40
  imageUrl="https://randomuser.me/api/portraits/men/44.jpg"
39
41
  marginBottom="sm"
40
42
  size="lg"
43
+ {...props}
41
44
  />
42
45
 
43
46
  <Avatar
@@ -50,7 +53,8 @@ const AvatarCircleIconComponentOverlay = () => {
50
53
  imageUrl="https://randomuser.me/api/portraits/men/44.jpg"
51
54
  marginBottom="sm"
52
55
  size="xl"
53
- />
56
+ {...props}
57
+ />
54
58
  </div>
55
59
  )
56
60
  }
@@ -65,7 +65,7 @@ module Playbook
65
65
  end
66
66
 
67
67
  def tag
68
- link ? "a" : "button"
68
+ link && !disabled ? "a" : "button"
69
69
  end
70
70
 
71
71
  def valid_emoji(icon)
@@ -0,0 +1,3 @@
1
+ .pb_copy_button_kit {
2
+
3
+ }
@@ -0,0 +1,92 @@
1
+
2
+ import React, { useState } from 'react'
3
+ import classnames from 'classnames'
4
+ import { buildAriaProps, buildCss, buildDataProps } from '../utilities/props'
5
+ import { globalProps } from '../utilities/globalProps'
6
+
7
+ import Button from '../pb_button/_button'
8
+ import Tooltip from '../pb_tooltip/_tooltip'
9
+
10
+ type CopyButtonProps = {
11
+ aria?: { [key: string]: string },
12
+ className?: string,
13
+ data?: { [key: string]: string },
14
+ id?: string,
15
+ from?: string,
16
+ text?: string,
17
+ tooltipPlacement?: "top" | "right" | "bottom" | "left",
18
+ tooltipText?: string,
19
+ value?: string,
20
+ }
21
+
22
+ const CopyButton = (props: CopyButtonProps) => {
23
+ const {
24
+ aria = {},
25
+ className,
26
+ data = {},
27
+ from = '',
28
+ id,
29
+ text= 'Copy',
30
+ tooltipPlacement= 'bottom',
31
+ tooltipText = 'Copied!',
32
+ value = '',
33
+ } = props
34
+
35
+ const [copied, setCopied] = useState(false)
36
+
37
+ const ariaProps = buildAriaProps(aria)
38
+ const dataProps = buildDataProps(data)
39
+ const classes = classnames(buildCss('pb_copy_button_kit'), globalProps(props), className)
40
+
41
+ const copy = () => {
42
+ if (!from && !value) {
43
+ return
44
+ }
45
+
46
+ if (value) {
47
+ navigator.clipboard.writeText(value)
48
+ } else if (from) {
49
+ const copyElement = document.getElementById(from);
50
+ let copyText = copyElement?.innerText
51
+
52
+ if (copyElement instanceof HTMLTextAreaElement || copyElement instanceof HTMLInputElement) {
53
+ copyText = copyElement.value;
54
+ }
55
+
56
+ if (copyText) {
57
+ navigator.clipboard.writeText(copyText)
58
+ }
59
+ }
60
+
61
+ setCopied(true)
62
+
63
+ setTimeout(() => {
64
+ setCopied(false)
65
+ }, 1000);
66
+ }
67
+
68
+ return (
69
+ <div
70
+ {...ariaProps}
71
+ {...dataProps}
72
+ className={classes}
73
+ id={id}
74
+ >
75
+ <Tooltip
76
+ forceOpenTooltip={copied}
77
+ placement={tooltipPlacement}
78
+ showTooltip={false}
79
+ text={tooltipText}
80
+ >
81
+ <Button
82
+ icon='copy'
83
+ onClick={copy}
84
+ >
85
+ { text }
86
+ </Button>
87
+ </Tooltip>
88
+ </div>
89
+ )
90
+ }
91
+
92
+ export default CopyButton
@@ -0,0 +1,64 @@
1
+ import React from 'react';
2
+ import { CopyButton } from 'playbook-ui'
3
+ import { ensureAccessible, renderKit, render, fireEvent, screen } from '../utilities/test-utils'
4
+
5
+ const props = {
6
+ data: { testid: 'default', value: 'copy' }
7
+ }
8
+
9
+ test('returns namespaced class name', () => {
10
+ const kit = renderKit(CopyButton, props)
11
+ expect(kit).toBeInTheDocument()
12
+ expect(kit).toHaveClass('pb_copy_button_kit')
13
+ })
14
+
15
+ it('should be accessible', async () => {
16
+ ensureAccessible(CopyButton, props)
17
+ })
18
+
19
+ // It's difficult to actually use navigator.clipboard.readText, so we mock
20
+ it('copies the value to clipboard and pastes it into an input', async () => {
21
+ Object.defineProperty(global, 'navigator', {
22
+ value: {
23
+ clipboard: {
24
+ writeText: jest.fn().mockResolvedValueOnce(undefined),
25
+ },
26
+ },
27
+ writable: true,
28
+ })
29
+
30
+ render(<CopyButton {...props} />)
31
+
32
+ const copyButton = screen.getByTestId('default')
33
+ fireEvent.click(copyButton)
34
+
35
+ await navigator.clipboard.writeText('copy')
36
+
37
+ expect(navigator.clipboard.writeText).toHaveBeenCalledWith("copy");
38
+ })
39
+
40
+ test('passes text and tooltip props to button', () => {
41
+ render(
42
+ <CopyButton
43
+ data={{ testid: 'text-test' }}
44
+ text={"text"}
45
+ tooltipPlacement="right"
46
+ tooltipText="Text copied!"
47
+ value="copy"
48
+ />
49
+ )
50
+
51
+ const content = screen.getByText("text")
52
+ expect(content).toHaveTextContent("text")
53
+
54
+ const kit = screen.getByTestId('text-test')
55
+ const button = kit.querySelector('.pb_button_kit_primary_inline_enabled')
56
+ expect(button).toBeInTheDocument()
57
+
58
+ fireEvent.click(button)
59
+ const tooltipContent = screen.getByText("Text copied!")
60
+ expect(tooltipContent).toHaveTextContent("Text copied!")
61
+
62
+ const tooltip = kit.querySelector('.pb_tooltip_kit')
63
+ expect(tooltip).toBeInTheDocument()
64
+ })
@@ -0,0 +1,21 @@
1
+ import React from 'react'
2
+ import { CopyButton, Textarea } from 'playbook-ui'
3
+
4
+ const CopyButtonDefault = (props) => (
5
+ <div>
6
+ <CopyButton
7
+ {...props}
8
+ text="Copy Text"
9
+ tooltipPlacement="right"
10
+ tooltipText="Text copied!"
11
+ value="Playbook makes it easy to support bleeding edge, or legacy systems. Use Playbook’s 200+ components and end-to-end design language to create simple, intuitive and beautiful experiences with ease."
12
+ />
13
+
14
+ <Textarea
15
+ {...props}
16
+ placeholder="Copy and paste here"
17
+ />
18
+ </div>
19
+ )
20
+
21
+ export default CopyButtonDefault
@@ -0,0 +1,45 @@
1
+ import React, { useState } from 'react'
2
+ import { CopyButton, Body, TextInput, Textarea } from 'playbook-ui'
3
+
4
+ const CopyButtonFrom = (props) => {
5
+ const [text, setText] = useState("Copy this text input text")
6
+
7
+ const handleChange = (event) => {
8
+ setText(event.target.value);
9
+ }
10
+
11
+ return (<div>
12
+ <Body id="body">Copy this body text!</Body>
13
+ <CopyButton
14
+ {...props}
15
+ from="body"
16
+ marginBottom="sm"
17
+ text="Copy Body text"
18
+ tooltipPlacement="right"
19
+ tooltipText="Body text copied!"
20
+ />
21
+
22
+ <TextInput
23
+ {...props}
24
+ id="textinput"
25
+ onChange={handleChange}
26
+ value={text}
27
+ />
28
+ <CopyButton
29
+ {...props}
30
+ from="textinput"
31
+ marginBottom="sm"
32
+ text="Copy Text Input"
33
+ tooltipPlacement="right"
34
+ tooltipText="Text input copied!"
35
+ />
36
+
37
+ <Textarea
38
+ {...props}
39
+ placeholder="Copy and paste here"
40
+ />
41
+ </div>
42
+ )
43
+ }
44
+
45
+ export default CopyButtonFrom
@@ -0,0 +1 @@
1
+ Provide an element's ID as the `from` parameter, and its text will be copied. If the element is an input, its `value` will be copied; otherwise, the `innerText` will be used. Additionally, if a `value` prop is provided, it will override the content from the `from` element and be copied instead.
@@ -0,0 +1,8 @@
1
+ examples:
2
+
3
+
4
+ react:
5
+ - copy_button_default: Default
6
+ - copy_button_from: Copy From
7
+
8
+
@@ -0,0 +1,2 @@
1
+ export { default as CopyButtonDefault } from './_copy_button_default.jsx'
2
+ export { default as CopyButtonFrom } from './_copy_button_from.jsx'
@@ -14,6 +14,7 @@ type PbDateProps = {
14
14
  alignment?: "left" | "center" | "right";
15
15
  aria?: { [key: string]: string };
16
16
  className?: string;
17
+ dark?: boolean;
17
18
  data?: { [key: string]: string };
18
19
  htmlOptions?: { [key: string]: string | number | boolean | (() => void) };
19
20
  id?: string;
@@ -29,6 +30,7 @@ const PbDate = (props: PbDateProps): React.ReactElement => {
29
30
  aria = {},
30
31
  alignment = "left",
31
32
  className,
33
+ dark = false,
32
34
  data = {},
33
35
  htmlOptions = {},
34
36
  id,
@@ -56,7 +58,7 @@ const PbDate = (props: PbDateProps): React.ReactElement => {
56
58
  );
57
59
 
58
60
  return (
59
- <div
61
+ <div
60
62
  {...ariaProps}
61
63
  {...dataProps}
62
64
  {...htmlProps}
@@ -93,7 +95,9 @@ const PbDate = (props: PbDateProps): React.ReactElement => {
93
95
  </>
94
96
  : size == "md" || size == "lg"
95
97
  ? (
96
- <Title size={4}
98
+ <Title
99
+ dark={dark}
100
+ size={4}
97
101
  tag="h4"
98
102
  >
99
103
  {showIcon && (
@@ -127,6 +131,7 @@ const PbDate = (props: PbDateProps): React.ReactElement => {
127
131
  <>
128
132
  {showIcon && (
129
133
  <Caption className="pb_icon_kit_container"
134
+ dark={dark}
130
135
  tag="span"
131
136
  >
132
137
  <Icon fixedWidth
@@ -138,15 +143,20 @@ const PbDate = (props: PbDateProps): React.ReactElement => {
138
143
 
139
144
  {showDayOfWeek && (
140
145
  <>
141
- <Caption tag="div">{weekday}</Caption>
146
+ <Caption dark={dark}
147
+ tag="div">
148
+ {weekday}
149
+ </Caption>
142
150
  <Caption color="light"
151
+ dark={dark}
143
152
  tag="div"
144
153
  text=" • "
145
154
  />
146
155
  </>
147
156
  )}
148
157
 
149
- <Caption tag="span">
158
+ <Caption dark={dark}
159
+ tag="span">
150
160
  {month} {day}
151
161
  {currentYear != year && <>{`, ${year}`}</>}
152
162
  </Caption>
@@ -18,7 +18,7 @@ const DateDefault = (props) => {
18
18
  value={"2012-08-03"}
19
19
  {...props}
20
20
  />
21
- <Caption>{"(Hyphenated Date)"}</Caption>
21
+ <Caption {...props}>{"(Hyphenated Date)"}</Caption>
22
22
  </div>
23
23
 
24
24
  <br />
@@ -56,6 +56,7 @@ const DateDefault = (props) => {
56
56
  <Title
57
57
  size={4}
58
58
  text={"(Hyphenated Date)"}
59
+ {...props}
59
60
  />
60
61
  </div>
61
62
 
@@ -4,7 +4,8 @@ import { Caption, Date as FormattedDate, Title } from 'playbook-ui'
4
4
  const DateUnstyled = (props) => {
5
5
  return (
6
6
  <>
7
- <Caption size="xs"
7
+ <Caption {...props}
8
+ size="xs"
8
9
  text="Basic unstyled example"
9
10
  />
10
11
  <FormattedDate
@@ -15,10 +16,14 @@ const DateUnstyled = (props) => {
15
16
 
16
17
  <br />
17
18
 
18
- <Caption size="xs"
19
+ <Caption {...props}
20
+
21
+ size="xs"
19
22
  text="Example with wrapping typography kit"
20
23
  />
21
- <Title size={1}>
24
+ <Title {...props}
25
+ size={1}
26
+ >
22
27
  <FormattedDate
23
28
  unstyled
24
29
  value={new Date('25 Dec 1995')}
@@ -28,10 +33,13 @@ const DateUnstyled = (props) => {
28
33
 
29
34
  <br />
30
35
 
31
- <Caption size="xs"
36
+ <Caption {...props}
37
+ size="xs"
32
38
  text="Example with icon + subcaption"
33
39
  />
34
- <Caption size="xs">
40
+ <Caption {...props}
41
+ size="xs"
42
+ >
35
43
  <FormattedDate
36
44
  showDayOfWeek
37
45
  showIcon
@@ -0,0 +1,193 @@
1
+ <% content_for :helper_methods do %>
2
+ <% def badge_properties(container)
3
+ case container
4
+ when "To Do"
5
+ { text: "queue", variant: "warning" }
6
+ when "In Progress"
7
+ { text: "progress", variant: "primary" }
8
+ else
9
+ { text: "done", variant: "success" }
10
+ end
11
+ end %>
12
+ <% end %>
13
+
14
+ <% containers = [
15
+ "To Do",
16
+ "In Progress",
17
+ "Done"
18
+ ] %>
19
+
20
+ <% items_data = [
21
+ {
22
+ id: "11",
23
+ container: "To Do",
24
+ title: "Task 1",
25
+ description: "Bug fixes",
26
+ assignee_name: "Terry Miles",
27
+ assignee_img: "https://randomuser.me/api/portraits/men/44.jpg",
28
+ },
29
+ {
30
+ id: "12",
31
+ container: "To Do",
32
+ title: "Task 2",
33
+ description: "Documentation",
34
+ assignee_name: "Sophia Miles",
35
+ assignee_img: "https://randomuser.me/api/portraits/women/8.jpg",
36
+ },
37
+ {
38
+ id: "13",
39
+ container: "In Progress",
40
+ title: "Task 3",
41
+ description: "Add a variant",
42
+ assignee_name: "Alice Jones",
43
+ assignee_img: "https://randomuser.me/api/portraits/women/10.jpg",
44
+ },
45
+ {
46
+ id: "14",
47
+ container: "To Do",
48
+ title: "Task 4",
49
+ description: "Add jest tests",
50
+ assignee_name: "Mike James",
51
+ assignee_img: "https://randomuser.me/api/portraits/men/8.jpg",
52
+ },
53
+ {
54
+ id: "15",
55
+ container: "Done",
56
+ title: "Task 5",
57
+ description: "Alpha testing",
58
+ assignee_name: "James Guy",
59
+ assignee_img: "https://randomuser.me/api/portraits/men/18.jpg",
60
+ },
61
+ {
62
+ id: "16",
63
+ container: "In Progress",
64
+ title: "Task 6",
65
+ description: "Release",
66
+ assignee_name: "Sally Jones",
67
+ assignee_img: "https://randomuser.me/api/portraits/women/28.jpg",
68
+ },
69
+ ] %>
70
+
71
+ <%= pb_rails("draggable", props: { initial_items: items_data }) do %>
72
+ <%= pb_rails("flex", props: { justify_content: "center" }) do %>
73
+ <% containers.each do |container| %>
74
+ <%= pb_rails("draggable/draggable_container", props: {
75
+ container: container,
76
+ width: "xs",
77
+ height: "xs",
78
+ padding: "sm",
79
+ data: { container: container }
80
+ }) do %>
81
+ <%= pb_rails("caption", props: { text_align: "center" }) do %><%= container %><% end %>
82
+ <%= pb_rails("flex", props: {align_items: "stretch", orientation: "column"}) do %>
83
+ <% items_data.select { |item| item[:container] == container }.each do |item| %>
84
+ <%= pb_rails("draggable/draggable_item", props: {
85
+ container: container,
86
+ drag_id: item[:id]
87
+ }) do %>
88
+ <%= pb_rails("card", props: { margin_bottom: "sm", padding: "sm"}) do %>
89
+ <%= pb_rails("flex", props: { justify: "between" }) do %>
90
+ <%= pb_rails("flex/flex_item") do %>
91
+ <%= pb_rails("flex") do %>
92
+ <%= pb_rails("avatar", props: {
93
+ image_url: item[:assignee_img],
94
+ name: item[:assignee_name],
95
+ size: "xxs"
96
+ }) %>
97
+ <%= pb_rails("title", props: {
98
+ padding_left: "xs",
99
+ size: 4,
100
+ text: item[:title]
101
+ }) %>
102
+ <% end %>
103
+ <% end %>
104
+ <%= pb_rails("badge", props: {
105
+ margin_left: "sm",
106
+ rounded: true,
107
+ text: badge_properties(container)[:text],
108
+ variant: badge_properties(container)[:variant],
109
+ data: {
110
+ pb_status_badge: true,
111
+ pb_tag_type: "badge"
112
+ }
113
+ }) %>
114
+ <% end %>
115
+ <%= pb_rails("body", props: { padding_top: "xs", text: item[:description] }) %>
116
+ <% end %>
117
+ <% end %>
118
+ <% end %>
119
+ <% end %>
120
+ <% end %>
121
+ <% end %>
122
+ <% end %>
123
+ <% end %>
124
+
125
+ <script>
126
+ const initBadgeUpdates = function() {
127
+ const updateBadge = function(container) {
128
+ switch (container) {
129
+ case 'To Do':
130
+ return { text: 'queue', variant: 'warning' };
131
+ case 'In Progress':
132
+ return { text: 'progress', variant: 'primary' };
133
+ case 'Done':
134
+ return { text: 'done', variant: 'success' };
135
+ default:
136
+ return { text: 'queue', variant: 'warning' };
137
+ }
138
+ };
139
+
140
+ let draggedElement = null;
141
+ let startContainer = null;
142
+
143
+ document.querySelectorAll('.pb_draggable_item').forEach(function(item) {
144
+ item.addEventListener('dragstart', function(event) {
145
+ draggedElement = event.target;
146
+ startContainer = draggedElement.closest('.pb_draggable_container');
147
+ });
148
+ });
149
+
150
+ document.querySelectorAll('.pb_draggable_container').forEach(function(container) {
151
+ container.addEventListener('drop', function(event) {
152
+ event.preventDefault();
153
+
154
+ if (draggedElement) {
155
+ setTimeout(function() {
156
+ const currentContainer = container;
157
+
158
+ if (currentContainer && startContainer && currentContainer !== startContainer) {
159
+ const containerData = currentContainer.getAttribute('data-container');
160
+ const badge = draggedElement.querySelector('[data-pb-status-badge]');
161
+
162
+ if (badge && containerData) {
163
+ const newProperties = updateBadge(containerData);
164
+ const span = badge.querySelector('span');
165
+
166
+ if (span) {
167
+ span.textContent = newProperties.text;
168
+ }
169
+
170
+ badge.classList.forEach(function(className) {
171
+ if (className.includes('pb_badge_kit_') || className === 'ml_sm') {
172
+ badge.classList.remove(className);
173
+ }
174
+ });
175
+
176
+ badge.className = `pb_badge_kit_${newProperties.variant}_rounded ml_sm`;
177
+ }
178
+ }
179
+
180
+ draggedElement = null;
181
+ startContainer = null;
182
+ }, 50);
183
+ }
184
+ });
185
+ });
186
+ };
187
+
188
+ if (document.readyState === 'loading') {
189
+ document.addEventListener('DOMContentLoaded', initBadgeUpdates);
190
+ } else {
191
+ initBadgeUpdates();
192
+ }
193
+ </script>
@@ -0,0 +1 @@
1
+ The Draggable kit can also be used to achieve more complex, multiple container functionality as shown here. This complex usage requires the full subcomponent structure.