mixin.js 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. /*
  2. Copyright (c) 2004-2012, The Dojo Foundation All Rights Reserved.
  3. Available via Academic Free License >= 2.1 OR the modified BSD license.
  4. see: http://dojotoolkit.org/license for details
  5. */
  6. if(!dojo._hasResource["dojox.lang.oo.mixin"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
  7. dojo._hasResource["dojox.lang.oo.mixin"] = true;
  8. dojo.provide("dojox.lang.oo.mixin");
  9. dojo.experimental("dojox.lang.oo.mixin");
  10. dojo.require("dojox.lang.oo.Filter");
  11. dojo.require("dojox.lang.oo.Decorator");
  12. (function(){
  13. var oo = dojox.lang.oo, Filter = oo.Filter, Decorator = oo.Decorator, empty = {},
  14. defaultFilter = function(name){ return name; },
  15. defaultDecorator = function(name, newValue, oldValue){ return newValue; },
  16. defaultMixer = function(target, name, newValue, oldValue){ target[name] = newValue; },
  17. defaults = {}, // for the internal use in the mixin()
  18. extraNames = dojo._extraNames, extraLen = extraNames.length,
  19. applyDecorator = oo.applyDecorator = function(decorator, name, newValue, oldValue){
  20. // summary:
  21. // applies a decorator unraveling all embedded decorators
  22. // decorator: Function:
  23. // top-level decorator to apply
  24. // name: String:
  25. // name of the property
  26. // newValue: Object:
  27. // new value of the property
  28. // oldValue: Object:
  29. // old value of the property
  30. // returns: Object:
  31. // returns the final value of the property
  32. if(newValue instanceof Decorator){
  33. var d = newValue.decorator;
  34. newValue = applyDecorator(decorator, name, newValue.value, oldValue);
  35. return d(name, newValue, oldValue);
  36. }
  37. return decorator(name, newValue, oldValue);
  38. };
  39. /*=====
  40. dojox.lang.oo.__MixinDefaults = function(){
  41. // summary:
  42. // a dict of default parameters for dojox.lang.oo._mixin
  43. // decorator: Function:
  44. // a decorator function to be used in absence of other decorators
  45. // filter: Function:
  46. // a filter function to be used in absence of other filters
  47. // mixer: Function:
  48. // a mixer function to be used to mix in new properties
  49. this.decorator = decorator;
  50. this.filter = filter;
  51. this.mixer = mixer;
  52. };
  53. =====*/
  54. oo.__mixin = function(target, source, decorator, filter, mixer){
  55. // summary:
  56. // mixes in two objects processing decorators and filters
  57. // target: Object:
  58. // target to receive new/updated properties
  59. // source: Object:
  60. // source of properties
  61. // defaults: dojox.lang.oo.__MixinDefaults?:
  62. // default functions for various aspects of mixing
  63. // returns: Object:
  64. // target
  65. var name, targetName, prop, newValue, oldValue, i;
  66. // start mixing in properties
  67. for(name in source){
  68. prop = source[name];
  69. if(!(name in empty) || empty[name] !== prop){
  70. targetName = filter(name, target, source, prop);
  71. if(targetName && (!(targetName in target) || !(targetName in empty) || empty[targetName] !== prop)){
  72. // name is accepted
  73. oldValue = target[targetName];
  74. newValue = applyDecorator(decorator, targetName, prop, oldValue);
  75. if(oldValue !== newValue){
  76. mixer(target, targetName, newValue, oldValue);
  77. }
  78. }
  79. }
  80. }
  81. if(extraLen){
  82. for(i = 0; i < extraLen; ++i){
  83. name = extraNames[i];
  84. // repeating the body above
  85. prop = source[name];
  86. if(!(name in empty) || empty[name] !== prop){
  87. targetName = filter(name, target, source, prop);
  88. if(targetName && (!(targetName in target) || !(targetName in empty) || empty[targetName] !== prop)){
  89. // name is accepted
  90. oldValue = target[targetName];
  91. newValue = applyDecorator(decorator, targetName, prop, oldValue);
  92. if(oldValue !== newValue){
  93. mixer(target, targetName, newValue, oldValue);
  94. }
  95. }
  96. }
  97. }
  98. }
  99. return target; // Object
  100. };
  101. oo.mixin = function(target, source){
  102. // summary:
  103. // mixes in two or more objects processing decorators and filters
  104. // using defaults as a fallback
  105. // target: Object:
  106. // target to receive new/updated properties
  107. // source: Object...:
  108. // source of properties, more than one source is allowed
  109. // returns: Object:
  110. // target
  111. var decorator, filter, i = 1, l = arguments.length;
  112. for(; i < l; ++i){
  113. source = arguments[i];
  114. if(source instanceof Filter){
  115. filter = source.filter;
  116. source = source.bag;
  117. }else{
  118. filter = defaultFilter;
  119. }
  120. if(source instanceof Decorator){
  121. decorator = source.decorator;
  122. source = source.value;
  123. }else{
  124. decorator = defaultDecorator;
  125. }
  126. oo.__mixin(target, source, decorator, filter, defaultMixer);
  127. }
  128. return target; // Object
  129. };
  130. })();
  131. }