playbook_ui 14.16.0.pre.alpha.PLAY1929bracketlayout6957 → 14.16.0.pre.alpha.PLAY1938completetooltipfloatinguitransition6830

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 (74) hide show
  1. checksums.yaml +4 -4
  2. data/app/pb_kits/playbook/pb_advanced_table/_advanced_table.scss +2 -34
  3. data/app/pb_kits/playbook/pb_advanced_table/_advanced_table.tsx +84 -86
  4. data/app/pb_kits/playbook/pb_advanced_table/advanced_table.html.erb +2 -2
  5. data/app/pb_kits/playbook/pb_advanced_table/advanced_table.rb +0 -10
  6. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_pagination.jsx +1 -0
  7. data/app/pb_kits/playbook/pb_advanced_table/docs/example.yml +0 -2
  8. data/app/pb_kits/playbook/pb_advanced_table/index.js +1 -1
  9. data/app/pb_kits/playbook/pb_advanced_table/table_body.rb +2 -4
  10. data/app/pb_kits/playbook/pb_advanced_table/table_header.html.erb +9 -19
  11. data/app/pb_kits/playbook/pb_advanced_table/table_header.rb +1 -38
  12. data/app/pb_kits/playbook/pb_advanced_table/table_row.html.erb +37 -49
  13. data/app/pb_kits/playbook/pb_advanced_table/table_row.rb +0 -39
  14. data/app/pb_kits/playbook/pb_button/_button.scss +5 -5
  15. data/app/pb_kits/playbook/pb_date_picker/_date_picker.tsx +34 -34
  16. data/app/pb_kits/playbook/pb_date_picker/date_picker.html.erb +2 -2
  17. data/app/pb_kits/playbook/pb_date_picker/date_picker_helper.ts +0 -16
  18. data/app/pb_kits/playbook/pb_date_picker/docs/_date_picker_inline.html.erb +11 -0
  19. data/app/pb_kits/playbook/pb_date_picker/docs/_date_picker_inline.jsx +7 -0
  20. data/app/pb_kits/playbook/pb_date_picker/sass_partials/_inline_styles.scss +24 -28
  21. data/app/pb_kits/playbook/pb_filter/Filter/CurrentFilters.tsx +4 -3
  22. data/app/pb_kits/playbook/pb_filter/Filter/SortMenu.tsx +3 -2
  23. data/app/pb_kits/playbook/pb_form/pb_form_validation.js +1 -1
  24. data/app/pb_kits/playbook/pb_layout/_layout.scss +0 -58
  25. data/app/pb_kits/playbook/pb_layout/_layout.tsx +7 -20
  26. data/app/pb_kits/playbook/pb_layout/docs/example.yml +0 -1
  27. data/app/pb_kits/playbook/pb_layout/docs/index.js +0 -1
  28. data/app/pb_kits/playbook/pb_layout/layout.test.js +0 -4
  29. data/app/pb_kits/playbook/pb_lightbox/hooks/useVisibility.js +1 -1
  30. data/app/pb_kits/playbook/pb_multi_level_select/_multi_level_select.tsx +1 -2
  31. data/app/pb_kits/playbook/pb_rich_text_editor/_rich_text_editor.tsx +11 -29
  32. data/app/pb_kits/playbook/pb_rich_text_editor/docs/_rich_text_editor_default.html.erb +1 -1
  33. data/app/pb_kits/playbook/pb_rich_text_editor/rich_text_editor.html.erb +4 -4
  34. data/app/pb_kits/playbook/pb_rich_text_editor/rich_text_editor.rb +0 -2
  35. data/app/pb_kits/playbook/pb_table/_table.tsx +0 -4
  36. data/app/pb_kits/playbook/pb_table/docs/example.yml +0 -2
  37. data/app/pb_kits/playbook/pb_table/docs/index.js +0 -2
  38. data/app/pb_kits/playbook/pb_table/styles/_headers.scss +0 -76
  39. data/app/pb_kits/playbook/pb_table/subcomponents/_table_head.tsx +1 -11
  40. data/app/pb_kits/playbook/pb_table/subcomponents/_table_header.tsx +1 -11
  41. data/app/pb_kits/playbook/pb_table/subcomponents/_table_row.tsx +0 -5
  42. data/app/pb_kits/playbook/pb_table/table.test.js +0 -17
  43. data/app/pb_kits/playbook/pb_tooltip/index.js +1 -1
  44. data/app/pb_kits/playbook/pb_typeahead/index.ts +2 -2
  45. data/app/pb_kits/playbook/utilities/globalProps.ts +1 -1
  46. data/app/pb_kits/playbook/utilities/object.test.js +1 -149
  47. data/app/pb_kits/playbook/utilities/object.ts +42 -124
  48. data/dist/chunks/_typeahead-BEyzuDQy.js +22 -0
  49. data/dist/chunks/_weekday_stacked-BWYgED9z.js +45 -0
  50. data/dist/chunks/{lib-BGzBzFZX.js → lib-BgzBJfYr.js} +3 -3
  51. data/dist/chunks/{pb_form_validation-BvNy9Bd6.js → pb_form_validation-CJD-PyIw.js} +1 -1
  52. data/dist/chunks/vendor.js +1 -1
  53. data/dist/playbook-doc.js +1 -1
  54. data/dist/playbook-rails-react-bindings.js +1 -1
  55. data/dist/playbook-rails.js +1 -1
  56. data/dist/playbook.css +1 -1
  57. data/lib/playbook/version.rb +1 -1
  58. metadata +8 -20
  59. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_selectable_rows.html.erb +0 -39
  60. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_selectable_rows_no_subrows.html.erb +0 -33
  61. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_selectable_rows_no_subrows_rails.md +0 -1
  62. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_selectable_rows_rails.md +0 -6
  63. data/app/pb_kits/playbook/pb_layout/docs/_layout_bracket.jsx +0 -190
  64. data/app/pb_kits/playbook/pb_layout/docs/_layout_bracket.md +0 -5
  65. data/app/pb_kits/playbook/pb_layout/subcomponents/_game.tsx +0 -90
  66. data/app/pb_kits/playbook/pb_layout/subcomponents/_round.tsx +0 -57
  67. data/app/pb_kits/playbook/pb_table/docs/_table_with_header_style_borderless.jsx +0 -50
  68. data/app/pb_kits/playbook/pb_table/docs/_table_with_header_style_borderless_react.md +0 -1
  69. data/app/pb_kits/playbook/pb_table/docs/_table_with_header_style_floating.jsx +0 -59
  70. data/app/pb_kits/playbook/pb_table/docs/_table_with_header_style_floating_react.md +0 -1
  71. data/dist/chunks/_typeahead-Djo6qCne.js +0 -22
  72. data/dist/chunks/_weekday_stacked-C9nJ2j2C.js +0 -45
  73. /data/app/pb_kits/playbook/pb_advanced_table/docs/{_advanced_table_selectable_rows_react.md → _advanced_table_selectable_rows.md} +0 -0
  74. /data/app/pb_kits/playbook/pb_advanced_table/docs/{_advanced_table_selectable_rows_no_subrows_react.md → _advanced_table_selectable_rows_no_subrows.md} +0 -0
@@ -1 +1 @@
1
- <%= pb_rails("rich_text_editor", props: {input_options: { id: 'hidden_input_id', name: "hidden_input_name" }, value: "Add your text here. You can format your text, add links, quotes, and bullets."}) %>
1
+ <%= pb_rails("rich_text_editor", props: {id: "default", value: "Add your text here. You can format your text, add links, quotes, and bullets."}) %>
@@ -1,5 +1,5 @@
1
1
  <%= react_component('RichTextEditor',
2
- object.rich_text_options,
3
- aria: object.aria,
4
- data: object.data
5
- ) %>
2
+ object.rich_text_options,
3
+ aria: object.aria,
4
+ data: object.data
5
+ ) %>
@@ -20,7 +20,6 @@ module Playbook
20
20
  prop :value
21
21
  prop :template
22
22
  prop :placeholder
23
- prop :input_options
24
23
 
25
24
  def classname
26
25
  generate_classname("pb_rich_text_editor_kit", simple_class, focus_class, sticky_class, separator: " ")
@@ -50,7 +49,6 @@ module Playbook
50
49
  value: value,
51
50
  template: template,
52
51
  placeholder: placeholder,
53
- inputOptions: input_options,
54
52
  }
55
53
  end
56
54
  end
@@ -21,7 +21,6 @@ type TableProps = {
21
21
  data?: { [key: string]: string },
22
22
  dataTable: boolean,
23
23
  disableHover?: boolean,
24
- headerStyle?: "default" | "borderless" | "floating"
25
24
  htmlOptions?: { [key: string]: string | number | boolean | (() => void) },
26
25
  id?: string,
27
26
  outerPadding?: "none" | "xxs" | "xs" | "sm" | "md" | "lg" | "xl",
@@ -47,7 +46,6 @@ const Table = (props: TableProps): React.ReactElement => {
47
46
  data = {},
48
47
  dataTable = false,
49
48
  disableHover = false,
50
- headerStyle = "default",
51
49
  htmlOptions = {},
52
50
  id,
53
51
  outerPadding = '',
@@ -87,8 +85,6 @@ const Table = (props: TableProps): React.ReactElement => {
87
85
  'sticky-left-column': stickyLeftColumn,
88
86
  'sticky-right-column': stickyRightColumn,
89
87
  'striped': striped,
90
- 'header-borderless': headerStyle === 'borderless',
91
- 'header-floating': headerStyle === 'floating',
92
88
  [outerPaddingCss]: outerPadding !== '',
93
89
  },
94
90
  globalProps(props),
@@ -75,5 +75,3 @@ examples:
75
75
  - table_with_collapsible_with_nested_table: Table with Collapsible with Nested Table
76
76
  - table_with_clickable_rows: Table with Clickable Rows
77
77
  - table_with_selectable_rows: Table with Selectable Rows
78
- - table_with_header_style_borderless: Header Style Borderless
79
- - table_with_header_style_floating: Header Style Floating
@@ -35,5 +35,3 @@ export { default as TableWithCollapsibleWithNestedRows } from './_table_with_col
35
35
  export { default as TableWithCollapsibleWithCustomClick } from './_table_with_collapsible_with_custom_click.jsx'
36
36
  export { default as TableWithSelectableRows } from './_table_with_selectable_rows.jsx'
37
37
  export { default as TableWithClickableRows } from './_table_with_clickable_rows.jsx'
38
- export { default as TableWithHeaderStyleBorderless } from './_table_with_header_style_borderless.jsx'
39
- export { default as TableWithHeaderStyleFloating } from './_table_with_header_style_floating.jsx'
@@ -14,79 +14,3 @@
14
14
  }
15
15
  }
16
16
  }
17
- // remove all outward facing borders from header
18
- .header-borderless > thead,
19
- .header-borderless > thead > tr,
20
- .header-borderless > thead > tr > th,
21
- .header-borderless > .pb_table_thead,
22
- .header-borderless > .pb_table_thead > .pb_table_tr,
23
- .header-borderless > .pb_table_thead .pb_table_th {
24
- border-top: none !important;
25
- border-left: none !important;
26
- border-right: none !important;
27
- }
28
-
29
- // remove bottom border (internal to table) on header cells - only light mode needs this redundancy handled likely due to a difference in the base scss and table-dark.scss
30
- .header-borderless:not(.table-dark) > thead > tr:last-child > th,
31
- .header-borderless:not(.table-dark) > .pb_table_thead > .pb_table_tr:last-child > .pb_table_th {
32
- border-bottom: none !important;
33
- }
34
-
35
- // preserves top rounded corners to header at top of table-card for unnested card
36
- .header-borderless > thead > tr:first-child > th:first-child,
37
- .header-borderless > .pb_table_thead > .pb_table_tr:first-child > .pb_table_th:first-child {
38
- border-top-left-radius: $border_radius_md !important;
39
- }
40
-
41
- .header-borderless > thead > tr:first-child > th:last-child,
42
- .header-borderless > .pb_table_thead > .pb_table_tr:first-child > .pb_table_th:last-child {
43
- border-top-right-radius: $border_radius_md !important;
44
- }
45
-
46
- // ensures top border is from first row of body to prevent double borders
47
- .header-borderless > tbody > tr:first-child > td,
48
- .header-borderless > .pb_table_tbody > .pb_table_tr:first-child > .pb_table_td {
49
- border-top: 1px solid $border_light !important;
50
- }
51
-
52
- // floating code - this carries over everything for header-borderless
53
- .header-floating > thead,
54
- .header-floating > thead > tr,
55
- .header-floating > thead > tr > th,
56
- .header-floating > .pb_table_thead,
57
- .header-floating > .pb_table_thead > .pb_table_tr,
58
- .header-floating > .pb_table_thead .pb_table_th {
59
- border-top: none !important;
60
- border-left: none !important;
61
- border-right: none !important;
62
- }
63
-
64
- .header-floating:not(.table-dark) > thead > tr:last-child > th,
65
- .header-floating:not(.table-dark) > .pb_table_thead > .pb_table_tr:last-child > .pb_table_th {
66
- border-bottom: none !important;
67
- }
68
-
69
- .header-floating > thead > tr:first-child > th:first-child,
70
- .header-floating > .pb_table_thead > .pb_table_tr:first-child > .pb_table_th:first-child {
71
- border-top-left-radius: $border_radius_md !important;
72
- }
73
-
74
- .header-floating > thead > tr:first-child > th:last-child,
75
- .header-floating > .pb_table_thead > .pb_table_tr:first-child > .pb_table_th:last-child {
76
- border-top-right-radius: $border_radius_md !important;
77
- }
78
-
79
- .header-floating > tbody > tr:first-child > td,
80
- .header-floating > .pb_table_tbody > .pb_table_tr:first-child > .pb_table_td {
81
- border-top: 1px solid $border_light !important;
82
- }
83
-
84
- // flatten out corners for floating headerstyle variant to avoid small triangle of white/empty space
85
- .header-floating > thead > tr:first-child > th:first-child,
86
- .header-floating > .pb_table_thead > .pb_table_tr:first-child > .pb_table_th:first-child {
87
- border-top-left-radius: 0 !important;
88
- }
89
- .header-floating > thead > tr:first-child > th:last-child,
90
- .header-floating > .pb_table_thead > .pb_table_tr:first-child > .pb_table_th:last-child {
91
- border-top-right-radius: 0 !important;
92
- }
@@ -12,7 +12,6 @@ type TableHeadPropTypes = {
12
12
  children: React.ReactNode[] | React.ReactNode;
13
13
  className: string;
14
14
  data?: { [key: string]: string };
15
- headerStyle?: "default" | "borderless" | "floating";
16
15
  htmlOptions?: { [key: string]: string | number | boolean | (() => void) };
17
16
  id?: string;
18
17
  tag?: "table" | "div";
@@ -24,7 +23,6 @@ const TableHead = (props: TableHeadPropTypes): React.ReactElement => {
24
23
  children,
25
24
  className,
26
25
  data = {},
27
- headerStyle = "default",
28
26
  htmlOptions = {},
29
27
  id,
30
28
  tag = "table",
@@ -33,15 +31,7 @@ const TableHead = (props: TableHeadPropTypes): React.ReactElement => {
33
31
  const ariaProps = buildAriaProps(aria);
34
32
  const dataProps = buildDataProps(data);
35
33
  const htmlProps = buildHtmlProps(htmlOptions);
36
- const classes = classnames(
37
- "pb_table_thead",
38
- {
39
- "pb_table_thead_borderless": headerStyle === "borderless" || headerStyle === "floating",
40
- "pb_table_thead_floating": headerStyle === "floating",
41
- },
42
- globalProps(props),
43
- className
44
- );
34
+ const classes = classnames("pb_table_thead", globalProps(props), className);
45
35
  const isTableTag = tag === "table";
46
36
 
47
37
  return (
@@ -12,7 +12,6 @@ type TableHeaderPropTypes = {
12
12
  children: React.ReactNode[] | React.ReactNode;
13
13
  className: string;
14
14
  data?: { [key: string]: string };
15
- headerStyle?: "default" | "borderless" | "floating";
16
15
  htmlOptions?: { [key: string]: string | number | boolean | (() => void) };
17
16
  id?: string;
18
17
  tag?: "table" | "div";
@@ -25,7 +24,6 @@ const TableHeader = (props: TableHeaderPropTypes): React.ReactElement => {
25
24
  children,
26
25
  className,
27
26
  data = {},
28
- headerStyle = "default",
29
27
  htmlOptions = {},
30
28
  id,
31
29
  tag = "table",
@@ -35,15 +33,7 @@ const TableHeader = (props: TableHeaderPropTypes): React.ReactElement => {
35
33
  const ariaProps = buildAriaProps(aria);
36
34
  const dataProps = buildDataProps(data);
37
35
  const htmlProps = buildHtmlProps(htmlOptions);
38
- const classes = classnames(
39
- "pb_table_th",
40
- {
41
- "pb_table_thead_borderless": headerStyle === "borderless" || headerStyle === "floating",
42
- "pb_table_thead_floating": headerStyle === "floating",
43
- },
44
- globalProps(props),
45
- className
46
- );
36
+ const classes = classnames("pb_table_th", globalProps(props), className);
47
37
  const isTableTag = tag === "table";
48
38
 
49
39
  return (
@@ -22,7 +22,6 @@ type TableRowPropTypes = {
22
22
  dark?: boolean;
23
23
  dragId?: string;
24
24
  draggableItem?: boolean;
25
- headerStyle?: "default" | "borderless" | "floating";
26
25
  htmlOptions?: { [key: string]: string | number | boolean | (() => void) };
27
26
  id?: string;
28
27
  toggleCellId?: string;
@@ -42,7 +41,6 @@ const TableRow = (props: TableRowPropTypes): React.ReactElement => {
42
41
  dark = false,
43
42
  dragId,
44
43
  draggableItem = false,
45
- headerStyle = "default",
46
44
  htmlOptions = {},
47
45
  id,
48
46
  toggleCellId,
@@ -62,9 +60,6 @@ const TableRow = (props: TableRowPropTypes): React.ReactElement => {
62
60
  const classes = classnames(
63
61
  buildCss("pb_table_row_kit", sideHighlightClass),
64
62
  "pb_table_tr",
65
- {
66
- "pb_table_tr_borderless_header": headerStyle === "borderless",
67
- },
68
63
  collapsibleRow,
69
64
  globalProps(props),
70
65
  className
@@ -167,20 +167,3 @@ test("Should have outerPadding class", () => {
167
167
  const kit = renderKit(Table, props, { outerPadding: "sm" })
168
168
  expect(kit).toHaveClass("pb_table table-sm table-responsive-collapse table-card outer_padding_space_sm table-collapse-sm")
169
169
  })
170
-
171
- test("when headerStyle is default", () => {
172
- const kit = renderKit(Table, props)
173
- expect(kit).toHaveClass("pb_table table-sm table-responsive-collapse table-card table-collapse-sm")
174
- expect(kit).not.toHaveClass("header-borderless")
175
- expect(kit).not.toHaveClass("header-floating")
176
- })
177
-
178
- test("when headerStyle is borderless", () => {
179
- const kit = renderKit(Table, props, { headerStyle: "borderless" })
180
- expect(kit).toHaveClass("pb_table table-sm table-responsive-collapse table-card header-borderless table-collapse-sm")
181
- })
182
-
183
- test("when headerStyle is floating", () => {
184
- const kit = renderKit(Table, props, { headerStyle: "floating" })
185
- expect(kit).toHaveClass("pb_table table-sm table-responsive-collapse table-card header-floating table-collapse-sm")
186
- })
@@ -5,7 +5,7 @@ const TOOLTIP_OFFSET = 20
5
5
  const TOOLTIP_TIMEOUT = 250
6
6
  const SAFE_ZONE_MARGIN = 1
7
7
 
8
- export default class PbTooltip extends PbEnhancedElement {
8
+ export default class PbTooltipFloatingUi extends PbEnhancedElement {
9
9
  static get selector() {
10
10
  return '[data-pb-tooltip-kit]'
11
11
  }
@@ -1,5 +1,5 @@
1
1
  import PbEnhancedElement from '../pb_enhanced_element'
2
- import { debounce } from '../utilities/object'
2
+ import { debounce } from 'lodash'
3
3
 
4
4
  export default class PbTypeahead extends PbEnhancedElement {
5
5
  _searchInput: HTMLInputElement
@@ -284,4 +284,4 @@ export default class PbTypeahead extends PbEnhancedElement {
284
284
  if (visible) visibilityProperty = '1'
285
285
  this.resultsLoadingIndicator.style.opacity = visibilityProperty
286
286
  }
287
- }
287
+ }
@@ -1,4 +1,4 @@
1
- import { omit } from './object'
1
+ import { omit } from 'lodash'
2
2
  import { camelToSnakeCase } from './text'
3
3
 
4
4
  import {
@@ -1,4 +1,4 @@
1
- import { isEmpty, get, isString, uniqueId, omitBy, noop, merge, filter, find, partial, map, cloneDeep, omit, debounce } from './object';
1
+ import { isEmpty, get, isString, uniqueId, omitBy, noop, merge, filter, find, partial } from './object';
2
2
 
3
3
  describe('Lodash functions', () => {
4
4
  describe('isEmpty', () => {
@@ -234,152 +234,4 @@ describe('Lodash functions', () => {
234
234
  expect(joinPartial('b', 'c')).toBe('a_b_c');
235
235
  });
236
236
  });
237
-
238
- describe('map', () => {
239
- test('maps over an array with a function iteratee', () => {
240
- const arr = [1, 2, 3];
241
- const result = map(arr, (num) => num * 2);
242
- expect(result).toEqual([2, 4, 6]);
243
- });
244
-
245
- test('maps over an array with a string iteratee', () => {
246
- const arr = [{ value: 1 }, { value: 2 }, { value: 3 }];
247
- const result = map(arr, 'value');
248
- expect(result).toEqual([1, 2, 3]);
249
- });
250
-
251
- test('maps over an object with a function iteratee', () => {
252
- const obj = { a: 1, b: 2, c: 3 };
253
- const result = map(obj, (val, key) => key + val);
254
- expect(result.sort()).toEqual(['a1', 'b2', 'c3'].sort());
255
- });
256
-
257
- test('maps over an object with a string iteratee', () => {
258
- const obj = {
259
- one: { num: 1 },
260
- two: { num: 2 },
261
- three: { num: 3 },
262
- };
263
- const result = map(obj, 'num');
264
- expect(result.sort()).toEqual([1, 2, 3].sort());
265
- });
266
-
267
- test('returns original values if no iteratee provided', () => {
268
- const arr = [1, 2, 3];
269
- const result = map(arr);
270
- expect(result).toEqual([1, 2, 3]);
271
- });
272
- });
273
-
274
- describe('cloneDeep', () => {
275
- test('clones primitive values', () => {
276
- expect(cloneDeep(42)).toBe(42);
277
- expect(cloneDeep('test')).toBe('test');
278
- expect(cloneDeep(null)).toBe(null);
279
- });
280
-
281
- test('clones arrays deeply', () => {
282
- const arr = [1, [2, 3]];
283
- const cloned = cloneDeep(arr);
284
- expect(cloned).toEqual(arr);
285
- cloned[1][0] = 99;
286
- expect(arr[1][0]).toBe(2);
287
- });
288
-
289
- test('clones objects deeply', () => {
290
- const obj = { a: { b: 2 } };
291
- const cloned = cloneDeep(obj);
292
- expect(cloned).toEqual(obj);
293
- cloned.a.b = 99;
294
- expect(obj.a.b).toBe(2);
295
- });
296
-
297
- test('clones Date objects', () => {
298
- const date = new Date();
299
- const cloned = cloneDeep(date);
300
- expect(cloned).not.toBe(date);
301
- expect(cloned.getTime()).toBe(date.getTime());
302
- });
303
-
304
- test('clones RegExp objects', () => {
305
- const regex = /test/gi;
306
- const cloned = cloneDeep(regex);
307
- expect(cloned).not.toBe(regex);
308
- expect(cloned.source).toBe(regex.source);
309
- expect(cloned.flags).toBe(regex.flags);
310
- });
311
- });
312
-
313
- describe('omit', () => {
314
- test('omits specified keys from object', () => {
315
- const obj = { a: 1, b: 2, c: 3 };
316
- expect(omit(obj, 'a', 'c')).toEqual({ b: 2 });
317
- });
318
-
319
- test('supports array of keys to omit', () => {
320
- const obj = { a: 1, b: 2, c: 3 };
321
- expect(omit(obj, ['b'])).toEqual({ a: 1, c: 3 });
322
- });
323
-
324
- test('returns empty object for null or non-object input', () => {
325
- expect(omit(null, 'a')).toEqual({});
326
- expect(omit("string", 'a')).toEqual({});
327
- });
328
-
329
- test('returns original object if no keys match', () => {
330
- const obj = { a: 1, b: 2 };
331
- expect(omit(obj, 'c')).toEqual({ a: 1, b: 2 });
332
- });
333
- });
334
-
335
- describe('debounce', () => {
336
- beforeEach(() => {
337
- jest.useFakeTimers();
338
- });
339
-
340
- afterEach(() => {
341
- jest.useRealTimers();
342
- });
343
-
344
- test('delays execution until wait time has passed', () => {
345
- const func = jest.fn();
346
- const debounced = debounce(func, 1000);
347
- debounced();
348
- expect(func).not.toHaveBeenCalled();
349
- jest.advanceTimersByTime(500);
350
- expect(func).not.toHaveBeenCalled();
351
- jest.advanceTimersByTime(500);
352
- expect(func).toHaveBeenCalledTimes(1);
353
- });
354
-
355
- test('calls function only once when called repeatedly', () => {
356
- const func = jest.fn();
357
- const debounced = debounce(func, 1000);
358
- debounced();
359
- debounced();
360
- debounced();
361
- jest.advanceTimersByTime(1000);
362
- expect(func).toHaveBeenCalledTimes(1);
363
- });
364
-
365
- test('immediate option calls function on first call', () => {
366
- const func = jest.fn();
367
- const debounced = debounce(func, 1000, true);
368
- debounced();
369
- expect(func).toHaveBeenCalledTimes(1);
370
- debounced();
371
- debounced();
372
- jest.advanceTimersByTime(1000);
373
- expect(func).toHaveBeenCalledTimes(1);
374
- });
375
-
376
- test('subsequent call after wait period works with immediate option', () => {
377
- const func = jest.fn();
378
- const debounced = debounce(func, 1000, true);
379
- debounced();
380
- jest.advanceTimersByTime(1100);
381
- debounced();
382
- expect(func).toHaveBeenCalledTimes(2);
383
- });
384
- });
385
237
  });
@@ -1,7 +1,6 @@
1
1
  /* 🛠️ Any commonly used lodash functions can be added here. 🤙 */
2
2
 
3
- export const isEmpty = (obj: any) =>
4
- [Object, Array].includes((obj || {}).constructor) && !Object.entries((obj || {})).length
3
+ export const isEmpty = (obj: any) => [Object, Array].includes((obj || {}).constructor) && !Object.entries((obj || {})).length;
5
4
 
6
5
  export const get = <T, R = any>(obj: T, path: string, defaultValue?: R): R | any => {
7
6
  const travel = (regexp: RegExp): any =>
@@ -25,175 +24,94 @@ export const omitBy = (obj: Record<string, any>, predicate: (value: any, key: st
25
24
  if (obj === null || typeof obj !== 'object') return {}
26
25
  return Object.keys(obj).reduce((result: Record<string, any>, key: string) => {
27
26
  if (!predicate(obj[key], key)) {
28
- result[key] = obj[key]
27
+ result[key] = obj[key];
29
28
  }
30
- return result
29
+ return result;
31
30
  }, {})
32
31
  }
33
32
 
34
- export const omit = (
35
- obj: Record<string, any>,
36
- ...paths: (string | string[])[]
37
- ): Record<string, any> => {
38
- if (obj === null || typeof obj !== 'object') return {}
39
- const keysToOmit = new Set<string>()
40
- paths.forEach(p => {
41
- if (Array.isArray(p)) {
42
- p.forEach(key => keysToOmit.add(key))
43
- } else {
44
- keysToOmit.add(p)
45
- }
46
- })
47
- const result: Record<string, any> = {}
48
- for (const key in obj) {
49
- if (!keysToOmit.has(key)) {
50
- result[key] = obj[key]
51
- }
52
- }
53
- return result
54
- }
55
-
56
33
  export const noop = (): void => {
57
34
  // empty
58
- }
59
-
60
- export const cloneDeep = (value: any): any => {
61
- if (value === null || typeof value !== 'object') {
62
- return value
63
- }
64
- if (Array.isArray(value)) {
65
- return value.map(cloneDeep)
66
- }
67
- if (value instanceof Date) {
68
- return new Date(value.getTime())
69
- }
70
- if (value instanceof RegExp) {
71
- return new RegExp(value.source, value.flags)
72
- }
73
- const clonedObj: any = {}
74
- for (const key in value) {
75
- if (Object.prototype.hasOwnProperty.call(value, key)) {
76
- clonedObj[key] = cloneDeep(value[key])
77
- }
78
- }
79
- return clonedObj
80
- }
35
+ };
81
36
 
82
37
  export function merge(
83
38
  ...objects: Array<Record<string, unknown> | null | undefined>
84
39
  ): Record<string, unknown> {
85
40
  const isPlainObject = (obj: unknown): obj is Record<string, unknown> =>
86
- !!obj && typeof obj === 'object' && !Array.isArray(obj)
41
+ !!obj && typeof obj === 'object' && !Array.isArray(obj);
87
42
 
88
- const result: Record<string, unknown> = {}
43
+ const result: Record<string, unknown> = {};
89
44
 
90
45
  for (const obj of objects) {
91
- if (!obj || typeof obj !== 'object') continue
46
+ if (!obj || typeof obj !== 'object') continue;
92
47
 
93
48
  for (const key of Object.keys(obj)) {
94
- const oldVal = result[key]
95
- const newVal = (obj as Record<string, unknown>)[key]
49
+ const oldVal = result[key];
50
+ const newVal = (obj as Record<string, unknown>)[key];
96
51
 
97
52
  if (Array.isArray(oldVal) && Array.isArray(newVal)) {
98
- result[key] = newVal
53
+ result[key] = newVal;
99
54
  } else if (isPlainObject(oldVal) && isPlainObject(newVal)) {
100
- result[key] = merge(oldVal, newVal)
55
+ result[key] = merge(oldVal, newVal);
101
56
  } else if (Array.isArray(oldVal) && isPlainObject(newVal)) {
102
- result[key] = oldVal
57
+ result[key] = oldVal;
103
58
  } else if (isPlainObject(oldVal) && Array.isArray(newVal)) {
104
- result[key] = oldVal
59
+ result[key] = oldVal;
105
60
  } else {
106
- result[key] = newVal
61
+ result[key] = newVal;
107
62
  }
108
63
  }
109
64
  }
110
- return result
65
+ return result;
111
66
  }
112
67
 
113
68
  const createIteratee = (predicate: any) => {
114
69
  if (typeof predicate === 'function') {
115
- return predicate
70
+ return predicate;
116
71
  }
117
72
  if (typeof predicate === 'string') {
118
- return (obj: any) => obj[predicate]
73
+ return (obj: any) => obj[predicate];
119
74
  }
120
75
  if (Array.isArray(predicate)) {
121
- const [key, value] = predicate
122
- return (obj: any) => obj[key] === value
76
+ const [key, value] = predicate;
77
+ return (obj: any) => obj[key] === value;
123
78
  }
124
79
  if (typeof predicate === 'object' && predicate !== null) {
125
80
  return (obj: any) => {
126
81
  for (const key in predicate) {
127
82
  if (Object.prototype.hasOwnProperty.call(predicate, key)) {
128
- if (obj[key] !== predicate[key]) return false
83
+ if (obj[key] !== predicate[key]) return false;
129
84
  }
130
85
  }
131
- return true
132
- }
86
+ return true;
87
+ };
133
88
  }
134
- return () => false
135
- }
89
+ return () => false;
90
+ };
136
91
 
137
92
  export const filter = <T>(array: T[], predicate: any): T[] => {
138
- const iteratee = createIteratee(predicate)
139
- return array.filter(iteratee)
140
- }
93
+ const iteratee = createIteratee(predicate);
94
+ return array.filter(iteratee);
95
+ };
141
96
 
142
97
  export const find = <T>(array: T[], predicate: any): T | undefined => {
143
- const iteratee = createIteratee(predicate)
144
- return array.find(iteratee)
145
- }
98
+ const iteratee = createIteratee(predicate);
99
+ return array.find(iteratee);
100
+ };
146
101
 
147
- export const partial = <F extends (...args: any[]) => any>(
102
+ export const partial = <F extends (...args: unknown[]) => unknown>(
148
103
  fn: F,
149
- ...partials: any[]
150
- ): ((...args: any[]) => ReturnType<F>) => {
151
- const placeholder = partial.placeholder
152
- return (...args: any[]): ReturnType<F> => {
153
- let argIndex = 0
104
+ ...partials: unknown[]
105
+ ): ((...args: unknown[]) => ReturnType<F>) => {
106
+ const placeholder = partial.placeholder;
107
+ return (...args: unknown[]): ReturnType<F> => {
108
+ let argIndex = 0;
154
109
  const finalArgs = partials.map(arg =>
155
110
  arg === placeholder ? args[argIndex++] : arg
156
- )
157
- return fn(...(finalArgs.concat(args.slice(argIndex)) as Parameters<F>)) as ReturnType<F>
158
- }
159
- }
160
-
161
- partial.placeholder = Symbol()
162
- export const _ = partial.placeholder
163
-
164
- export const map = <T, U>(
165
- collection: T[] | Record<string, T>,
166
- iteratee?: ((value: T, index: number | string, collection: T[] | Record<string, T>) => U) | string
167
- ): U[] => {
168
- const fn =
169
- typeof iteratee === "function"
170
- ? iteratee
171
- : typeof iteratee === "string"
172
- ? (item: T) => (item as any)[iteratee]
173
- : (item: T) => item as unknown as U
174
-
175
- if (Array.isArray(collection)) {
176
- return collection.map((value, index) => fn(value, index, collection))
177
- }
178
- return Object.keys(collection).map(key => fn(collection[key], key, collection))
179
- }
180
-
181
- export function debounce<F extends (...args: any[]) => any>(
182
- func: F,
183
- wait: number,
184
- immediate?: boolean
185
- ): (...args: Parameters<F>) => void {
186
- let timeout: ReturnType<typeof setTimeout> | null;
187
- return function(this: any, ...args: any[]) {
188
- if (timeout) clearTimeout(timeout);
189
- if (immediate && !timeout) {
190
- func.apply(this, args);
191
- }
192
- timeout = setTimeout(() => {
193
- timeout = null;
194
- if (!immediate) {
195
- func.apply(this, args);
196
- }
197
- }, wait);
111
+ );
112
+ return fn(...(finalArgs.concat(args.slice(argIndex)) as Parameters<F>)) as ReturnType<F>;
198
113
  };
199
- }
114
+ };
115
+
116
+ partial.placeholder = Symbol();
117
+ export const _ = partial.placeholder;