playbook_ui 15.3.0.pre.alpha.PLAY260211779 → 15.3.0.pre.alpha.play2464flexitemcantusehtmloptions12226
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/Components/RegularTableView.tsx +2 -1
- data/app/pb_kits/playbook/pb_advanced_table/Hooks/useTableState.ts +4 -4
- data/app/pb_kits/playbook/pb_advanced_table/Utilities/RowUtils.ts +1 -1
- data/app/pb_kits/playbook/pb_advanced_table/_advanced_table.scss +4 -4
- data/app/pb_kits/playbook/pb_advanced_table/advanced_table.test.jsx +68 -5
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_background_control_rails.html.erb +4 -0
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_background_control_rails.md +1 -1
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_column_border_color.md +1 -1
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_column_border_color_rails.md +1 -1
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_column_styling.jsx +3 -1
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_column_styling.md +2 -0
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_column_styling_column_headers.jsx +1 -1
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_column_styling_rails.html.erb +1 -0
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_column_styling_rails.md +2 -0
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_padding_control.jsx +9 -1
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_padding_control.md +1 -1
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_sticky_columns_and_header.md +1 -1
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_table_props_sticky_header_rails.md +1 -1
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_table_props_sticky_header_react.md +1 -1
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_with_custom_header_multi_header.jsx +16 -0
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_with_custom_header_multi_header_rails.html.erb +104 -0
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_with_custom_header_multi_header_rails.md +1 -0
- data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_with_custom_header_rails.html.erb +1 -1
- data/app/pb_kits/playbook/pb_advanced_table/docs/example.yml +1 -0
- data/app/pb_kits/playbook/pb_advanced_table/flat_advanced_table.js +2 -2
- data/app/pb_kits/playbook/pb_advanced_table/index.js +7 -7
- data/app/pb_kits/playbook/pb_advanced_table/scss_partials/advanced_table_sticky_mixin.scss +2 -2
- data/app/pb_kits/playbook/pb_advanced_table/table_header.rb +90 -20
- data/app/pb_kits/playbook/pb_advanced_table/table_row.rb +32 -3
- data/app/pb_kits/playbook/pb_background/background.html.erb +10 -2
- data/app/pb_kits/playbook/pb_background/docs/_background_category.md +1 -1
- data/app/pb_kits/playbook/pb_badge/_badge.tsx +4 -1
- data/app/pb_kits/playbook/pb_badge/badge.test.js +13 -0
- data/app/pb_kits/playbook/pb_card/docs/_card_background.md +1 -1
- data/app/pb_kits/playbook/pb_card/docs/_card_header.md +1 -1
- data/app/pb_kits/playbook/pb_card/docs/_card_highlight.md +1 -1
- data/app/pb_kits/playbook/pb_card/docs/_card_light.md +1 -1
- data/app/pb_kits/playbook/pb_currency/_currency.tsx +20 -7
- data/app/pb_kits/playbook/pb_currency/currency.rb +35 -8
- data/app/pb_kits/playbook/pb_currency/currency.test.js +47 -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_date_picker/date_picker_helper.ts +16 -4
- data/app/pb_kits/playbook/pb_date_picker/docs/_date_picker_and_dropdown_range.jsx +38 -0
- data/app/pb_kits/playbook/pb_date_picker/docs/_date_picker_and_dropdown_range.md +14 -0
- data/app/pb_kits/playbook/pb_date_picker/docs/example.yml +2 -1
- data/app/pb_kits/playbook/pb_date_picker/docs/index.js +2 -1
- data/app/pb_kits/playbook/pb_distribution_bar/docs/_distribution_bar_custom_colors.md +1 -1
- data/app/pb_kits/playbook/pb_dropdown/_dropdown.scss +1 -0
- data/app/pb_kits/playbook/pb_dropdown/_dropdown.tsx +111 -6
- data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_quickpick.jsx +18 -0
- data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_quickpick.md +4 -0
- data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_quickpick_default_dates.jsx +18 -0
- data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_quickpick_default_dates.md +1 -0
- data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_quickpick_range_end.jsx +19 -0
- data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_quickpick_range_end.md +1 -0
- data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_quickpick_with_date_pickers.jsx +38 -0
- data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_quickpick_with_date_pickers.md +14 -0
- data/app/pb_kits/playbook/pb_dropdown/docs/_dropdown_with_custom_active_style_options_react.md +1 -1
- data/app/pb_kits/playbook/pb_dropdown/docs/example.yml +5 -0
- data/app/pb_kits/playbook/pb_dropdown/docs/index.js +5 -1
- data/app/pb_kits/playbook/pb_dropdown/dropdown.test.jsx +148 -2
- data/app/pb_kits/playbook/pb_dropdown/quickpick/index.ts +60 -0
- data/app/pb_kits/playbook/pb_filter/docs/_filter_max_width.md +1 -1
- data/app/pb_kits/playbook/pb_fixed_confirmation_toast/docs/_fixed_confirmation_toast_auto_close.html.erb +15 -1
- data/app/pb_kits/playbook/pb_fixed_confirmation_toast/docs/_fixed_confirmation_toast_multi_line.html.erb +9 -8
- data/app/pb_kits/playbook/pb_fixed_confirmation_toast/docs/_fixed_confirmation_toast_positions.html.erb +11 -10
- data/app/pb_kits/playbook/pb_flex/_flex_item.tsx +12 -5
- data/app/pb_kits/playbook/pb_flex/docs/_flex_item_example.jsx +1 -0
- data/app/pb_kits/playbook/pb_form/pb_form_validation.js +44 -11
- data/app/pb_kits/playbook/pb_form_pill/_form_pill.tsx +1 -1
- data/app/pb_kits/playbook/pb_form_pill/docs/_form_pill_truncated_text_rails.md +1 -1
- data/app/pb_kits/playbook/pb_form_pill/docs/_form_pill_truncated_text_react.md +1 -1
- data/app/pb_kits/playbook/pb_icon/docs/_icon_color.md +1 -1
- 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_layout/docs/_layout_collection.md +1 -1
- data/app/pb_kits/playbook/pb_nav/docs/_collapsible_nav_item_spacing.md +1 -1
- data/app/pb_kits/playbook/pb_nav/docs/_nav_with_spacing_control.md +1 -1
- data/app/pb_kits/playbook/pb_overlay/docs/_overlay_layout.md +1 -1
- data/app/pb_kits/playbook/pb_phone_number_input/_phone_number_input.tsx +110 -17
- data/app/pb_kits/playbook/pb_pill/docs/_description.md +1 -1
- 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_rich_text_editor/docs/_description.md +1 -1
- data/app/pb_kits/playbook/pb_section_separator/docs/_description.md +1 -1
- data/app/pb_kits/playbook/pb_selectable_card/_selectable_card.scss +29 -0
- data/app/pb_kits/playbook/pb_skeleton_loading/docs/_skeleton_loading_border_radius_rails.md +1 -1
- data/app/pb_kits/playbook/pb_skeleton_loading/docs/_skeleton_loading_border_radius_react.md +1 -1
- data/app/pb_kits/playbook/pb_skeleton_loading/docs/_skeleton_loading_layout.md +1 -1
- data/app/pb_kits/playbook/pb_table/docs/_table_side_highlight.md +1 -1
- data/app/pb_kits/playbook/pb_table/docs/_table_sm.md +1 -1
- 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.scss +7 -0
- data/app/pb_kits/playbook/pb_typeahead/_typeahead.test.jsx +105 -1
- data/app/pb_kits/playbook/pb_typeahead/_typeahead.tsx +23 -9
- data/app/pb_kits/playbook/pb_typeahead/components/MultiValue.tsx +33 -1
- 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/_typeahead_truncated_text.md +1 -1
- 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-oRTpJgi9.js} +1 -1
- data/dist/chunks/_typeahead-CDf2C2se.js +6 -0
- data/dist/chunks/_weekday_stacked-BPxUGfCf.js +37 -0
- data/dist/chunks/{lib-CGxXTQ75.js → lib-BXBHAZMY.js} +1 -1
- data/dist/chunks/pb_form_validation-BNfSnIUF.js +1 -0
- 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/dist/playbook.css +1 -1
- data/lib/playbook/version.rb +1 -1
- metadata +31 -8
- data/dist/chunks/_typeahead-U8AjZIIW.js +0 -6
- data/dist/chunks/_weekday_stacked-DHXLQcpy.js +0 -37
- data/dist/chunks/pb_form_validation-DebqlUKZ.js +0 -1
- /data/app/pb_kits/playbook/pb_popover/docs/{_popover_append_to.md → _popover_append_to_rails.md} +0 -0
|
@@ -82,32 +82,62 @@ module Playbook
|
|
|
82
82
|
end
|
|
83
83
|
end
|
|
84
84
|
|
|
85
|
-
# Get original column definition for custom rendering
|
|
85
|
+
# Get original column definition for custom rendering by accessor
|
|
86
86
|
def find_original_column_def(accessor)
|
|
87
87
|
find_column_def_by_accessor(column_definitions, accessor)
|
|
88
88
|
end
|
|
89
89
|
|
|
90
|
+
# Get original column definition for custom rendering by id
|
|
91
|
+
def find_original_column_def_by_id(id)
|
|
92
|
+
find_column_def_by_id(column_definitions, id)
|
|
93
|
+
end
|
|
94
|
+
|
|
90
95
|
# Check if a header cell has a custom renderer
|
|
91
96
|
def has_header_renderer?(cell)
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
97
|
+
original_def = find_original_column_def_for_cell(cell)
|
|
98
|
+
original_def&.dig(:header).present?
|
|
99
|
+
rescue
|
|
100
|
+
false
|
|
96
101
|
end
|
|
97
102
|
|
|
98
103
|
# Render custom header content
|
|
99
104
|
def render_header(cell)
|
|
100
105
|
return cell[:label] unless has_header_renderer?(cell)
|
|
101
106
|
|
|
102
|
-
original_def =
|
|
107
|
+
original_def = find_original_column_def_for_cell(cell)
|
|
108
|
+
return cell[:label] unless original_def
|
|
109
|
+
|
|
103
110
|
custom_renderer = original_def[:header]
|
|
111
|
+
return cell[:label] unless custom_renderer
|
|
104
112
|
|
|
105
113
|
# Call the custom renderer with the cell data and label
|
|
106
|
-
|
|
114
|
+
begin
|
|
115
|
+
result = custom_renderer.call(cell, cell[:label])
|
|
116
|
+
result.present? ? result.to_s : cell[:label]
|
|
117
|
+
rescue
|
|
118
|
+
cell[:label]
|
|
119
|
+
end
|
|
107
120
|
end
|
|
108
121
|
|
|
109
122
|
private
|
|
110
123
|
|
|
124
|
+
# Find the original column definition for a cell
|
|
125
|
+
def find_original_column_def_for_cell(cell)
|
|
126
|
+
# Try accessor first (for leaf columns)
|
|
127
|
+
if cell[:accessor].present?
|
|
128
|
+
found = find_original_column_def(cell[:accessor])
|
|
129
|
+
return found if found
|
|
130
|
+
end
|
|
131
|
+
|
|
132
|
+
# Try id if accessor lookup didn't find it (for grouped columns or leaf columns with id)
|
|
133
|
+
if cell[:id].present?
|
|
134
|
+
found = find_original_column_def_by_id(cell[:id])
|
|
135
|
+
return found if found
|
|
136
|
+
end
|
|
137
|
+
|
|
138
|
+
nil
|
|
139
|
+
end
|
|
140
|
+
|
|
111
141
|
def compute_max_depth(columns)
|
|
112
142
|
columns.map do |col|
|
|
113
143
|
col[:columns] ? 1 + compute_max_depth(col[:columns]) : 1
|
|
@@ -118,21 +148,24 @@ module Playbook
|
|
|
118
148
|
total_columns = columns.size
|
|
119
149
|
columns.each_with_index do |col, index|
|
|
120
150
|
is_last = index == total_columns - 1
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
151
|
+
nested_columns = col[:columns]
|
|
152
|
+
if nested_columns
|
|
153
|
+
colspan = compute_leaf_columns(nested_columns)
|
|
154
|
+
cell_hash = {
|
|
124
155
|
label: col[:label],
|
|
125
156
|
colspan: colspan,
|
|
126
157
|
is_last_in_group: is_last && current_depth.positive?,
|
|
127
158
|
}
|
|
159
|
+
cell_hash[:id] = col[:id] if col[:id].present?
|
|
160
|
+
rows[current_depth] << cell_hash
|
|
128
161
|
|
|
129
|
-
process_columns(
|
|
162
|
+
process_columns(nested_columns, rows, current_depth + 1, max_depth)
|
|
130
163
|
else
|
|
131
|
-
raw_styling
|
|
132
|
-
header_alignment
|
|
164
|
+
raw_styling = col[:column_styling] || {}
|
|
165
|
+
header_alignment = raw_styling[:header_alignment]
|
|
133
166
|
|
|
134
167
|
colspan = 1
|
|
135
|
-
|
|
168
|
+
cell_hash = {
|
|
136
169
|
label: col[:label],
|
|
137
170
|
colspan: colspan,
|
|
138
171
|
accessor: col[:accessor],
|
|
@@ -140,6 +173,8 @@ module Playbook
|
|
|
140
173
|
is_last_in_group: is_last && current_depth.positive?,
|
|
141
174
|
header_alignment: header_alignment,
|
|
142
175
|
}
|
|
176
|
+
cell_hash[:id] = col[:id] if col[:id].present?
|
|
177
|
+
rows[current_depth] << cell_hash
|
|
143
178
|
end
|
|
144
179
|
end
|
|
145
180
|
end
|
|
@@ -154,11 +189,15 @@ module Playbook
|
|
|
154
189
|
max_depth = compute_max_depth(column_definitions)
|
|
155
190
|
|
|
156
191
|
column_definitions.map do |col|
|
|
157
|
-
if col
|
|
158
|
-
|
|
192
|
+
if col[:columns]
|
|
193
|
+
nested_columns = col[:columns]
|
|
194
|
+
wrapped = {
|
|
159
195
|
label: col[:label],
|
|
160
|
-
columns: wrap_leaf_columns(
|
|
196
|
+
columns: wrap_leaf_columns(nested_columns),
|
|
161
197
|
}
|
|
198
|
+
wrapped[:id] = col[:id] if col[:id].present?
|
|
199
|
+
wrapped[:header] = col[:header] if col[:header].present?
|
|
200
|
+
wrapped
|
|
162
201
|
else
|
|
163
202
|
# For leaf columns, wrap with empty labels up to max depth to get proper structure
|
|
164
203
|
wrap_leaf_column(col, max_depth)
|
|
@@ -173,6 +212,7 @@ module Playbook
|
|
|
173
212
|
sort_menu: col[:sort_menu] || nil,
|
|
174
213
|
column_styling: col[:column_styling] || {},
|
|
175
214
|
}
|
|
215
|
+
wrapped[:id] = col[:id] if col[:id].present?
|
|
176
216
|
(max_depth - 1).times do
|
|
177
217
|
wrapped = { label: "", columns: [wrapped] }
|
|
178
218
|
end
|
|
@@ -180,11 +220,41 @@ module Playbook
|
|
|
180
220
|
end
|
|
181
221
|
|
|
182
222
|
def find_column_def_by_accessor(defs, target_accessor)
|
|
223
|
+
return nil if target_accessor.blank?
|
|
224
|
+
|
|
183
225
|
defs.each do |col|
|
|
184
|
-
|
|
226
|
+
col_accessor = col[:accessor]
|
|
227
|
+
next if col_accessor.blank?
|
|
228
|
+
|
|
229
|
+
return col if col_accessor.to_s == target_accessor.to_s
|
|
230
|
+
|
|
231
|
+
nested_columns = col[:columns]
|
|
232
|
+
if nested_columns.is_a?(Array)
|
|
233
|
+
found = find_column_def_by_accessor(nested_columns, target_accessor)
|
|
234
|
+
return found if found
|
|
235
|
+
end
|
|
236
|
+
end
|
|
237
|
+
nil
|
|
238
|
+
end
|
|
239
|
+
|
|
240
|
+
def find_column_def_by_id(defs, target_id)
|
|
241
|
+
return nil if target_id.blank?
|
|
242
|
+
|
|
243
|
+
defs.each do |col|
|
|
244
|
+
col_id = col[:id]
|
|
245
|
+
|
|
246
|
+
return col if col_id.present? && col_id.to_s == target_id.to_s
|
|
247
|
+
|
|
248
|
+
# Recursively search nested columns, even if current col has no id or doesn't match
|
|
249
|
+
nested_columns = col[:columns]
|
|
250
|
+
|
|
251
|
+
next unless nested_columns.present?
|
|
252
|
+
|
|
253
|
+
# Convert to array if needed (for edge cases where is_a?(Array) might fail)
|
|
254
|
+
array_columns = Array(nested_columns)
|
|
185
255
|
|
|
186
|
-
if
|
|
187
|
-
found =
|
|
256
|
+
if array_columns.any?
|
|
257
|
+
found = find_column_def_by_id(array_columns, target_id)
|
|
188
258
|
return found if found
|
|
189
259
|
end
|
|
190
260
|
end
|
|
@@ -39,7 +39,7 @@ module Playbook
|
|
|
39
39
|
end
|
|
40
40
|
|
|
41
41
|
def classname
|
|
42
|
-
generate_classname("pb_table_tr", "bg-white", subrow_depth_classname, separator: " ")
|
|
42
|
+
generate_classname("pb_table_tr", "pb-bg-row-white", subrow_depth_classname, separator: " ")
|
|
43
43
|
end
|
|
44
44
|
|
|
45
45
|
def td_classname(column, index)
|
|
@@ -79,15 +79,44 @@ module Playbook
|
|
|
79
79
|
cell_background_color(column).present?
|
|
80
80
|
end
|
|
81
81
|
|
|
82
|
+
def cell_font_color(column)
|
|
83
|
+
return nil unless column[:accessor].present?
|
|
84
|
+
|
|
85
|
+
orig_def = find_column_def_by_accessor(column_definitions, column[:accessor])
|
|
86
|
+
if orig_def && orig_def[:column_styling].is_a?(Hash) && orig_def[:column_styling][:font_color].present?
|
|
87
|
+
font_color = orig_def[:column_styling][:font_color]
|
|
88
|
+
if font_color.respond_to?(:call)
|
|
89
|
+
font_color.call(row)
|
|
90
|
+
else
|
|
91
|
+
font_color
|
|
92
|
+
end
|
|
93
|
+
end
|
|
94
|
+
end
|
|
95
|
+
|
|
82
96
|
# Uses a regular table/table_cell component if there is no custom background color; if there is a cell_background_color uses a background component with tag "td"
|
|
83
97
|
def cell_component_info(column, index, bg_color, font_color)
|
|
98
|
+
column_font_color = cell_font_color(column)
|
|
99
|
+
effective_font_color = column_font_color || font_color
|
|
100
|
+
|
|
84
101
|
if has_custom_background_color?(column)
|
|
85
102
|
custom_bg_color = cell_background_color(column)
|
|
86
103
|
component_name = "background"
|
|
87
|
-
component_props = {
|
|
104
|
+
component_props = {
|
|
105
|
+
background_color: custom_bg_color,
|
|
106
|
+
tag: "td",
|
|
107
|
+
classname: td_classname(column, index),
|
|
108
|
+
}
|
|
109
|
+
component_props[:html_options] = { style: { color: effective_font_color } } if effective_font_color.present?
|
|
88
110
|
else
|
|
89
111
|
component_name = "table/table_cell"
|
|
90
|
-
|
|
112
|
+
style_hash = { "background-color": bg_color }
|
|
113
|
+
style_hash[:color] = effective_font_color if effective_font_color.present?
|
|
114
|
+
component_props = {
|
|
115
|
+
html_options: {
|
|
116
|
+
style: style_hash,
|
|
117
|
+
},
|
|
118
|
+
classname: td_classname(column, index),
|
|
119
|
+
}
|
|
91
120
|
end
|
|
92
121
|
|
|
93
122
|
{ name: component_name, props: component_props }
|
|
@@ -1,3 +1,11 @@
|
|
|
1
|
+
<%
|
|
2
|
+
html_options_style = ""
|
|
3
|
+
if object.html_options[:style].is_a?(Hash)
|
|
4
|
+
html_options_style = object.html_options[:style].map { |k, v| "#{k.to_s.tr('_', '-')}: #{v}" }.join("; ")
|
|
5
|
+
elsif object.html_options[:style].is_a?(String)
|
|
6
|
+
html_options_style = object.html_options[:style]
|
|
7
|
+
end
|
|
8
|
+
%>
|
|
1
9
|
<% if object.image_url.present? %>
|
|
2
10
|
<%= pb_content_tag(object.tag,
|
|
3
11
|
style: "background-image: url('#{object.image_url}');
|
|
@@ -9,8 +17,8 @@
|
|
|
9
17
|
<% end %>
|
|
10
18
|
<% else %>
|
|
11
19
|
<%= pb_content_tag(object.tag,
|
|
12
|
-
style: object.custom_background_color
|
|
20
|
+
style: "#{object.custom_background_color}#{html_options_style.present? ? "; #{html_options_style}" : ""}"
|
|
13
21
|
) do %>
|
|
14
22
|
<%= content.presence %>
|
|
15
23
|
<% end %>
|
|
16
|
-
<% end %>
|
|
24
|
+
<% end %>
|
|
@@ -1 +1 @@
|
|
|
1
|
-
Category colors can be passed into the background kit. Values `category_1` to `category_21` are accepted. List of all category and status colors can be viewed <a href="https://playbook.powerapp.cloud/
|
|
1
|
+
Category colors can be passed into the background kit. Values `category_1` to `category_21` are accepted. List of all category and status colors can be viewed <a href="https://playbook.powerapp.cloud/tokens/colors" target="_blank">here</a>.
|
|
@@ -20,11 +20,12 @@ type BadgeProps = {
|
|
|
20
20
|
onTouchEnd?: React.TouchEventHandler<HTMLSpanElement>,
|
|
21
21
|
},
|
|
22
22
|
data?: {[key: string]: string},
|
|
23
|
-
htmlOptions?: {[key: string]: string | number | boolean | (() => void)},
|
|
23
|
+
htmlOptions?: {[key: string]: string | number | boolean | (() => void) | ((event: any) => void) | any},
|
|
24
24
|
id?: string,
|
|
25
25
|
removeIcon?: boolean,
|
|
26
26
|
removeOnClick?: React.MouseEventHandler<HTMLSpanElement>,
|
|
27
27
|
rounded?: boolean,
|
|
28
|
+
tabIndex?: number,
|
|
28
29
|
text?: string,
|
|
29
30
|
variant?: "error" | "info" | "neutral" | "notification" | "notificationError" | "primary" | "success" | "warning",
|
|
30
31
|
} & GlobalProps
|
|
@@ -39,6 +40,7 @@ const Badge = (props: BadgeProps): React.ReactElement => {
|
|
|
39
40
|
removeIcon = false,
|
|
40
41
|
removeOnClick,
|
|
41
42
|
rounded = false,
|
|
43
|
+
tabIndex,
|
|
42
44
|
text,
|
|
43
45
|
variant = 'neutral',
|
|
44
46
|
} = props
|
|
@@ -61,6 +63,7 @@ const Badge = (props: BadgeProps): React.ReactElement => {
|
|
|
61
63
|
{...htmlProps}
|
|
62
64
|
className={css}
|
|
63
65
|
id={id}
|
|
66
|
+
tabIndex={tabIndex}
|
|
64
67
|
>
|
|
65
68
|
<span>
|
|
66
69
|
{text}
|
|
@@ -112,3 +112,16 @@ test('displays notification variants', () => {
|
|
|
112
112
|
cleanup()
|
|
113
113
|
})
|
|
114
114
|
})
|
|
115
|
+
|
|
116
|
+
test('should allow tabIndex to be set', () => {
|
|
117
|
+
render(
|
|
118
|
+
<Badge
|
|
119
|
+
data={{ testid: testId }}
|
|
120
|
+
tabIndex={0}
|
|
121
|
+
text="+1"
|
|
122
|
+
/>
|
|
123
|
+
)
|
|
124
|
+
|
|
125
|
+
const kit = screen.getByTestId(testId)
|
|
126
|
+
expect(kit).toHaveAttribute('tabIndex', '0')
|
|
127
|
+
})
|
|
@@ -1 +1 @@
|
|
|
1
|
-
Add a background color by passing the color name to background. List of all colors can be viewed <a href="https://playbook.powerapp.cloud/
|
|
1
|
+
Add a background color by passing the color name to background. List of all colors can be viewed <a href="https://playbook.powerapp.cloud/tokens/colors" target="_blank">here</a> under Product Colors and Status: Subtle Variations.
|
|
@@ -1 +1 @@
|
|
|
1
|
-
Card headers pass category, product, status and background colors only. List of all category, product, status and background colors can be viewed <a href="https://playbook.powerapp.cloud/
|
|
1
|
+
Card headers pass category, product, status and background colors only. List of all category, product, status and background colors can be viewed <a href="https://playbook.powerapp.cloud/token/colors" target="_blank">here</a>.
|
|
@@ -1 +1 @@
|
|
|
1
|
-
Card highlight can pass status, product, and category colors. List of all colors can be viewed <a href="https://playbook.powerapp.cloud/
|
|
1
|
+
Card highlight can pass status, product, and category colors. List of all colors can be viewed <a href="https://playbook.powerapp.cloud/token/colors" target="_blank">here</a>.
|
|
@@ -1 +1 @@
|
|
|
1
|
-
Card can leverage the max-width property. Learn more in our <a href="https://playbook.powerapp.cloud/
|
|
1
|
+
Card can leverage the max-width property. Learn more in our <a href="https://playbook.powerapp.cloud/global_props/max_width" target="_blank">visual guidelines.</a>
|
|
@@ -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,19 @@ 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
|
+
if (input === 0 && !nullDisplay) {
|
|
66
|
+
return ""
|
|
67
|
+
}
|
|
68
|
+
return input.toFixed(2)
|
|
69
|
+
}
|
|
70
|
+
return input
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
const currencyAmount = convertAmount(amount)
|
|
74
|
+
|
|
62
75
|
const emphasizedClass = emphasized ? '' : '_deemphasized'
|
|
63
76
|
|
|
64
77
|
let variantClass
|
|
@@ -68,7 +81,7 @@ const Currency = (props: CurrencyProps): React.ReactElement => {
|
|
|
68
81
|
variantClass = '_bold'
|
|
69
82
|
}
|
|
70
83
|
|
|
71
|
-
const [whole, decimal = '00'] =
|
|
84
|
+
const [whole, decimal = '00'] = currencyAmount.split('.')
|
|
72
85
|
const ariaProps = buildAriaProps(aria)
|
|
73
86
|
const dataProps = buildDataProps(data)
|
|
74
87
|
const htmlProps = buildHtmlProps(htmlOptions)
|
|
@@ -92,19 +105,19 @@ const Currency = (props: CurrencyProps): React.ReactElement => {
|
|
|
92
105
|
return isAmount ? num.slice(0, -1) : isUnit ? num.slice(-1) : ''
|
|
93
106
|
}
|
|
94
107
|
|
|
95
|
-
const getMatchingDecimalAmount = decimals === "matching" ?
|
|
108
|
+
const getMatchingDecimalAmount = decimals === "matching" ? currencyAmount : whole
|
|
96
109
|
const getMatchingDecimalValue = decimals === "matching" ? '' : `.${decimal}`
|
|
97
110
|
|
|
98
111
|
const formatAmount = (amount: string) => {
|
|
99
112
|
if (!commaSeparator) return amount;
|
|
100
|
-
|
|
113
|
+
|
|
101
114
|
const [wholePart, decimalPart] = amount.split('.');
|
|
102
115
|
const formattedWhole = new Intl.NumberFormat('en-US').format(parseInt(wholePart));
|
|
103
116
|
return decimalPart ? `${formattedWhole}.${decimalPart}` : formattedWhole;
|
|
104
117
|
}
|
|
105
118
|
|
|
106
119
|
const swapNegative = size === "sm" && symbol !== ""
|
|
107
|
-
const handleNegative =
|
|
120
|
+
const handleNegative = currencyAmount.startsWith("-") && swapNegative ? "-" : ""
|
|
108
121
|
const getAbsoluteAmount = (amountString: string) => amountString.replace(/^-/,'')
|
|
109
122
|
const getAbbrOrFormatAmount = abbreviate ? getAbbreviatedValue('amount') : formatAmount(getMatchingDecimalAmount)
|
|
110
123
|
const getAmount = swapNegative ? getAbsoluteAmount(getAbbrOrFormatAmount) : getAbbrOrFormatAmount
|
|
@@ -152,7 +165,7 @@ const Currency = (props: CurrencyProps): React.ReactElement => {
|
|
|
152
165
|
>
|
|
153
166
|
{handleNegative}{symbol}
|
|
154
167
|
</Body>
|
|
155
|
-
|
|
168
|
+
|
|
156
169
|
<Title
|
|
157
170
|
className="pb_currency_value"
|
|
158
171
|
dark={dark}
|
|
@@ -160,7 +173,7 @@ const Currency = (props: CurrencyProps): React.ReactElement => {
|
|
|
160
173
|
>
|
|
161
174
|
{getAmount}
|
|
162
175
|
</Title>
|
|
163
|
-
|
|
176
|
+
|
|
164
177
|
<Body
|
|
165
178
|
className="unit"
|
|
166
179
|
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,32 @@ 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
|
+
if input.zero? && null_display.nil?
|
|
129
|
+
""
|
|
130
|
+
else
|
|
131
|
+
format("%.2f", input)
|
|
132
|
+
end
|
|
133
|
+
# Handle string representations of zero
|
|
134
|
+
elsif input.to_s.strip.match?(/^-?0+(\.0+)?$/) && null_display.nil?
|
|
135
|
+
""
|
|
136
|
+
else
|
|
137
|
+
input.to_s
|
|
138
|
+
end
|
|
139
|
+
end
|
|
140
|
+
|
|
122
141
|
def whole_value
|
|
123
|
-
|
|
142
|
+
return "" if currency_amount.blank?
|
|
143
|
+
|
|
144
|
+
value = currency_amount.split(".").first
|
|
124
145
|
if comma_separator
|
|
125
146
|
number_with_delimiter(value.gsub(",", ""))
|
|
126
147
|
else
|
|
@@ -129,7 +150,9 @@ module Playbook
|
|
|
129
150
|
end
|
|
130
151
|
|
|
131
152
|
def decimal_value
|
|
132
|
-
|
|
153
|
+
return "00" if currency_amount.blank?
|
|
154
|
+
|
|
155
|
+
currency_amount.split(".")[1] || "00"
|
|
133
156
|
end
|
|
134
157
|
|
|
135
158
|
def units_element
|
|
@@ -147,7 +170,9 @@ module Playbook
|
|
|
147
170
|
end
|
|
148
171
|
|
|
149
172
|
def abbreviated_value(index = 0..-2)
|
|
150
|
-
|
|
173
|
+
return "" if currency_amount.blank?
|
|
174
|
+
|
|
175
|
+
value = currency_amount.split(".").first.gsub(",", "").to_i
|
|
151
176
|
abbreviated_num = number_to_human(value, units: { thousand: "K", million: "M", billion: "B", trillion: "T" }).gsub(/\s+/, "")
|
|
152
177
|
abbreviated_num[index]
|
|
153
178
|
end
|
|
@@ -174,9 +199,11 @@ module Playbook
|
|
|
174
199
|
|
|
175
200
|
if decimals == "matching"
|
|
176
201
|
if comma_separator
|
|
177
|
-
|
|
202
|
+
return "" if currency_amount.blank?
|
|
203
|
+
|
|
204
|
+
number_with_delimiter(currency_amount.gsub(",", ""))
|
|
178
205
|
else
|
|
179
|
-
|
|
206
|
+
currency_amount
|
|
180
207
|
end
|
|
181
208
|
else
|
|
182
209
|
whole_value
|
|
@@ -133,3 +133,50 @@ 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
|
+
<Currency
|
|
169
|
+
amount={0.00}
|
|
170
|
+
data={{ testid: 'test-numeric-null' }}
|
|
171
|
+
/>
|
|
172
|
+
</>
|
|
173
|
+
)
|
|
174
|
+
|
|
175
|
+
expect(screen.getByTestId('test-numeric-default')).toHaveTextContent('$320')
|
|
176
|
+
expect(screen.getByTestId('test-numeric-millions')).toHaveTextContent('$3.2M')
|
|
177
|
+
expect(screen.getByTestId('test-numeric-comma-decimals')).toHaveTextContent('$123,456.78')
|
|
178
|
+
expect(screen.getByTestId('test-numeric-no-symbol')).toHaveTextContent('400.50')
|
|
179
|
+
expect(screen.getByTestId('test-numeric-medium-size')).toHaveTextContent('$500.55')
|
|
180
|
+
expect(screen.getByTestId('test-numeric-negative')).toHaveTextContent('-$600.70')
|
|
181
|
+
expect(screen.getByTestId('test-numeric-null')).toHaveTextContent('$.00')
|
|
182
|
+
})
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
**NOTE:** The value passed into the `amount` prop can be either a string or numeric value.
|
|
@@ -350,8 +350,14 @@ const datePickerHelper = (config: DatePickerConfig, scrollContainer: string | HT
|
|
|
350
350
|
if (syncStartWith) {
|
|
351
351
|
picker.config.onClose.push((selectedDates: string) => {
|
|
352
352
|
if (selectedDates?.length) {
|
|
353
|
-
const
|
|
354
|
-
|
|
353
|
+
const element = document.querySelector(`#${syncStartWith}`) as any;
|
|
354
|
+
// Check if it's a Dropdown QuickPick (has _dropdownRef) or DatePicker QuickPick (has _flatpickr)
|
|
355
|
+
if (element?._dropdownRef?.current) {
|
|
356
|
+
element._dropdownRef.current.clearSelected();
|
|
357
|
+
} else {
|
|
358
|
+
const quickpick = element?._flatpickr;
|
|
359
|
+
quickpick?.clear();
|
|
360
|
+
}
|
|
355
361
|
}
|
|
356
362
|
});
|
|
357
363
|
}
|
|
@@ -360,8 +366,14 @@ const datePickerHelper = (config: DatePickerConfig, scrollContainer: string | HT
|
|
|
360
366
|
if (syncEndWith) {
|
|
361
367
|
picker.config.onClose.push((selectedDates: string) => {
|
|
362
368
|
if (selectedDates?.length) {
|
|
363
|
-
const
|
|
364
|
-
|
|
369
|
+
const element = document.querySelector(`#${syncEndWith}`) as any;
|
|
370
|
+
// Check if it's a Dropdown QuickPick (has _dropdownRef) or DatePicker QuickPick (has _flatpickr)
|
|
371
|
+
if (element?._dropdownRef?.current) {
|
|
372
|
+
element._dropdownRef.current.clearSelected();
|
|
373
|
+
} else {
|
|
374
|
+
const quickpick = element?._flatpickr;
|
|
375
|
+
quickpick?.clear();
|
|
376
|
+
}
|
|
365
377
|
}
|
|
366
378
|
});
|
|
367
379
|
}
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import React from "react";
|
|
2
|
+
import Dropdown from "../../pb_dropdown/_dropdown";
|
|
3
|
+
import DatePicker from "../../pb_date_picker/_date_picker";
|
|
4
|
+
|
|
5
|
+
const DatePickerAndDropdownRange = (props) => {
|
|
6
|
+
return (
|
|
7
|
+
<>
|
|
8
|
+
<Dropdown
|
|
9
|
+
controlsEndId="end-date-picker1"
|
|
10
|
+
controlsStartId="start-date-picker1"
|
|
11
|
+
id="dropdown-as-quickpick"
|
|
12
|
+
label="Date Range"
|
|
13
|
+
marginBottom="sm"
|
|
14
|
+
placeholder="Select a Date Range"
|
|
15
|
+
variant="quickpick"
|
|
16
|
+
{...props}
|
|
17
|
+
/>
|
|
18
|
+
|
|
19
|
+
<DatePicker
|
|
20
|
+
label="Start Date"
|
|
21
|
+
pickerId="start-date-picker1"
|
|
22
|
+
placeholder="Select a Start Date"
|
|
23
|
+
syncStartWith="dropdown-as-quickpick"
|
|
24
|
+
{...props}
|
|
25
|
+
/>
|
|
26
|
+
|
|
27
|
+
<DatePicker
|
|
28
|
+
label="End Date"
|
|
29
|
+
pickerId="end-date-picker1"
|
|
30
|
+
placeholder="Select an End Date"
|
|
31
|
+
syncEndWith="dropdown-as-quickpick"
|
|
32
|
+
{...props}
|
|
33
|
+
/>
|
|
34
|
+
</>
|
|
35
|
+
);
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
export default DatePickerAndDropdownRange;
|