mixin.js 4.3 KB

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