playbook_ui 14.16.0.pre.alpha.PLAY1938completetooltipfloatinguitransition6830 → 14.16.0.pre.alpha.play1756pbcontenttag6933

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 (70) hide show
  1. checksums.yaml +4 -4
  2. data/app/pb_kits/playbook/pb_advanced_table/_advanced_table.scss +34 -2
  3. data/app/pb_kits/playbook/pb_advanced_table/_advanced_table.tsx +86 -84
  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 +10 -0
  6. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_pagination.jsx +0 -1
  7. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_selectable_rows.html.erb +39 -0
  8. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_selectable_rows_no_subrows.html.erb +33 -0
  9. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_selectable_rows_no_subrows_rails.md +1 -0
  10. data/app/pb_kits/playbook/pb_advanced_table/docs/_advanced_table_selectable_rows_rails.md +6 -0
  11. data/app/pb_kits/playbook/pb_advanced_table/docs/example.yml +2 -0
  12. data/app/pb_kits/playbook/pb_advanced_table/index.js +1 -1
  13. data/app/pb_kits/playbook/pb_advanced_table/table_body.rb +4 -2
  14. data/app/pb_kits/playbook/pb_advanced_table/table_header.html.erb +19 -9
  15. data/app/pb_kits/playbook/pb_advanced_table/table_header.rb +38 -1
  16. data/app/pb_kits/playbook/pb_advanced_table/table_row.html.erb +49 -37
  17. data/app/pb_kits/playbook/pb_advanced_table/table_row.rb +39 -0
  18. data/app/pb_kits/playbook/pb_button/_button.scss +5 -5
  19. data/app/pb_kits/playbook/pb_date_picker/_date_picker.tsx +34 -34
  20. data/app/pb_kits/playbook/pb_date_picker/date_picker.html.erb +2 -2
  21. data/app/pb_kits/playbook/pb_date_picker/date_picker_helper.ts +16 -0
  22. data/app/pb_kits/playbook/pb_date_picker/docs/_date_picker_inline.html.erb +0 -11
  23. data/app/pb_kits/playbook/pb_date_picker/docs/_date_picker_inline.jsx +0 -7
  24. data/app/pb_kits/playbook/pb_date_picker/sass_partials/_inline_styles.scss +28 -24
  25. data/app/pb_kits/playbook/pb_filter/Filter/CurrentFilters.tsx +3 -4
  26. data/app/pb_kits/playbook/pb_filter/Filter/SortMenu.tsx +2 -3
  27. data/app/pb_kits/playbook/pb_form/pb_form_validation.js +1 -1
  28. data/app/pb_kits/playbook/pb_lightbox/hooks/useVisibility.js +1 -1
  29. data/app/pb_kits/playbook/pb_multi_level_select/_multi_level_select.tsx +2 -1
  30. data/app/pb_kits/playbook/pb_rich_text_editor/_rich_text_editor.tsx +29 -11
  31. data/app/pb_kits/playbook/pb_rich_text_editor/docs/_rich_text_editor_default.html.erb +1 -1
  32. data/app/pb_kits/playbook/pb_rich_text_editor/rich_text_editor.html.erb +4 -4
  33. data/app/pb_kits/playbook/pb_rich_text_editor/rich_text_editor.rb +2 -0
  34. data/app/pb_kits/playbook/pb_table/_table.tsx +4 -0
  35. data/app/pb_kits/playbook/pb_table/docs/_table_with_header_style_borderless.jsx +50 -0
  36. data/app/pb_kits/playbook/pb_table/docs/_table_with_header_style_borderless_react.md +1 -0
  37. data/app/pb_kits/playbook/pb_table/docs/_table_with_header_style_floating.jsx +59 -0
  38. data/app/pb_kits/playbook/pb_table/docs/_table_with_header_style_floating_react.md +1 -0
  39. data/app/pb_kits/playbook/pb_table/docs/example.yml +2 -0
  40. data/app/pb_kits/playbook/pb_table/docs/index.js +2 -0
  41. data/app/pb_kits/playbook/pb_table/styles/_headers.scss +76 -0
  42. data/app/pb_kits/playbook/pb_table/subcomponents/_table_head.tsx +11 -1
  43. data/app/pb_kits/playbook/pb_table/subcomponents/_table_header.tsx +11 -1
  44. data/app/pb_kits/playbook/pb_table/subcomponents/_table_row.tsx +5 -0
  45. data/app/pb_kits/playbook/pb_table/table.test.js +17 -0
  46. data/app/pb_kits/playbook/pb_tooltip/index.js +1 -1
  47. data/app/pb_kits/playbook/pb_tooltip/tooltip.html.erb +2 -5
  48. data/app/pb_kits/playbook/pb_typeahead/index.ts +2 -2
  49. data/app/pb_kits/playbook/pb_typeahead/typeahead.html.erb +2 -5
  50. data/app/pb_kits/playbook/pb_user/user.html.erb +1 -6
  51. data/app/pb_kits/playbook/pb_user_badge/user_badge.html.erb +1 -6
  52. data/app/pb_kits/playbook/utilities/globalProps.ts +1 -1
  53. data/app/pb_kits/playbook/utilities/object.test.js +149 -1
  54. data/app/pb_kits/playbook/utilities/object.ts +124 -42
  55. data/dist/chunks/_typeahead-Djo6qCne.js +22 -0
  56. data/dist/chunks/_weekday_stacked-DIIHW0OV.js +45 -0
  57. data/dist/chunks/{lib-BgzBJfYr.js → lib-BGzBzFZX.js} +3 -3
  58. data/dist/chunks/{pb_form_validation-CJD-PyIw.js → pb_form_validation-BvNy9Bd6.js} +1 -1
  59. data/dist/chunks/vendor.js +1 -1
  60. data/dist/playbook-doc.js +1 -1
  61. data/dist/playbook-rails-react-bindings.js +1 -1
  62. data/dist/playbook-rails.js +1 -1
  63. data/dist/playbook.css +1 -1
  64. data/lib/playbook/kit_base.rb +4 -4
  65. data/lib/playbook/version.rb +1 -1
  66. metadata +16 -8
  67. data/dist/chunks/_typeahead-BEyzuDQy.js +0 -22
  68. data/dist/chunks/_weekday_stacked-BWYgED9z.js +0 -45
  69. /data/app/pb_kits/playbook/pb_advanced_table/docs/{_advanced_table_selectable_rows_no_subrows.md → _advanced_table_selectable_rows_no_subrows_react.md} +0 -0
  70. /data/app/pb_kits/playbook/pb_advanced_table/docs/{_advanced_table_selectable_rows.md → _advanced_table_selectable_rows_react.md} +0 -0
@@ -1,4 +1,4 @@
1
- import { isEmpty, get, isString, uniqueId, omitBy, noop, merge, filter, find, partial } from './object';
1
+ import { isEmpty, get, isString, uniqueId, omitBy, noop, merge, filter, find, partial, map, cloneDeep, omit, debounce } from './object';
2
2
 
3
3
  describe('Lodash functions', () => {
4
4
  describe('isEmpty', () => {
@@ -234,4 +234,152 @@ 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
+ });
237
385
  });
@@ -1,6 +1,7 @@
1
1
  /* 🛠️ Any commonly used lodash functions can be added here. 🤙 */
2
2
 
3
- export const isEmpty = (obj: any) => [Object, Array].includes((obj || {}).constructor) && !Object.entries((obj || {})).length;
3
+ export const isEmpty = (obj: any) =>
4
+ [Object, Array].includes((obj || {}).constructor) && !Object.entries((obj || {})).length
4
5
 
5
6
  export const get = <T, R = any>(obj: T, path: string, defaultValue?: R): R | any => {
6
7
  const travel = (regexp: RegExp): any =>
@@ -24,94 +25,175 @@ export const omitBy = (obj: Record<string, any>, predicate: (value: any, key: st
24
25
  if (obj === null || typeof obj !== 'object') return {}
25
26
  return Object.keys(obj).reduce((result: Record<string, any>, key: string) => {
26
27
  if (!predicate(obj[key], key)) {
27
- result[key] = obj[key];
28
+ result[key] = obj[key]
28
29
  }
29
- return result;
30
+ return result
30
31
  }, {})
31
32
  }
32
33
 
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
+
33
56
  export const noop = (): void => {
34
57
  // empty
35
- };
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
+ }
36
81
 
37
82
  export function merge(
38
83
  ...objects: Array<Record<string, unknown> | null | undefined>
39
84
  ): Record<string, unknown> {
40
85
  const isPlainObject = (obj: unknown): obj is Record<string, unknown> =>
41
- !!obj && typeof obj === 'object' && !Array.isArray(obj);
86
+ !!obj && typeof obj === 'object' && !Array.isArray(obj)
42
87
 
43
- const result: Record<string, unknown> = {};
88
+ const result: Record<string, unknown> = {}
44
89
 
45
90
  for (const obj of objects) {
46
- if (!obj || typeof obj !== 'object') continue;
91
+ if (!obj || typeof obj !== 'object') continue
47
92
 
48
93
  for (const key of Object.keys(obj)) {
49
- const oldVal = result[key];
50
- const newVal = (obj as Record<string, unknown>)[key];
94
+ const oldVal = result[key]
95
+ const newVal = (obj as Record<string, unknown>)[key]
51
96
 
52
97
  if (Array.isArray(oldVal) && Array.isArray(newVal)) {
53
- result[key] = newVal;
98
+ result[key] = newVal
54
99
  } else if (isPlainObject(oldVal) && isPlainObject(newVal)) {
55
- result[key] = merge(oldVal, newVal);
100
+ result[key] = merge(oldVal, newVal)
56
101
  } else if (Array.isArray(oldVal) && isPlainObject(newVal)) {
57
- result[key] = oldVal;
102
+ result[key] = oldVal
58
103
  } else if (isPlainObject(oldVal) && Array.isArray(newVal)) {
59
- result[key] = oldVal;
104
+ result[key] = oldVal
60
105
  } else {
61
- result[key] = newVal;
106
+ result[key] = newVal
62
107
  }
63
108
  }
64
109
  }
65
- return result;
110
+ return result
66
111
  }
67
112
 
68
113
  const createIteratee = (predicate: any) => {
69
114
  if (typeof predicate === 'function') {
70
- return predicate;
115
+ return predicate
71
116
  }
72
117
  if (typeof predicate === 'string') {
73
- return (obj: any) => obj[predicate];
118
+ return (obj: any) => obj[predicate]
74
119
  }
75
120
  if (Array.isArray(predicate)) {
76
- const [key, value] = predicate;
77
- return (obj: any) => obj[key] === value;
121
+ const [key, value] = predicate
122
+ return (obj: any) => obj[key] === value
78
123
  }
79
124
  if (typeof predicate === 'object' && predicate !== null) {
80
125
  return (obj: any) => {
81
126
  for (const key in predicate) {
82
127
  if (Object.prototype.hasOwnProperty.call(predicate, key)) {
83
- if (obj[key] !== predicate[key]) return false;
128
+ if (obj[key] !== predicate[key]) return false
84
129
  }
85
130
  }
86
- return true;
87
- };
131
+ return true
132
+ }
88
133
  }
89
- return () => false;
90
- };
134
+ return () => false
135
+ }
91
136
 
92
137
  export const filter = <T>(array: T[], predicate: any): T[] => {
93
- const iteratee = createIteratee(predicate);
94
- return array.filter(iteratee);
95
- };
138
+ const iteratee = createIteratee(predicate)
139
+ return array.filter(iteratee)
140
+ }
96
141
 
97
142
  export const find = <T>(array: T[], predicate: any): T | undefined => {
98
- const iteratee = createIteratee(predicate);
99
- return array.find(iteratee);
100
- };
143
+ const iteratee = createIteratee(predicate)
144
+ return array.find(iteratee)
145
+ }
101
146
 
102
- export const partial = <F extends (...args: unknown[]) => unknown>(
147
+ export const partial = <F extends (...args: any[]) => any>(
103
148
  fn: F,
104
- ...partials: unknown[]
105
- ): ((...args: unknown[]) => ReturnType<F>) => {
106
- const placeholder = partial.placeholder;
107
- return (...args: unknown[]): ReturnType<F> => {
108
- let argIndex = 0;
149
+ ...partials: any[]
150
+ ): ((...args: any[]) => ReturnType<F>) => {
151
+ const placeholder = partial.placeholder
152
+ return (...args: any[]): ReturnType<F> => {
153
+ let argIndex = 0
109
154
  const finalArgs = partials.map(arg =>
110
155
  arg === placeholder ? args[argIndex++] : arg
111
- );
112
- return fn(...(finalArgs.concat(args.slice(argIndex)) as Parameters<F>)) as ReturnType<F>;
113
- };
114
- };
156
+ )
157
+ return fn(...(finalArgs.concat(args.slice(argIndex)) as Parameters<F>)) as ReturnType<F>
158
+ }
159
+ }
115
160
 
116
- partial.placeholder = Symbol();
117
- export const _ = partial.placeholder;
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);
198
+ };
199
+ }