mixin.js 2.2 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374
  1. import arrayEach from './_arrayEach.js';
  2. import arrayPush from './_arrayPush.js';
  3. import baseFunctions from './_baseFunctions.js';
  4. import copyArray from './_copyArray.js';
  5. import isFunction from './isFunction.js';
  6. import isObject from './isObject.js';
  7. import keys from './keys.js';
  8. /**
  9. * Adds all own enumerable string keyed function properties of a source
  10. * object to the destination object. If `object` is a function, then methods
  11. * are added to its prototype as well.
  12. *
  13. * **Note:** Use `_.runInContext` to create a pristine `lodash` function to
  14. * avoid conflicts caused by modifying the original.
  15. *
  16. * @static
  17. * @since 0.1.0
  18. * @memberOf _
  19. * @category Util
  20. * @param {Function|Object} [object=lodash] The destination object.
  21. * @param {Object} source The object of functions to add.
  22. * @param {Object} [options={}] The options object.
  23. * @param {boolean} [options.chain=true] Specify whether mixins are chainable.
  24. * @returns {Function|Object} Returns `object`.
  25. * @example
  26. *
  27. * function vowels(string) {
  28. * return _.filter(string, function(v) {
  29. * return /[aeiou]/i.test(v);
  30. * });
  31. * }
  32. *
  33. * _.mixin({ 'vowels': vowels });
  34. * _.vowels('fred');
  35. * // => ['e']
  36. *
  37. * _('fred').vowels().value();
  38. * // => ['e']
  39. *
  40. * _.mixin({ 'vowels': vowels }, { 'chain': false });
  41. * _('fred').vowels();
  42. * // => ['e']
  43. */
  44. function mixin(object, source, options) {
  45. var props = keys(source),
  46. methodNames = baseFunctions(source, props);
  47. var chain = !(isObject(options) && 'chain' in options) || !!options.chain,
  48. isFunc = isFunction(object);
  49. arrayEach(methodNames, function(methodName) {
  50. var func = source[methodName];
  51. object[methodName] = func;
  52. if (isFunc) {
  53. object.prototype[methodName] = function() {
  54. var chainAll = this.__chain__;
  55. if (chain || chainAll) {
  56. var result = object(this.__wrapped__),
  57. actions = result.__actions__ = copyArray(this.__actions__);
  58. actions.push({ 'func': func, 'args': arguments, 'thisArg': object });
  59. result.__chain__ = chainAll;
  60. return result;
  61. }
  62. return func.apply(object, arrayPush([this.value()], arguments));
  63. };
  64. }
  65. });
  66. return object;
  67. }
  68. export default mixin;