1. 1 : /**
  2. 2 : * @file obj.js
  3. 3 : * @module obj
  4. 4 : */
  5. 5 :
  6. 6 : /**
  7. 7 : * @callback obj:EachCallback
  8. 8 : *
  9. 9 : * @param {*} value
  10. 10 : * The current key for the object that is being iterated over.
  11. 11 : *
  12. 12 : * @param {string} key
  13. 13 : * The current key-value for object that is being iterated over
  14. 14 : */
  15. 15 :
  16. 16 : /**
  17. 17 : * @callback obj:ReduceCallback
  18. 18 : *
  19. 19 : * @param {*} accum
  20. 20 : * The value that is accumulating over the reduce loop.
  21. 21 : *
  22. 22 : * @param {*} value
  23. 23 : * The current key for the object that is being iterated over.
  24. 24 : *
  25. 25 : * @param {string} key
  26. 26 : * The current key-value for object that is being iterated over
  27. 27 : *
  28. 28 : * @return {*}
  29. 29 : * The new accumulated value.
  30. 30 : */
  31. 31 : const toString = Object.prototype.toString;
  32. 32 :
  33. 33 : /**
  34. 34 : * Get the keys of an Object
  35. 35 : *
  36. 36 : * @param {Object}
  37. 37 : * The Object to get the keys from
  38. 38 : *
  39. 39 : * @return {string[]}
  40. 40 : * An array of the keys from the object. Returns an empty array if the
  41. 41 : * object passed in was invalid or had no keys.
  42. 42 : *
  43. 43 : * @private
  44. 44 : */
  45. 45 : const keys = function(object) {
  46. 46 : return isObject(object) ? Object.keys(object) : [];
  47. 47 : };
  48. 48 :
  49. 49 : /**
  50. 50 : * Array-like iteration for objects.
  51. 51 : *
  52. 52 : * @param {Object} object
  53. 53 : * The object to iterate over
  54. 54 : *
  55. 55 : * @param {obj:EachCallback} fn
  56. 56 : * The callback function which is called for each key in the object.
  57. 57 : */
  58. 58 : export function each(object, fn) {
  59. 59 : keys(object).forEach(key => fn(object[key], key));
  60. 60 : }
  61. 61 :
  62. 62 : /**
  63. 63 : * Array-like reduce for objects.
  64. 64 : *
  65. 65 : * @param {Object} object
  66. 66 : * The Object that you want to reduce.
  67. 67 : *
  68. 68 : * @param {Function} fn
  69. 69 : * A callback function which is called for each key in the object. It
  70. 70 : * receives the accumulated value and the per-iteration value and key
  71. 71 : * as arguments.
  72. 72 : *
  73. 73 : * @param {*} [initial = 0]
  74. 74 : * Starting value
  75. 75 : *
  76. 76 : * @return {*}
  77. 77 : * The final accumulated value.
  78. 78 : */
  79. 79 : export function reduce(object, fn, initial = 0) {
  80. 80 : return keys(object).reduce((accum, key) => fn(accum, object[key], key), initial);
  81. 81 : }
  82. 82 :
  83. 83 : /**
  84. 84 : * Returns whether a value is an object of any kind - including DOM nodes,
  85. 85 : * arrays, regular expressions, etc. Not functions, though.
  86. 86 : *
  87. 87 : * This avoids the gotcha where using `typeof` on a `null` value
  88. 88 : * results in `'object'`.
  89. 89 : *
  90. 90 : * @param {Object} value
  91. 91 : * @return {boolean}
  92. 92 : */
  93. 93 : export function isObject(value) {
  94. 94 : return !!value && typeof value === 'object';
  95. 95 : }
  96. 96 :
  97. 97 : /**
  98. 98 : * Returns whether an object appears to be a "plain" object - that is, a
  99. 99 : * direct instance of `Object`.
  100. 100 : *
  101. 101 : * @param {Object} value
  102. 102 : * @return {boolean}
  103. 103 : */
  104. 104 : export function isPlain(value) {
  105. 105 : return isObject(value) &&
  106. 106 : toString.call(value) === '[object Object]' &&
  107. 107 : value.constructor === Object;
  108. 108 : }
  109. 109 :
  110. 110 : /**
  111. 111 : * Merge two objects recursively.
  112. 112 : *
  113. 113 : * Performs a deep merge like
  114. 114 : * {@link https://lodash.com/docs/4.17.10#merge|lodash.merge}, but only merges
  115. 115 : * plain objects (not arrays, elements, or anything else).
  116. 116 : *
  117. 117 : * Non-plain object values will be copied directly from the right-most
  118. 118 : * argument.
  119. 119 : *
  120. 120 : * @param {Object[]} sources
  121. 121 : * One or more objects to merge into a new object.
  122. 122 : *
  123. 123 : * @return {Object}
  124. 124 : * A new object that is the merged result of all sources.
  125. 125 : */
  126. 126 : export function merge(...sources) {
  127. 127 : const result = {};
  128. 128 :
  129. 129 : sources.forEach(source => {
  130. 130 : if (!source) {
  131. 131 : return;
  132. 132 : }
  133. 133 :
  134. 134 : each(source, (value, key) => {
  135. 135 : if (!isPlain(value)) {
  136. 136 : result[key] = value;
  137. 137 : return;
  138. 138 : }
  139. 139 :
  140. 140 : if (!isPlain(result[key])) {
  141. 141 : result[key] = {};
  142. 142 : }
  143. 143 :
  144. 144 : result[key] = merge(result[key], value);
  145. 145 : });
  146. 146 : });
  147. 147 :
  148. 148 : return result;
  149. 149 : }
  150. 150 :
  151. 151 : /**
  152. 152 : * Object.defineProperty but "lazy", which means that the value is only set after
  153. 153 : * it is retrieved the first time, rather than being set right away.
  154. 154 : *
  155. 155 : * @param {Object} obj the object to set the property on
  156. 156 : * @param {string} key the key for the property to set
  157. 157 : * @param {Function} getValue the function used to get the value when it is needed.
  158. 158 : * @param {boolean} setter whether a setter should be allowed or not
  159. 159 : */
  160. 160 : export function defineLazyProperty(obj, key, getValue, setter = true) {
  161. 161 : const set = (value) =>
  162. 162 : Object.defineProperty(obj, key, {value, enumerable: true, writable: true});
  163. 163 :
  164. 164 : const options = {
  165. 165 : configurable: true,
  166. 166 : enumerable: true,
  167. 167 : get() {
  168. 168 : const value = getValue();
  169. 169 :
  170. 170 : set(value);
  171. 171 :
  172. 172 : return value;
  173. 173 : }
  174. 174 : };
  175. 175 :
  176. 176 : if (setter) {
  177. 177 : options.set = set;
  178. 178 : }
  179. 179 :
  180. 180 : return Object.defineProperty(obj, key, options);
  181. 181 : }