playbook_ui 15.3.0.pre.alpha.PLAY2001selectablecarddisabledstyles11833 → 15.3.0.pre.alpha.PLAY2012currencyallownonstring11930
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.
- checksums.yaml +4 -4
- data/app/pb_kits/playbook/pb_advanced_table/Hooks/useTableState.ts +15 -0
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_pagination_with_props.jsx +4 -1
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_pagination_with_props.md +3 -2
- data/app/pb_kits/playbook/pb_currency/_currency.tsx +17 -7
- data/app/pb_kits/playbook/pb_currency/currency.rb +20 -8
- data/app/pb_kits/playbook/pb_currency/currency.test.js +42 -0
- data/app/pb_kits/playbook/pb_currency/docs/_currency_variants.html.erb +1 -1
- data/app/pb_kits/playbook/pb_currency/docs/_currency_variants.jsx +1 -1
- data/app/pb_kits/playbook/pb_currency/docs/_currency_variants.md +1 -0
- data/app/pb_kits/playbook/pb_date_picker/date_picker.rb +2 -0
- data/app/pb_kits/playbook/pb_icon_circle/docs/_icon_circle_color.md +1 -1
- data/app/pb_kits/playbook/pb_icon_stat_value/docs/_icon_stat_value_color.html.erb +7 -14
- data/app/pb_kits/playbook/pb_icon_stat_value/docs/_icon_stat_value_color.jsx +6 -15
- data/app/pb_kits/playbook/pb_popover/_popover.tsx +69 -34
- data/app/pb_kits/playbook/pb_popover/docs/_popover_append_to.jsx +68 -0
- data/app/pb_kits/playbook/pb_popover/docs/_popover_append_to_react.md +1 -0
- data/app/pb_kits/playbook/pb_popover/docs/example.yml +1 -0
- data/app/pb_kits/playbook/pb_popover/docs/index.js +1 -0
- data/app/pb_kits/playbook/pb_popover/popover.test.js +80 -0
- data/app/pb_kits/playbook/pb_table/docs/_table_with_dynamic_collapsible.html.erb +63 -0
- data/app/pb_kits/playbook/pb_table/docs/_table_with_dynamic_collapsible.jsx +89 -0
- data/app/pb_kits/playbook/pb_table/docs/_table_with_dynamic_collapsible_rails.md +4 -0
- data/app/pb_kits/playbook/pb_table/docs/_table_with_dynamic_collapsible_react.md +3 -0
- data/app/pb_kits/playbook/pb_table/docs/example.yml +2 -0
- data/app/pb_kits/playbook/pb_table/docs/index.js +1 -0
- data/app/pb_kits/playbook/pb_text_input/text_input.rb +2 -2
- data/app/pb_kits/playbook/pb_typeahead/_typeahead.test.jsx +41 -0
- data/app/pb_kits/playbook/pb_typeahead/_typeahead.tsx +23 -9
- data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_custom_options.html.erb +64 -0
- data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_custom_options.jsx +70 -0
- data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_custom_options.md +1 -0
- data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_default_options.html.erb +67 -1
- data/app/pb_kits/playbook/pb_typeahead/docs/_typeahead_default_value.jsx +68 -6
- data/app/pb_kits/playbook/pb_typeahead/docs/example.yml +2 -0
- data/app/pb_kits/playbook/pb_typeahead/docs/index.js +2 -1
- data/dist/chunks/{_line_graph-h5H-imfn.js → _line_graph-CqE0-dq5.js} +1 -1
- data/dist/chunks/_typeahead-3ZAbZUqU.js +6 -0
- data/dist/chunks/{_weekday_stacked-c6_R08J-.js → _weekday_stacked-D6fNzH0S.js} +2 -2
- data/dist/chunks/vendor.js +1 -1
- data/dist/playbook-doc.js +2 -2
- data/dist/playbook-rails-react-bindings.js +1 -1
- data/dist/playbook-rails.js +1 -1
- data/lib/playbook/version.rb +1 -1
- metadata +16 -6
- data/dist/chunks/_typeahead-U8AjZIIW.js +0 -6
- /data/app/pb_kits/playbook/pb_popover/docs/{_popover_append_to.md → _popover_append_to_rails.md} +0 -0
checksums.yaml
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
---
|
|
2
2
|
SHA256:
|
|
3
|
-
metadata.gz:
|
|
4
|
-
data.tar.gz:
|
|
3
|
+
metadata.gz: cd5097981398626c3953e3fe56f20fc47cc519f82061f062197e810e12d50177
|
|
4
|
+
data.tar.gz: 4642684e43e2da087c898fc24e95576261bad559a32fb0cbdf8940780f2ac12a
|
|
5
5
|
SHA512:
|
|
6
|
-
metadata.gz:
|
|
7
|
-
data.tar.gz:
|
|
6
|
+
metadata.gz: f38257a4548743ef4e0c3ae67a52b4a98dfb86ab0a75de3886112958d3cb7e82b3264486ab71bf7322b0749f0621b94dd3d72ed12f91c16925391aee6b4c3c0e
|
|
7
|
+
data.tar.gz: 2820e3e0d73fb5b8e559444f1d93060ed495a98d8a54497c4d489e97dddf44e7876a59b5c2df6b3f61cbbae8a6f339a9e368f65cb9f930552eee3da25e609528
|
|
@@ -250,6 +250,21 @@ export function useTableState({
|
|
|
250
250
|
}
|
|
251
251
|
}, [pagination, inlineRowLoading, paginationProps?.pageIndex, paginationProps?.pageSize]);
|
|
252
252
|
|
|
253
|
+
// Call the callback with the new page index when localPagination.pageIndex changes (with inlineRowLoading)
|
|
254
|
+
useEffect(() => {
|
|
255
|
+
if (pagination && inlineRowLoading && paginationProps?.onPageChange) {
|
|
256
|
+
paginationProps.onPageChange(localPagination.pageIndex);
|
|
257
|
+
}
|
|
258
|
+
}, [localPagination.pageIndex, pagination, inlineRowLoading, paginationProps]);
|
|
259
|
+
|
|
260
|
+
// Call the callback with the new page index when localPagination.pageIndex changes (without inlineRowLoading)
|
|
261
|
+
useEffect(() => {
|
|
262
|
+
if (pagination && !inlineRowLoading && paginationProps?.onPageChange) {
|
|
263
|
+
const currentPageIndex = table.getState().pagination.pageIndex;
|
|
264
|
+
paginationProps.onPageChange(currentPageIndex);
|
|
265
|
+
}
|
|
266
|
+
}, [table.getState().pagination.pageIndex, pagination, inlineRowLoading, paginationProps]);
|
|
267
|
+
|
|
253
268
|
// Check if table has any sub-rows
|
|
254
269
|
const hasAnySubRows = table.getRowModel().rows.some(row => row.subRows && row.subRows.length > 0);
|
|
255
270
|
const selectedRowsLength = Object.keys(table.getState().rowSelection).length;
|
|
@@ -1,5 +1,6 @@
|
|
|
1
|
-
`paginationProps` is an optional prop that can be used to further customize pagination for the AdvancedTable. This prop is an object with
|
|
1
|
+
`paginationProps` is an optional prop that can be used to further customize pagination for the AdvancedTable. This prop is an object with the following optional items:
|
|
2
2
|
|
|
3
3
|
- `pageIndex`: An optional prop to set which page is set to open on initial load. Default is '0'
|
|
4
4
|
- `pageSize`: An optional prop to set total number of rows for each page before the Table paginates. Default is '20'
|
|
5
|
-
- `range`: The range prop determines how many pages to display in the Pagination component. Regardless of this value, the first two and last two pages are always visible to facilitate navigation to the beginning and end of the pagination. If these always-visible pages fall within the specified range, they are included in the display. If they fall outside the range, the pagination will show additional pages up to the number defined by the range prop. See [here for more details](https://playbook.powerapp.cloud/kits/pagination/react#default). Default is set to '5'
|
|
5
|
+
- `range`: The range prop determines how many pages to display in the Pagination component. Regardless of this value, the first two and last two pages are always visible to facilitate navigation to the beginning and end of the pagination. If these always-visible pages fall within the specified range, they are included in the display. If they fall outside the range, the pagination will show additional pages up to the number defined by the range prop. See [here for more details](https://playbook.powerapp.cloud/kits/pagination/react#default). Default is set to '5'
|
|
6
|
+
- `onPageChange`: A callback function that gives to access to the current page index.
|
|
@@ -11,7 +11,7 @@ import Title from '../pb_title/_title'
|
|
|
11
11
|
type CurrencyProps = {
|
|
12
12
|
abbreviate?: boolean,
|
|
13
13
|
align?: 'center' | 'left' | 'right',
|
|
14
|
-
amount: string,
|
|
14
|
+
amount: string | number,
|
|
15
15
|
aria?: {[key:string]:string},
|
|
16
16
|
className?: string,
|
|
17
17
|
dark?: boolean,
|
|
@@ -59,6 +59,16 @@ const Currency = (props: CurrencyProps): React.ReactElement => {
|
|
|
59
59
|
commaSeparator = false,
|
|
60
60
|
} = props
|
|
61
61
|
|
|
62
|
+
// Convert numeric input to string format
|
|
63
|
+
const convertAmount = (input: string | number): string => {
|
|
64
|
+
if (typeof input === 'number') {
|
|
65
|
+
return input.toFixed(2)
|
|
66
|
+
}
|
|
67
|
+
return input
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
const currencyAmount = convertAmount(amount)
|
|
71
|
+
|
|
62
72
|
const emphasizedClass = emphasized ? '' : '_deemphasized'
|
|
63
73
|
|
|
64
74
|
let variantClass
|
|
@@ -68,7 +78,7 @@ const Currency = (props: CurrencyProps): React.ReactElement => {
|
|
|
68
78
|
variantClass = '_bold'
|
|
69
79
|
}
|
|
70
80
|
|
|
71
|
-
const [whole, decimal = '00'] =
|
|
81
|
+
const [whole, decimal = '00'] = currencyAmount.split('.')
|
|
72
82
|
const ariaProps = buildAriaProps(aria)
|
|
73
83
|
const dataProps = buildDataProps(data)
|
|
74
84
|
const htmlProps = buildHtmlProps(htmlOptions)
|
|
@@ -92,19 +102,19 @@ const Currency = (props: CurrencyProps): React.ReactElement => {
|
|
|
92
102
|
return isAmount ? num.slice(0, -1) : isUnit ? num.slice(-1) : ''
|
|
93
103
|
}
|
|
94
104
|
|
|
95
|
-
const getMatchingDecimalAmount = decimals === "matching" ?
|
|
105
|
+
const getMatchingDecimalAmount = decimals === "matching" ? currencyAmount : whole
|
|
96
106
|
const getMatchingDecimalValue = decimals === "matching" ? '' : `.${decimal}`
|
|
97
107
|
|
|
98
108
|
const formatAmount = (amount: string) => {
|
|
99
109
|
if (!commaSeparator) return amount;
|
|
100
|
-
|
|
110
|
+
|
|
101
111
|
const [wholePart, decimalPart] = amount.split('.');
|
|
102
112
|
const formattedWhole = new Intl.NumberFormat('en-US').format(parseInt(wholePart));
|
|
103
113
|
return decimalPart ? `${formattedWhole}.${decimalPart}` : formattedWhole;
|
|
104
114
|
}
|
|
105
115
|
|
|
106
116
|
const swapNegative = size === "sm" && symbol !== ""
|
|
107
|
-
const handleNegative =
|
|
117
|
+
const handleNegative = currencyAmount.startsWith("-") && swapNegative ? "-" : ""
|
|
108
118
|
const getAbsoluteAmount = (amountString: string) => amountString.replace(/^-/,'')
|
|
109
119
|
const getAbbrOrFormatAmount = abbreviate ? getAbbreviatedValue('amount') : formatAmount(getMatchingDecimalAmount)
|
|
110
120
|
const getAmount = swapNegative ? getAbsoluteAmount(getAbbrOrFormatAmount) : getAbbrOrFormatAmount
|
|
@@ -152,7 +162,7 @@ const Currency = (props: CurrencyProps): React.ReactElement => {
|
|
|
152
162
|
>
|
|
153
163
|
{handleNegative}{symbol}
|
|
154
164
|
</Body>
|
|
155
|
-
|
|
165
|
+
|
|
156
166
|
<Title
|
|
157
167
|
className="pb_currency_value"
|
|
158
168
|
dark={dark}
|
|
@@ -160,7 +170,7 @@ const Currency = (props: CurrencyProps): React.ReactElement => {
|
|
|
160
170
|
>
|
|
161
171
|
{getAmount}
|
|
162
172
|
</Title>
|
|
163
|
-
|
|
173
|
+
|
|
164
174
|
<Body
|
|
165
175
|
className="unit"
|
|
166
176
|
color="light"
|
|
@@ -17,8 +17,7 @@ module Playbook
|
|
|
17
17
|
prop :symbol, type: Playbook::Props::String,
|
|
18
18
|
default: "$"
|
|
19
19
|
|
|
20
|
-
prop :amount,
|
|
21
|
-
required: true
|
|
20
|
+
prop :amount, required: true
|
|
22
21
|
|
|
23
22
|
prop :unit, type: Playbook::Props::String,
|
|
24
23
|
required: false
|
|
@@ -92,7 +91,7 @@ module Playbook
|
|
|
92
91
|
end
|
|
93
92
|
|
|
94
93
|
def negative_sign
|
|
95
|
-
|
|
94
|
+
currency_amount.starts_with?("-") && swap_negative ? "-" : ""
|
|
96
95
|
end
|
|
97
96
|
|
|
98
97
|
def body_props
|
|
@@ -117,10 +116,23 @@ module Playbook
|
|
|
117
116
|
end
|
|
118
117
|
end
|
|
119
118
|
|
|
119
|
+
def currency_amount
|
|
120
|
+
@currency_amount ||= convert_amount(amount)
|
|
121
|
+
end
|
|
122
|
+
|
|
120
123
|
private
|
|
121
124
|
|
|
125
|
+
# Convert numeric input to string format
|
|
126
|
+
def convert_amount(input)
|
|
127
|
+
if input.is_a?(Numeric)
|
|
128
|
+
format("%.2f", input)
|
|
129
|
+
else
|
|
130
|
+
input.to_s
|
|
131
|
+
end
|
|
132
|
+
end
|
|
133
|
+
|
|
122
134
|
def whole_value
|
|
123
|
-
value =
|
|
135
|
+
value = currency_amount.split(".").first
|
|
124
136
|
if comma_separator
|
|
125
137
|
number_with_delimiter(value.gsub(",", ""))
|
|
126
138
|
else
|
|
@@ -129,7 +141,7 @@ module Playbook
|
|
|
129
141
|
end
|
|
130
142
|
|
|
131
143
|
def decimal_value
|
|
132
|
-
|
|
144
|
+
currency_amount.split(".")[1] || "00"
|
|
133
145
|
end
|
|
134
146
|
|
|
135
147
|
def units_element
|
|
@@ -147,7 +159,7 @@ module Playbook
|
|
|
147
159
|
end
|
|
148
160
|
|
|
149
161
|
def abbreviated_value(index = 0..-2)
|
|
150
|
-
value =
|
|
162
|
+
value = currency_amount.split(".").first.gsub(",", "").to_i
|
|
151
163
|
abbreviated_num = number_to_human(value, units: { thousand: "K", million: "M", billion: "B", trillion: "T" }).gsub(/\s+/, "")
|
|
152
164
|
abbreviated_num[index]
|
|
153
165
|
end
|
|
@@ -174,9 +186,9 @@ module Playbook
|
|
|
174
186
|
|
|
175
187
|
if decimals == "matching"
|
|
176
188
|
if comma_separator
|
|
177
|
-
number_with_delimiter(
|
|
189
|
+
number_with_delimiter(currency_amount.gsub(",", ""))
|
|
178
190
|
else
|
|
179
|
-
|
|
191
|
+
currency_amount
|
|
180
192
|
end
|
|
181
193
|
else
|
|
182
194
|
whole_value
|
|
@@ -133,3 +133,45 @@ test('handles negative amounts correctly', () => {
|
|
|
133
133
|
expect(screen.getByTestId('test-negative-no-symbol')).toHaveTextContent('-400.50')
|
|
134
134
|
expect(screen.getByTestId('test-negative-medium-size')).toHaveTextContent('$-500.55')
|
|
135
135
|
})
|
|
136
|
+
|
|
137
|
+
test('handles numeric amounts correctly', () => {
|
|
138
|
+
render(
|
|
139
|
+
<>
|
|
140
|
+
<Currency
|
|
141
|
+
amount={320}
|
|
142
|
+
data={{ testid: 'test-numeric-default' }}
|
|
143
|
+
/>
|
|
144
|
+
<Currency
|
|
145
|
+
abbreviate
|
|
146
|
+
amount={3200000}
|
|
147
|
+
data={{ testid: 'test-numeric-millions' }}
|
|
148
|
+
/>
|
|
149
|
+
<Currency
|
|
150
|
+
amount={123456.78}
|
|
151
|
+
commaSeparator
|
|
152
|
+
data={{ testid: 'test-numeric-comma-decimals' }}
|
|
153
|
+
/>
|
|
154
|
+
<Currency
|
|
155
|
+
amount={400.50}
|
|
156
|
+
data={{ testid: 'test-numeric-no-symbol' }}
|
|
157
|
+
symbol=""
|
|
158
|
+
/>
|
|
159
|
+
<Currency
|
|
160
|
+
amount={500.55}
|
|
161
|
+
data={{ testid: 'test-numeric-medium-size' }}
|
|
162
|
+
size="md"
|
|
163
|
+
/>
|
|
164
|
+
<Currency
|
|
165
|
+
amount={-600.70}
|
|
166
|
+
data={{ testid: 'test-numeric-negative' }}
|
|
167
|
+
/>
|
|
168
|
+
</>
|
|
169
|
+
)
|
|
170
|
+
|
|
171
|
+
expect(screen.getByTestId('test-numeric-default')).toHaveTextContent('$320')
|
|
172
|
+
expect(screen.getByTestId('test-numeric-millions')).toHaveTextContent('$3.2M')
|
|
173
|
+
expect(screen.getByTestId('test-numeric-comma-decimals')).toHaveTextContent('$123,456.78')
|
|
174
|
+
expect(screen.getByTestId('test-numeric-no-symbol')).toHaveTextContent('400.50')
|
|
175
|
+
expect(screen.getByTestId('test-numeric-medium-size')).toHaveTextContent('$500.55')
|
|
176
|
+
expect(screen.getByTestId('test-numeric-negative')).toHaveTextContent('-$600.70')
|
|
177
|
+
})
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
**NOTE:** The value passed into the `amount` prop can be either a string or numeric value.
|
|
@@ -1 +1 @@
|
|
|
1
|
-
Customize your icon circle by passing one of our seven base colors to the `variant` prop: `royal` `
|
|
1
|
+
Customize your icon circle by passing one of our seven base colors to the `variant` prop: `royal` `orange` `purple` `teal` `red` `yellow` `green`
|
|
@@ -3,48 +3,41 @@
|
|
|
3
3
|
text: "Mercury",
|
|
4
4
|
unit: "AU",
|
|
5
5
|
value: 0.39,
|
|
6
|
-
variant:"
|
|
6
|
+
variant:"royal"
|
|
7
7
|
}) %>
|
|
8
8
|
<br>
|
|
9
9
|
<%= pb_rails("icon_stat_value", props: { icon: "planet-ringed",
|
|
10
10
|
text: "Venus",
|
|
11
11
|
unit: "AU",
|
|
12
12
|
value: 0.723,
|
|
13
|
-
variant:"
|
|
13
|
+
variant:"purple"
|
|
14
14
|
}) %>
|
|
15
15
|
<br>
|
|
16
16
|
<%= pb_rails("icon_stat_value", props: { icon: "planet-moon",
|
|
17
17
|
text: "Earth",
|
|
18
18
|
unit: "AU",
|
|
19
19
|
value: 1.0,
|
|
20
|
-
variant:"
|
|
20
|
+
variant:"teal"
|
|
21
21
|
}) %>
|
|
22
22
|
<br>
|
|
23
23
|
<%= pb_rails("icon_stat_value", props: { icon: "solar-system",
|
|
24
24
|
text: "Mars",
|
|
25
25
|
unit: "AU",
|
|
26
26
|
value: 1.524,
|
|
27
|
-
variant:"
|
|
27
|
+
variant:"red"
|
|
28
28
|
}) %>
|
|
29
29
|
<br>
|
|
30
30
|
<%= pb_rails("icon_stat_value", props: { icon: "globe-americas",
|
|
31
|
-
text: "
|
|
31
|
+
text: "Jupiter",
|
|
32
32
|
unit: "AU",
|
|
33
33
|
value: 5.203,
|
|
34
|
-
variant:"
|
|
34
|
+
variant:"yellow"
|
|
35
35
|
}) %>
|
|
36
36
|
<br>
|
|
37
37
|
<%= pb_rails("icon_stat_value", props: { icon: "globe-africa",
|
|
38
38
|
text: "Saturn",
|
|
39
39
|
unit: "AU",
|
|
40
40
|
value: 9.539,
|
|
41
|
-
variant:"yellow"
|
|
42
|
-
}) %>
|
|
43
|
-
<br>
|
|
44
|
-
<%= pb_rails("icon_stat_value", props: { icon: "globe",
|
|
45
|
-
text: "Uranus",
|
|
46
|
-
unit: "AU",
|
|
47
|
-
value: 19.18,
|
|
48
41
|
variant:"green"
|
|
49
42
|
}) %>
|
|
50
43
|
<br>
|
|
@@ -53,4 +46,4 @@
|
|
|
53
46
|
unit: "AU",
|
|
54
47
|
value: 19.18,
|
|
55
48
|
variant:"orange"
|
|
56
|
-
}) %>
|
|
49
|
+
}) %>
|
|
@@ -9,7 +9,7 @@ const IconStatValueColor = (props) => {
|
|
|
9
9
|
text="Mercury"
|
|
10
10
|
unit="AU"
|
|
11
11
|
value={0.39}
|
|
12
|
-
variant="
|
|
12
|
+
variant="royal"
|
|
13
13
|
{...props}
|
|
14
14
|
/>
|
|
15
15
|
<br />
|
|
@@ -18,7 +18,7 @@ const IconStatValueColor = (props) => {
|
|
|
18
18
|
text="Venus"
|
|
19
19
|
unit="AU"
|
|
20
20
|
value={0.723}
|
|
21
|
-
variant="
|
|
21
|
+
variant="purple"
|
|
22
22
|
{...props}
|
|
23
23
|
/>
|
|
24
24
|
<br />
|
|
@@ -27,7 +27,7 @@ const IconStatValueColor = (props) => {
|
|
|
27
27
|
text="Earth"
|
|
28
28
|
unit="AU"
|
|
29
29
|
value={1.0}
|
|
30
|
-
variant="
|
|
30
|
+
variant="teal"
|
|
31
31
|
{...props}
|
|
32
32
|
/>
|
|
33
33
|
<br />
|
|
@@ -36,16 +36,16 @@ const IconStatValueColor = (props) => {
|
|
|
36
36
|
text="Mars"
|
|
37
37
|
unit="AU"
|
|
38
38
|
value={1.524}
|
|
39
|
-
variant="
|
|
39
|
+
variant="red"
|
|
40
40
|
{...props}
|
|
41
41
|
/>
|
|
42
42
|
<br />
|
|
43
43
|
<IconStatValue
|
|
44
44
|
icon="globe-americas"
|
|
45
|
-
text="
|
|
45
|
+
text="Jupiter"
|
|
46
46
|
unit="AU"
|
|
47
47
|
value={5.203}
|
|
48
|
-
variant="
|
|
48
|
+
variant="yellow"
|
|
49
49
|
{...props}
|
|
50
50
|
/>
|
|
51
51
|
<br />
|
|
@@ -54,15 +54,6 @@ const IconStatValueColor = (props) => {
|
|
|
54
54
|
text="Saturn"
|
|
55
55
|
unit="AU"
|
|
56
56
|
value={9.539}
|
|
57
|
-
variant="yellow"
|
|
58
|
-
{...props}
|
|
59
|
-
/>
|
|
60
|
-
<br />
|
|
61
|
-
<IconStatValue
|
|
62
|
-
icon="globe"
|
|
63
|
-
text="Uranus"
|
|
64
|
-
unit="AU"
|
|
65
|
-
value={19.18}
|
|
66
57
|
variant="green"
|
|
67
58
|
{...props}
|
|
68
59
|
/>
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
/* eslint-disable react/no-multi-comp */
|
|
2
|
-
import React, { useEffect, useState } from "react";
|
|
2
|
+
import React, { useEffect, useRef, useState } from "react";
|
|
3
3
|
import ReactDOM from "react-dom";
|
|
4
4
|
import {
|
|
5
5
|
Popper,
|
|
@@ -24,6 +24,7 @@ import { uniqueId } from '../utilities/object';
|
|
|
24
24
|
type ModifiedGlobalProps = Omit<GlobalProps, 'minWidth' | 'maxHeight' | 'minHeight'>
|
|
25
25
|
|
|
26
26
|
type PbPopoverProps = {
|
|
27
|
+
appendTo?: string;
|
|
27
28
|
aria?: { [key: string]: string };
|
|
28
29
|
className?: string;
|
|
29
30
|
closeOnClick?: "outside" | "inside" | "any";
|
|
@@ -62,6 +63,25 @@ const popoverModifiers = ({
|
|
|
62
63
|
return offset ? modifiers.concat([POPOVER_MODIFIERS.offset]) : modifiers;
|
|
63
64
|
};
|
|
64
65
|
|
|
66
|
+
const getAppendTarget = (
|
|
67
|
+
appendTo: string | undefined,
|
|
68
|
+
targetId: string
|
|
69
|
+
): HTMLElement => {
|
|
70
|
+
if (!appendTo || appendTo === "body") return document.body;
|
|
71
|
+
|
|
72
|
+
if (appendTo === "parent") {
|
|
73
|
+
const referenceWrapper = document.querySelector(`#reference-${targetId}`);
|
|
74
|
+
if (referenceWrapper?.parentElement) {
|
|
75
|
+
return referenceWrapper.parentElement;
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
const selectorMatch = document.querySelector(appendTo);
|
|
80
|
+
if (selectorMatch instanceof HTMLElement) return selectorMatch;
|
|
81
|
+
|
|
82
|
+
return document.body;
|
|
83
|
+
};
|
|
84
|
+
|
|
65
85
|
const Popover = (props: PbPopoverProps) => {
|
|
66
86
|
const {
|
|
67
87
|
aria = {},
|
|
@@ -89,7 +109,7 @@ const Popover = (props: PbPopoverProps) => {
|
|
|
89
109
|
const popoverSpacing =
|
|
90
110
|
filteredGlobalProps.includes("dark") || !filteredGlobalProps
|
|
91
111
|
? "p_sm"
|
|
92
|
-
: filteredGlobalProps
|
|
112
|
+
: filteredGlobalProps
|
|
93
113
|
const overflowHandling = maxHeight || maxWidth ? "overflow_handling" : "";
|
|
94
114
|
const zIndexStyle = zIndex ? { zIndex: zIndex } : {};
|
|
95
115
|
const widthHeightStyles = () => {
|
|
@@ -113,7 +133,7 @@ const Popover = (props: PbPopoverProps) => {
|
|
|
113
133
|
|
|
114
134
|
return (
|
|
115
135
|
<Popper
|
|
116
|
-
modifiers={popoverModifiers({ modifiers, offset })}
|
|
136
|
+
modifiers={popoverModifiers({ modifiers, offset: offset || false })}
|
|
117
137
|
placement={placement}
|
|
118
138
|
referenceElement={referenceElement}
|
|
119
139
|
>
|
|
@@ -154,6 +174,7 @@ const Popover = (props: PbPopoverProps) => {
|
|
|
154
174
|
const PbReactPopover = (props: PbPopoverProps): React.ReactElement => {
|
|
155
175
|
const [targetId] = useState(uniqueId('id-'))
|
|
156
176
|
const {
|
|
177
|
+
appendTo,
|
|
157
178
|
className,
|
|
158
179
|
children,
|
|
159
180
|
modifiers = [],
|
|
@@ -170,42 +191,56 @@ const PbReactPopover = (props: PbPopoverProps): React.ReactElement => {
|
|
|
170
191
|
minHeight,
|
|
171
192
|
minWidth,
|
|
172
193
|
width,
|
|
194
|
+
closeOnClick,
|
|
195
|
+
shouldClosePopover = noop,
|
|
173
196
|
} = props;
|
|
174
197
|
|
|
198
|
+
// Store latest callback in a ref to avoid re-runs
|
|
199
|
+
const shouldClosePopoverRef = useRef(shouldClosePopover);
|
|
200
|
+
|
|
201
|
+
// Update ref on change
|
|
175
202
|
useEffect(() => {
|
|
176
|
-
|
|
203
|
+
shouldClosePopoverRef.current = shouldClosePopover;
|
|
204
|
+
}, [shouldClosePopover]);
|
|
177
205
|
|
|
206
|
+
useEffect(() => {
|
|
178
207
|
if (!closeOnClick) return;
|
|
179
208
|
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
209
|
+
// Function to handle popover event listener and targetId.
|
|
210
|
+
// Ensure that whenever the component is conditionally rendered
|
|
211
|
+
// that the old listener is removed and the new listener is
|
|
212
|
+
// updated with the targetId.
|
|
213
|
+
const handleClick = (e: MouseEvent) => {
|
|
214
|
+
const target = e.target as HTMLElement
|
|
184
215
|
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
216
|
+
const targetIsPopover =
|
|
217
|
+
target.closest("#" + targetId) !== null;
|
|
218
|
+
const targetIsReference =
|
|
219
|
+
target.closest("#reference-" + targetId) !== null;
|
|
189
220
|
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
221
|
+
const shouldClose = () => {
|
|
222
|
+
setTimeout(() => shouldClosePopoverRef.current(true), 0);
|
|
223
|
+
}
|
|
193
224
|
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
);
|
|
208
|
-
|
|
225
|
+
switch (closeOnClick) {
|
|
226
|
+
case "outside":
|
|
227
|
+
if (!targetIsPopover && !targetIsReference) shouldClose();
|
|
228
|
+
break;
|
|
229
|
+
case "inside":
|
|
230
|
+
if (targetIsPopover) shouldClose();
|
|
231
|
+
break;
|
|
232
|
+
case "any":
|
|
233
|
+
if (targetIsPopover || !targetIsPopover && !targetIsReference) shouldClose();
|
|
234
|
+
break;
|
|
235
|
+
}
|
|
236
|
+
};
|
|
237
|
+
|
|
238
|
+
document.body.addEventListener("click", handleClick, { capture: true });
|
|
239
|
+
|
|
240
|
+
return () => {
|
|
241
|
+
document.body.removeEventListener("click", handleClick, { capture: true });
|
|
242
|
+
};
|
|
243
|
+
}, [targetId, closeOnClick]);
|
|
209
244
|
|
|
210
245
|
const popoverComponent = (
|
|
211
246
|
<Popover
|
|
@@ -246,10 +281,10 @@ const PbReactPopover = (props: PbPopoverProps): React.ReactElement => {
|
|
|
246
281
|
{show &&
|
|
247
282
|
(usePortal ? (
|
|
248
283
|
<>
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
284
|
+
{ReactDOM.createPortal(
|
|
285
|
+
popoverComponent,
|
|
286
|
+
getAppendTarget(appendTo, targetId)
|
|
287
|
+
)}
|
|
253
288
|
</>
|
|
254
289
|
) : (
|
|
255
290
|
{ popoverComponent }
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import React, { useState } from "react";
|
|
2
|
+
import { PbReactPopover, CircleIconButton, Body, Flex } from "playbook-ui";
|
|
3
|
+
|
|
4
|
+
const PopoverAppendTo = (props) => {
|
|
5
|
+
const [showParentPopover, setShowParentPopover] = useState(false);
|
|
6
|
+
const [showSelectorPopover, setShowSelectorPopover] = useState(false);
|
|
7
|
+
|
|
8
|
+
const parentPopoverReference = (
|
|
9
|
+
<CircleIconButton
|
|
10
|
+
icon="info"
|
|
11
|
+
onClick={() => setShowParentPopover(!showParentPopover)}
|
|
12
|
+
variant="secondary"
|
|
13
|
+
/>
|
|
14
|
+
);
|
|
15
|
+
|
|
16
|
+
const selectorPopoverReference = (
|
|
17
|
+
<CircleIconButton
|
|
18
|
+
icon="info"
|
|
19
|
+
onClick={() => setShowSelectorPopover(!showSelectorPopover)}
|
|
20
|
+
variant="secondary"
|
|
21
|
+
/>
|
|
22
|
+
);
|
|
23
|
+
|
|
24
|
+
return (
|
|
25
|
+
<>
|
|
26
|
+
<Flex
|
|
27
|
+
marginBottom="md"
|
|
28
|
+
orientation="row"
|
|
29
|
+
vertical="center"
|
|
30
|
+
{...props}
|
|
31
|
+
>
|
|
32
|
+
<Body text="Click info for more details" />
|
|
33
|
+
|
|
34
|
+
<PbReactPopover
|
|
35
|
+
appendTo="parent"
|
|
36
|
+
offset
|
|
37
|
+
placement="top"
|
|
38
|
+
reference={parentPopoverReference}
|
|
39
|
+
show={showParentPopover}
|
|
40
|
+
{...props}
|
|
41
|
+
>
|
|
42
|
+
{'I\'m a popover. I have been appended to my parent element.'}
|
|
43
|
+
</PbReactPopover>
|
|
44
|
+
</Flex>
|
|
45
|
+
|
|
46
|
+
<Flex
|
|
47
|
+
orientation="row"
|
|
48
|
+
vertical="center"
|
|
49
|
+
{...props}
|
|
50
|
+
>
|
|
51
|
+
<Body text="Click info for more details" />
|
|
52
|
+
|
|
53
|
+
<PbReactPopover
|
|
54
|
+
appendTo=".kit-show-wrapper"
|
|
55
|
+
offset
|
|
56
|
+
placement="top"
|
|
57
|
+
reference={selectorPopoverReference}
|
|
58
|
+
show={showSelectorPopover}
|
|
59
|
+
{...props}
|
|
60
|
+
>
|
|
61
|
+
{'I\'m a popover. I have been appended to the .kit-show-wrapper.'}
|
|
62
|
+
</PbReactPopover>
|
|
63
|
+
</Flex>
|
|
64
|
+
</>
|
|
65
|
+
);
|
|
66
|
+
};
|
|
67
|
+
|
|
68
|
+
export default PopoverAppendTo;
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
By default, the popover tooltip attaches to the `<body>`. To attach it elsewhere, use the `appendTo` prop. Set it to `"parent"` to place the tooltip inside its parent element, or pass any CSS selector (`#id` or `.class`) to specify a custom container.
|
|
@@ -4,3 +4,4 @@ export { default as PopoverClose } from './_popover_close.jsx'
|
|
|
4
4
|
export { default as PopoverZIndex } from './_popover_z_index.jsx'
|
|
5
5
|
export { default as PopoverScrollHeight } from './_popover_scroll_height.jsx'
|
|
6
6
|
export { default as PopoverActionableContent } from './_popover_actionable_content.jsx'
|
|
7
|
+
export { default as PopoverAppendTo } from './_popover_append_to.jsx'
|