sproutcore 1.6.0.rc.2-x86-mingw32 → 1.6.0.1-x86-mingw32

Sign up to get free protection for your applications and to get access to all the features.
Files changed (209) hide show
  1. data/CHANGELOG +12 -0
  2. data/VERSION.yml +1 -1
  3. data/bin/sc-docs +6 -1
  4. data/lib/buildtasks/target.rake +1 -1
  5. data/lib/frameworks/sproutcore/Buildfile +5 -1
  6. data/lib/frameworks/sproutcore/CHANGELOG.md +175 -1
  7. data/lib/frameworks/sproutcore/apps/test_controls/controllers/select.js +12 -0
  8. data/lib/frameworks/sproutcore/apps/test_controls/resources/select_page.js +19 -5
  9. data/lib/frameworks/sproutcore/frameworks/ajax/system/request.js +28 -31
  10. data/lib/frameworks/sproutcore/frameworks/ajax/system/response.js +9 -2
  11. data/lib/frameworks/sproutcore/frameworks/core_foundation/controllers/controller.js +21 -1
  12. data/lib/frameworks/sproutcore/frameworks/core_foundation/mixins/responder_context.js +1 -1
  13. data/lib/frameworks/sproutcore/frameworks/core_foundation/mixins/template_helpers/checkbox_support.js +6 -1
  14. data/lib/frameworks/sproutcore/frameworks/core_foundation/mixins/template_helpers/text_field_support.js +26 -8
  15. data/lib/frameworks/sproutcore/frameworks/core_foundation/panes/keyboard.js +2 -0
  16. data/lib/frameworks/sproutcore/frameworks/core_foundation/panes/pane.js +12 -5
  17. data/lib/frameworks/sproutcore/frameworks/core_foundation/panes/template.js +25 -9
  18. data/lib/frameworks/sproutcore/frameworks/core_foundation/system/locale.js +157 -5
  19. data/lib/frameworks/sproutcore/frameworks/core_foundation/system/render_context.js +7 -6
  20. data/lib/frameworks/sproutcore/frameworks/core_foundation/system/root_responder.js +9 -3
  21. data/lib/frameworks/sproutcore/frameworks/core_foundation/system/sparse_array.js +8 -8
  22. data/lib/frameworks/sproutcore/frameworks/core_foundation/system/string.js +104 -4
  23. data/lib/frameworks/sproutcore/frameworks/core_foundation/system/theme.js +3 -56
  24. data/lib/frameworks/sproutcore/frameworks/core_foundation/system/utils.js +4 -2
  25. data/lib/frameworks/sproutcore/frameworks/core_foundation/tests/controllers/object/content_destroyed.js +59 -0
  26. data/lib/frameworks/sproutcore/frameworks/core_foundation/tests/mixins/string.js +41 -1
  27. data/lib/frameworks/sproutcore/frameworks/core_foundation/tests/mixins/template_helpers/text_field_support.js +10 -2
  28. data/lib/frameworks/sproutcore/frameworks/core_foundation/tests/panes/template.js +16 -1
  29. data/lib/frameworks/sproutcore/frameworks/core_foundation/tests/views/template/handlebars.js +1 -1
  30. data/lib/frameworks/sproutcore/frameworks/core_foundation/tests/views/view/clippingFrame.js +11 -0
  31. data/lib/frameworks/sproutcore/frameworks/core_foundation/tests/views/view/didAppendToDocument.js +18 -2
  32. data/lib/frameworks/sproutcore/frameworks/core_foundation/tests/views/view/insertBefore.js +10 -6
  33. data/lib/frameworks/sproutcore/frameworks/core_foundation/tests/views/view/keyboard.js +18 -1
  34. data/lib/frameworks/sproutcore/frameworks/core_foundation/views/template_collection.js +9 -1
  35. data/lib/frameworks/sproutcore/frameworks/core_foundation/views/view.js +9 -4
  36. data/lib/frameworks/sproutcore/frameworks/core_foundation/views/view/keyboard.js +15 -3
  37. data/lib/frameworks/sproutcore/frameworks/core_foundation/views/view/manipulation.js +14 -8
  38. data/lib/frameworks/sproutcore/frameworks/core_foundation/views/view/theming.js +8 -18
  39. data/lib/frameworks/sproutcore/frameworks/datastore/data_sources/fixtures.js +12 -2
  40. data/lib/frameworks/sproutcore/frameworks/datastore/mixins/relationship_support.js +296 -0
  41. data/lib/frameworks/sproutcore/frameworks/datastore/models/child_record.js +1 -1
  42. data/lib/frameworks/sproutcore/frameworks/datastore/models/record.js +330 -326
  43. data/lib/frameworks/sproutcore/frameworks/datastore/models/record_attribute.js +22 -1
  44. data/lib/frameworks/sproutcore/frameworks/datastore/system/nested_store.js +1 -1
  45. data/lib/frameworks/sproutcore/frameworks/datastore/system/store.js +614 -614
  46. data/lib/frameworks/sproutcore/frameworks/datastore/tests/data_sources/data_source.js +14 -1
  47. data/lib/frameworks/sproutcore/frameworks/datastore/tests/models/nested_records/nested_record.js +3 -1
  48. data/lib/frameworks/sproutcore/frameworks/datastore/tests/models/nested_records/nested_record_array_complex.js +2 -0
  49. data/lib/frameworks/sproutcore/frameworks/datastore/tests/models/nested_records/nested_record_complex.js +2 -0
  50. data/lib/frameworks/sproutcore/frameworks/datastore/tests/models/record/core_methods.js +20 -13
  51. data/lib/frameworks/sproutcore/frameworks/datastore/tests/models/record_attribute.js +61 -46
  52. data/lib/frameworks/sproutcore/frameworks/datastore/tests/system/nested_store/commitChangesFromNestedStore.js +30 -30
  53. data/lib/frameworks/sproutcore/frameworks/datastore/tests/system/store/commitChangesFromNestedStore.js +24 -24
  54. data/lib/frameworks/sproutcore/frameworks/datastore/tests/system/store/connectDataSource.js +31 -0
  55. data/lib/frameworks/sproutcore/frameworks/datastore/tests/system/store/pushRelationships.js +1177 -0
  56. data/lib/frameworks/sproutcore/frameworks/datetime/frameworks/localized/system/datetime.js +4 -63
  57. data/lib/frameworks/sproutcore/frameworks/desktop/mixins/border.js +1 -1
  58. data/lib/frameworks/sproutcore/frameworks/desktop/mixins/scrollable.js +1 -1
  59. data/lib/frameworks/sproutcore/frameworks/desktop/panes/alert.js +7 -8
  60. data/lib/frameworks/sproutcore/frameworks/desktop/panes/menu.js +18 -1
  61. data/lib/frameworks/sproutcore/frameworks/desktop/panes/panel.js +9 -0
  62. data/lib/frameworks/sproutcore/frameworks/desktop/protocols/drag_data_source.js +3 -3
  63. data/lib/frameworks/sproutcore/frameworks/desktop/protocols/drop_target.js +3 -3
  64. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/button.js +1 -1
  65. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/checkbox.js +1 -1
  66. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/collection.js +1 -1
  67. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/disclosure.js +1 -1
  68. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/image_button.js +1 -1
  69. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/master_detail.js +3 -2
  70. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/menu.js +12 -2
  71. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/panel.js +1 -1
  72. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/picker.js +1 -1
  73. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/progress.js +1 -1
  74. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/radio.js +1 -1
  75. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/radio_group.js +2 -2
  76. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/segment.js +1 -1
  77. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/segmented.js +1 -1
  78. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/slider.js +1 -1
  79. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/toolbar.js +1 -1
  80. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/well.js +1 -1
  81. data/lib/frameworks/sproutcore/frameworks/desktop/render_delegates/workspace.js +1 -1
  82. data/lib/frameworks/sproutcore/frameworks/desktop/resources/segmented.css +1 -0
  83. data/lib/frameworks/sproutcore/frameworks/desktop/tests/panes/alert/ui.js +33 -22
  84. data/lib/frameworks/sproutcore/frameworks/desktop/tests/panes/panel/methods.js +20 -1
  85. data/lib/frameworks/sproutcore/frameworks/desktop/tests/views/checkbox/methods.js +10 -3
  86. data/lib/frameworks/sproutcore/frameworks/desktop/tests/views/date_field/methods.js +34 -7
  87. data/lib/frameworks/sproutcore/frameworks/desktop/views/button.js +14 -15
  88. data/lib/frameworks/sproutcore/frameworks/desktop/views/checkbox.js +40 -14
  89. data/lib/frameworks/sproutcore/frameworks/desktop/views/collection.js +699 -700
  90. data/lib/frameworks/sproutcore/frameworks/desktop/views/list_item.js +2 -2
  91. data/lib/frameworks/sproutcore/frameworks/desktop/views/master_detail.js +11 -1
  92. data/lib/frameworks/sproutcore/frameworks/desktop/views/menu_item.js +16 -6
  93. data/lib/frameworks/sproutcore/frameworks/desktop/views/progress.js +0 -1
  94. data/lib/frameworks/sproutcore/frameworks/desktop/views/radio.js +49 -7
  95. data/lib/frameworks/sproutcore/frameworks/desktop/views/select_button.js +9 -0
  96. data/lib/frameworks/sproutcore/frameworks/desktop/views/select_field.js +6 -2
  97. data/lib/frameworks/sproutcore/frameworks/desktop/views/slider.js +4 -26
  98. data/lib/frameworks/sproutcore/frameworks/desktop/views/web.js +20 -19
  99. data/lib/frameworks/sproutcore/frameworks/experimental/Buildfile +2 -0
  100. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/designer/designers/view_designer.js +249 -249
  101. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/mixins/edit_mode.js +13 -5
  102. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/mixins/emptiness.js +53 -37
  103. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/render_delegates/form.js +2 -1
  104. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/render_delegates/form_row.js +3 -11
  105. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/tests/mixins/edit_mode.js +53 -0
  106. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/tests/mixins/emptiness.js +114 -0
  107. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/tests/views/form.js +174 -6
  108. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/tests/views/form_row.js +86 -6
  109. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/views/form.js +80 -110
  110. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/views/form_row.js +96 -97
  111. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/polymorphism/README.md +2 -1
  112. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/polymorphism/models/record.js +20 -36
  113. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/select_view/ext/menu.js +121 -0
  114. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/select_view/ext/menu_item.js +90 -0
  115. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/select_view/mixins/select_view_menu.js +139 -0
  116. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/select_view/render_delegates/select_button.js +14 -0
  117. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/select_view/tests/ext/menu_resizing.js +25 -0
  118. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/select_view/tests/mixins/select_view_menu/bindings.js +43 -0
  119. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/select_view/tests/mixins/select_view_menu/check_selected.js +32 -0
  120. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/select_view/tests/views/popup_button/menu_setup.js +40 -0
  121. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/select_view/tests/views/popup_button/show_menu.js +45 -0
  122. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/select_view/tests/views/select/menu_width.js +49 -0
  123. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/select_view/tests/views/select/selected_item.js +191 -0
  124. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/select_view/views/popup_button.js +264 -0
  125. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/select_view/views/select.js +450 -0
  126. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/split_view/mixins/split_child.js +14 -6
  127. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/split_view/mixins/split_thumb.js +1 -1
  128. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/split_view/render_delegates/split.js +1 -1
  129. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/split_view/render_delegates/split_divider.js +1 -1
  130. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/split_view/views/split.js +9 -0
  131. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/split_view/views/thumb.js +3 -2
  132. data/lib/frameworks/sproutcore/frameworks/foundation/mixins/auto_resize.js +7 -17
  133. data/lib/frameworks/sproutcore/frameworks/foundation/mixins/content_value_support.js +1 -1
  134. data/lib/frameworks/sproutcore/frameworks/foundation/mixins/flowed_layout.js +35 -8
  135. data/lib/frameworks/sproutcore/frameworks/foundation/mixins/inline_editable.js +1 -1
  136. data/lib/frameworks/sproutcore/frameworks/foundation/mixins/inline_editor.js +1 -1
  137. data/lib/frameworks/sproutcore/frameworks/foundation/mixins/inline_editor_delegate.js +1 -1
  138. data/lib/frameworks/sproutcore/frameworks/foundation/mixins/inner_frame.js +1 -1
  139. data/lib/frameworks/sproutcore/frameworks/foundation/render_delegates/canvas_image.js +1 -1
  140. data/lib/frameworks/sproutcore/frameworks/foundation/render_delegates/helpers/sizing.js +2 -0
  141. data/lib/frameworks/sproutcore/frameworks/foundation/render_delegates/image.js +1 -1
  142. data/lib/frameworks/sproutcore/frameworks/foundation/render_delegates/label.js +1 -1
  143. data/lib/frameworks/sproutcore/frameworks/foundation/render_delegates/render_delegate.js +6 -6
  144. data/lib/frameworks/sproutcore/frameworks/foundation/resources/images/favicon.ico +0 -0
  145. data/lib/frameworks/sproutcore/frameworks/foundation/resources/text_field.css +0 -5
  146. data/lib/frameworks/sproutcore/frameworks/foundation/system/exception_handler.js +4 -2
  147. data/lib/frameworks/sproutcore/frameworks/foundation/system/math.js +2 -1
  148. data/lib/frameworks/sproutcore/frameworks/foundation/system/module.js +13 -0
  149. data/lib/frameworks/sproutcore/frameworks/foundation/system/utils/string_measurement.js +6 -9
  150. data/lib/frameworks/sproutcore/frameworks/foundation/tests/mixins/flowed_layout/tests.js +912 -0
  151. data/lib/frameworks/sproutcore/frameworks/foundation/tests/views/text_field/methods.js +36 -7
  152. data/lib/frameworks/sproutcore/frameworks/foundation/tests/views/text_field/ui.js +58 -4
  153. data/lib/frameworks/sproutcore/frameworks/foundation/validators/validator.js +1 -3
  154. data/lib/frameworks/sproutcore/frameworks/foundation/views/field.js +0 -15
  155. data/lib/frameworks/sproutcore/frameworks/foundation/views/label.js +2 -2
  156. data/lib/frameworks/sproutcore/frameworks/foundation/views/text_field.js +25 -14
  157. data/lib/frameworks/sproutcore/frameworks/handlebars/handlebars.js +1 -1
  158. data/lib/frameworks/sproutcore/frameworks/runtime/core.js +15 -9
  159. data/lib/frameworks/sproutcore/frameworks/runtime/debug/test_suites/array/flatten.js +24 -0
  160. data/lib/frameworks/sproutcore/frameworks/runtime/ext/array.js +2 -1
  161. data/lib/frameworks/sproutcore/frameworks/runtime/ext/function.js +5 -5
  162. data/lib/frameworks/sproutcore/frameworks/runtime/mixins/array.js +19 -0
  163. data/lib/frameworks/sproutcore/frameworks/runtime/mixins/copyable.js +3 -2
  164. data/lib/frameworks/sproutcore/frameworks/runtime/mixins/freezable.js +1 -1
  165. data/lib/frameworks/sproutcore/frameworks/runtime/mixins/observable.js +1 -1
  166. data/lib/frameworks/sproutcore/frameworks/runtime/system/binding.js +14 -14
  167. data/lib/frameworks/sproutcore/frameworks/runtime/system/error.js +3 -0
  168. data/lib/frameworks/sproutcore/frameworks/runtime/system/logger.js +2 -2
  169. data/lib/frameworks/sproutcore/frameworks/runtime/system/object.js +2 -2
  170. data/lib/frameworks/sproutcore/frameworks/runtime/system/range_observer.js +1 -1
  171. data/lib/frameworks/sproutcore/frameworks/runtime/system/run_loop.js +3 -3
  172. data/lib/frameworks/sproutcore/frameworks/runtime/system/set.js +15 -16
  173. data/lib/frameworks/sproutcore/frameworks/runtime/tests/core/itemType.js +6 -2
  174. data/lib/frameworks/sproutcore/frameworks/runtime/tests/system/object/enhance.js +30 -0
  175. data/lib/frameworks/sproutcore/frameworks/runtime/tests/system/range_observer/create.js +17 -0
  176. data/lib/frameworks/sproutcore/frameworks/statechart/system/state.js +9 -2
  177. data/lib/frameworks/sproutcore/frameworks/statechart/system/statechart.js +3 -1
  178. data/lib/frameworks/sproutcore/frameworks/testing/resources/runner.css +0 -1
  179. data/lib/frameworks/sproutcore/frameworks/yuireset/resources/base.css +80 -0
  180. data/lib/frameworks/sproutcore/frameworks/yuireset/resources/core.css +0 -4
  181. data/lib/frameworks/sproutcore/lib/index.rhtml +2 -1
  182. data/lib/frameworks/sproutcore/themes/ace/resources/collection/normal/list.css +3 -3
  183. data/lib/frameworks/sproutcore/themes/ace/resources/collection/normal/list_item.css +2 -2
  184. data/lib/frameworks/sproutcore/themes/ace/resources/form/form.css +9 -0
  185. data/lib/frameworks/sproutcore/themes/ace/resources/menu/menu.css +3 -1
  186. data/lib/frameworks/sproutcore/themes/ace/resources/picker/popover/picker.js +1 -1
  187. data/lib/frameworks/sproutcore/themes/ace/resources/picker/popover/workspace.js +1 -1
  188. data/lib/frameworks/sproutcore/themes/legacy_theme/render_delegates/button.js +1 -1
  189. data/lib/frameworks/sproutcore/themes/legacy_theme/render_delegates/panel.js +1 -1
  190. data/lib/frameworks/sproutcore/themes/legacy_theme/render_delegates/progress.js +2 -0
  191. data/lib/frameworks/sproutcore/themes/legacy_theme/render_delegates/slider.js +1 -1
  192. data/lib/frameworks/sproutcore/themes/legacy_theme/render_delegates/well.js +1 -1
  193. data/lib/sproutcore/builders/base.rb +5 -1
  194. data/lib/sproutcore/builders/handlebars.rb +12 -1
  195. data/lib/sproutcore/models/target.rb +1 -9
  196. data/lib/sproutcore/rack/proxy.rb +238 -92
  197. data/lib/sproutcore/tools/docs.rb +1 -7
  198. data/spec/fixtures/builder_tests/apps/handlebars_test/Buildfile +1 -0
  199. data/spec/fixtures/builder_tests/apps/handlebars_test/{template.handlebars → templates/template.handlebars} +2 -0
  200. data/spec/lib/builders/handlebars_spec.rb +10 -4
  201. data/sproutcore.gemspec +3 -1
  202. metadata +73 -44
  203. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/english.lproj/default_styles.css +0 -5
  204. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/english.lproj/strings.js +0 -15
  205. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/tests/views/form_checkbox_field.js +0 -17
  206. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/tests/views/form_field.js +0 -17
  207. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/tests/views/form_label.js +0 -17
  208. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/tests/views/form_radio_field.js +0 -17
  209. data/lib/frameworks/sproutcore/frameworks/experimental/frameworks/forms/tests/views/form_text_field.js +0 -17
@@ -38,6 +38,10 @@
38
38
  equals(textFieldView.$('input').val(), "afterlife", "sets value of DOM to value property");
39
39
  });
40
40
 
41
+ // Not really sure how to test this without doing something like adding a selection then checking
42
+ // to see if it's still there after setting value
43
+ test("only update DOM if value changed");
44
+
41
45
  test("value binding works properly for inputs that haven't been created", function() {
42
46
  var view = SC.TemplateView.create(SC.TextFieldSupport, {
43
47
  template: SC.Handlebars.compile('<input type="text">'),
@@ -74,7 +78,7 @@
74
78
  equals(focusCalled, 1, "focus called after field receives focus");
75
79
 
76
80
  textFieldView.$('input').blur();
77
- equals(blurCalled, 1, "blur alled after field blurs");
81
+ equals(blurCalled, 1, "blur called after field blurs");
78
82
  });
79
83
 
80
84
  test("calls correct method for key events", function() {
@@ -130,6 +134,10 @@
130
134
  equals(textFieldView.$('input').val(), "afterlife", "sets value of DOM to value property");
131
135
  });
132
136
 
137
+ // Not really sure how to test this without doing something like adding a selection then checking
138
+ // to see if it's still there after setting value
139
+ test("only update DOM if value changed");
140
+
133
141
  test("value binding works properly for inputs that haven't been created", function() {
134
142
  var view = SC.TextField.create({
135
143
  valueBinding: 'TestObject.value'
@@ -165,7 +173,7 @@
165
173
  equals(focusCalled, 1, "focus called after field receives focus");
166
174
 
167
175
  textFieldView.$('input').blur();
168
- equals(blurCalled, 1, "blur alled after field blurs");
176
+ equals(blurCalled, 1, "blur called after field blurs");
169
177
  });
170
178
 
171
179
  test("calls correct method for key events", function() {
@@ -9,7 +9,6 @@ module("Template Panes");
9
9
  test("Template panes append a main pane to the document body", function() {
10
10
  var pane = SC.TemplatePane.append({
11
11
  layerId: 'template-panes-are-so-cool',
12
-
13
12
  template: SC.Handlebars.compile('<h1>foo bar baz</h1>')
14
13
  });
15
14
 
@@ -18,3 +17,19 @@ test("Template panes append a main pane to the document body", function() {
18
17
 
19
18
  pane.remove();
20
19
  });
20
+
21
+ test("Template panes should be awoken", function(){
22
+ var didAwake = false;
23
+
24
+ var originalAwake = SC.MainPane.prototype.awake; // null for now, but just in case
25
+ SC.MainPane.prototype.awake = function(){ didAwake = true; }
26
+
27
+ var pane = SC.TemplatePane.append({
28
+ layerId: 'template-panes-are-so-cool',
29
+ template: SC.Handlebars.compile('<h1>foo bar baz</h1>')
30
+ });
31
+
32
+ SC.MainPane.prototype.awake = originalAwake;
33
+
34
+ ok(didAwake, "should have awoken the pane");
35
+ });
@@ -898,7 +898,7 @@ test("should be able to bind class attribute with {{bindAttr}}", function() {
898
898
  });
899
899
 
900
900
  test("should be able to bind boolean element attributes using {{bindAttr}}", function() {
901
- var template = SC.Handlebars.compile('<input type="check" {{bindAttr disabled="content.isDisabled" checked="content.isChecked"}} />');
901
+ var template = SC.Handlebars.compile('<input type="checkbox" {{bindAttr disabled="content.isDisabled" checked="content.isChecked"}} />');
902
902
  var content = SC.Object.create({
903
903
  isDisabled: false,
904
904
  isChecked: true
@@ -115,3 +115,14 @@ test("notifies receiver and each child if parent clipping frame changes", functi
115
115
  // number.
116
116
  equals(callCount, 2, 'should invoke observer on child and nested child');
117
117
  });
118
+
119
+ test("returns 0, 0, W, H if parentView has no clippingFrame", function(){
120
+ a.clippingFrame = null;
121
+
122
+ var targetFrame = aa.get('clippingFrame');
123
+
124
+ equals(targetFrame.x, 0, "x should be 0");
125
+ equals(targetFrame.y, 0, "y should be 0");
126
+ equals(targetFrame.width, 40, "width should be 40");
127
+ equals(targetFrame.height, 40, "height should be 40");
128
+ });
@@ -21,6 +21,7 @@ module("SC.View#didAppendToDocument", {
21
21
  context.push('new string');
22
22
  },
23
23
  didAppendToDocument: function(){
24
+ ok(document.getElementById(this.get('layerId')), "view layer should exist");
24
25
  counter++;
25
26
  }
26
27
  })
@@ -30,6 +31,7 @@ module("SC.View#didAppendToDocument", {
30
31
 
31
32
  additionalView = SC.View.create({
32
33
  didAppendToDocument: function(){
34
+ ok(document.getElementById(this.get('layerId')), "additionalView layer should exist");
33
35
  counter++;
34
36
  }
35
37
  });
@@ -59,11 +61,12 @@ test("Check that didAppendToDocument gets called at the right moment", function(
59
61
  equals(counter, 3, "");
60
62
  });
61
63
 
64
+
62
65
  // Test for bug: when a childView has a non-fixed layout and we request its frame before the parentView has
63
66
  // a layer and the parentView uses static layout, then the frame returned will be {x: 0, y:0, width: 0, height: 0}
64
67
  // and any further requests for the childView's frame will not return a new value unless the parentViewDidChange
65
68
  // or parentViewDidResize. A weird case, but we prevent it from failing anyhow.
66
- test("Check that childView is updated if we have static layout and they don't have a fixed layout", function() {
69
+ test("Check that childView is updated if the pane has a static layout and view doesn't have a fixed layout", function() {
67
70
  var childFrame,
68
71
  wrongFrame = {x:0, y:0, width: 0, height: 0},
69
72
  correctFrame;
@@ -80,4 +83,17 @@ test("Check that childView is updated if we have static layout and they don't ha
80
83
  correctFrame = pane.get('frame');
81
84
 
82
85
  same(childFrame, correctFrame, 'getting frame after layer exists on non-fixed layout childView should return actual frame');
83
- });
86
+ });
87
+
88
+
89
+ test("Check that childView is updated if it has a static layout", function() {
90
+ var childFrame,
91
+ wrongFrame = {x:0, y:0, width: 0, height: 0},
92
+ correctFrame;
93
+
94
+ view.set('useStaticLayout', YES);
95
+
96
+ equals(counter, 0, "precond - has not been called yet");
97
+ pane.append(); // make sure there is a layer...
98
+ equals(counter, 1, "didAppendToDocument was called once");
99
+ });
@@ -80,7 +80,7 @@ test("invokes willAddChild() on receiver if defined before adding child" ,functi
80
80
 
81
81
 
82
82
  parent.insertBefore(child, otherChild);
83
- ok(callCount, 1, 'invoked');
83
+ equals(callCount, 1, 'invoked');
84
84
  });
85
85
 
86
86
  test("invokes willAddToParent() on child view if defined before adding child" ,function() {
@@ -102,7 +102,7 @@ test("invokes willAddToParent() on child view if defined before adding child" ,f
102
102
 
103
103
 
104
104
  parent.insertBefore(child, otherChild);
105
- ok(callCount, 1, 'invoked');
105
+ equals(callCount, 1, 'invoked');
106
106
  });
107
107
 
108
108
  test("invokes didAddChild() on receiver if defined after adding child" ,function() {
@@ -122,9 +122,11 @@ test("invokes didAddChild() on receiver if defined after adding child" ,function
122
122
  callCount++;
123
123
  };
124
124
 
125
-
125
+ SC.RunLoop.begin();
126
126
  parent.insertBefore(child, otherChild);
127
- ok(callCount, 1, 'invoked');
127
+ SC.RunLoop.end();
128
+
129
+ equals(callCount, 1, 'invoked');
128
130
  });
129
131
 
130
132
  test("invokes didAddToParent() on child view if defined after adding child" ,function() {
@@ -144,9 +146,11 @@ test("invokes didAddToParent() on child view if defined after adding child" ,fun
144
146
  callCount++;
145
147
  };
146
148
 
147
-
149
+ SC.RunLoop.begin();
148
150
  parent.insertBefore(child, otherChild);
149
- ok(callCount, 1, 'invoked');
151
+ SC.RunLoop.end();
152
+
153
+ equals(callCount, 1, 'invoked');
150
154
  });
151
155
 
152
156
  test("invokes parentViewDidChange() on child view. this is used by the view internals to update layer loc", function() {
@@ -4,7 +4,18 @@
4
4
  // ©2008-2011 Apple Inc. All rights reserved.
5
5
  // License: Licensed under MIT license (see license.js)
6
6
  // ==========================================================================
7
- module("SC.View - Keyboard support");
7
+
8
+ var originalTabbing;
9
+
10
+ module("SC.View - Keyboard support with Tabbing Only Inside Document", {
11
+ setup: function(){
12
+ originalTabbing = SC.TABBING_ONLY_INSIDE_DOCUMENT;
13
+ SC.TABBING_ONLY_INSIDE_DOCUMENT = YES;
14
+ },
15
+ teardown: function(){
16
+ SC.TABBING_ONLY_INSIDE_DOCUMENT = originalTabbing;
17
+ }
18
+ });
8
19
 
9
20
  test("Views only attempt to call performKeyEquivalent on child views that support it", function() {
10
21
  var performKeyEquivalentCalled = 0;
@@ -422,3 +433,9 @@ test("previousValidKeyView prioritizes parent's firstKeyView even if previousKey
422
433
  equals(pane.view2.view6.get('previousValidKeyView'), pane.view1.view4, "firstKeyView was respected; views before firstKeyView were skipped");
423
434
  });
424
435
 
436
+
437
+ module("SC.View - Keyboard support with Tabbing Outside of Document");
438
+
439
+ test("forward tab with no next responder moves out of document");
440
+
441
+ test("backwards tab with no previous responder moves out of document");
@@ -9,7 +9,15 @@ sc_require('views/template');
9
9
  */
10
10
  SC.TemplateCollectionView = SC.TemplateView.extend(
11
11
  /** @scope SC.TemplateCollectionView.prototype */{
12
+ /**
13
+ Name of the tag that is used for the collection
14
+
15
+ If the tag is a list ('ul' or 'ol') each item will be wrapped into a 'li' tag.
16
+ If the tag is a table ('table', 'thead', 'tbody') each item will be wrapped into a 'tr' tag.
12
17
 
18
+ @property {String}
19
+ @default ul
20
+ */
13
21
  tagName: 'ul',
14
22
  content: null,
15
23
  template: SC.Handlebars.compile(''),
@@ -297,7 +305,7 @@ SC.TemplateCollectionView = SC.TemplateView.extend(
297
305
  case 'thead':
298
306
  case 'tbody':
299
307
  case 'tfoot':
300
- return 'tr'
308
+ return 'tr';
301
309
  }
302
310
  }.property('tagName'),
303
311
 
@@ -24,7 +24,7 @@ SC.CONTEXT_MENU_ENABLED = YES;
24
24
  Default property to disable or enable if the focus can jump to the address
25
25
  bar or not.
26
26
  */
27
- SC.TABBING_ONLY_INSIDE_DOCUMENT = YES;
27
+ SC.TABBING_ONLY_INSIDE_DOCUMENT = NO;
28
28
 
29
29
  /**
30
30
  Tells the property (when fetched with themed()) to get its value from the renderer (if any).
@@ -674,7 +674,12 @@ SC.CoreView.reopen(
674
674
 
675
675
  if (this.get('isTextSelectable')) { context.addClass('allow-select'); }
676
676
  if (!this.get('isVisible')) { context.addClass('sc-hidden'); }
677
- if (this.get('isFirstResponder')) { context.addClass('focus'); }
677
+ if (this.get('isFirstResponder')) {
678
+ context.addClass('focus');
679
+ context.attr('tabindex', '0');
680
+ }else{
681
+ context.attr('tabindex', '-1');
682
+ }
678
683
 
679
684
  context.id(this.get('layerId'));
680
685
  context.attr('role', this.get('ariaRole'));
@@ -999,7 +1004,7 @@ SC.CoreView.reopen(
999
1004
  pv = this.get('parentView');
1000
1005
  if (pv) {
1001
1006
  cf = pv.get('clippingFrame');
1002
- if (!cf) return f;
1007
+ if (!cf) return { x: 0, y: 0, width: f.width, height: f.height};
1003
1008
  ret = SC.intersectRects(cf, f);
1004
1009
  }
1005
1010
  ret.x -= f.x;
@@ -1236,7 +1241,7 @@ SC.CoreView.reopen(
1236
1241
 
1237
1242
  });
1238
1243
 
1239
- SC.CoreView.mixin(/** @scope SC.View.prototype */ {
1244
+ SC.CoreView.mixin(/** @scope SC.CoreView.prototype */ {
1240
1245
 
1241
1246
  /** @private walk like a duck -- used by SC.Page */
1242
1247
  isViewClass: YES,
@@ -39,13 +39,20 @@ SC.View.reopen(
39
39
 
40
40
  /**
41
41
  Invokved just after the responder loses key responder status.
42
+ @param {SC.Responder} responder
42
43
  */
43
44
  didLoseKeyResponderTo: function(responder) {},
44
45
 
45
46
  /**
46
47
  Invoked just after the responder gains key responder status.
48
+ By default, it calls focus on the view root element. For accessibility
49
+ purposes.
50
+
51
+ @param {SC.Responder} responder
47
52
  */
48
- didBecomeKeyResponderFrom: function(responder) {},
53
+ didBecomeKeyResponderFrom: function(responder) {
54
+ this.$().focus();
55
+ },
49
56
 
50
57
  /**
51
58
  This method will process a key input event, attempting to convert it to
@@ -276,7 +283,10 @@ SC.View.reopen(
276
283
  }
277
284
 
278
285
  // if no parents have a next sibling, start over from the beginning
279
- if(!next) next = this.get('pane');
286
+ if(!next) {
287
+ if(!SC.TABBING_ONLY_INSIDE_DOCUMENT) break;
288
+ else next = this.get('pane');
289
+ }
280
290
 
281
291
  // if it's a valid firstResponder, we're done!
282
292
  if(next.get('isVisibleInWindow') && next.get('acceptsFirstResponder')) return next;
@@ -287,7 +297,6 @@ SC.View.reopen(
287
297
 
288
298
  // this will only happen if no views are visible and accept first responder
289
299
  return null;
290
-
291
300
  }.property('nextKeyView'),
292
301
 
293
302
  /**
@@ -355,6 +364,9 @@ SC.View.reopen(
355
364
  // normally, just try to get previous view's last child
356
365
  if(cur.get('parentView')) prev = cur._getPreviousKeyView();
357
366
 
367
+ // if we are the pane and address bar tabbing is enabled, trigger it now
368
+ else if(!SC.TABBING_ONLY_INSIDE_DOCUMENT) break;
369
+
358
370
  // if we are the pane, get our own last child
359
371
  else prev = cur;
360
372
 
@@ -126,16 +126,22 @@ SC.View.reopen(
126
126
  if(view.parentViewDidChange) view.parentViewDidChange();
127
127
  if(view.layoutDidChange) view.layoutDidChange();
128
128
 
129
- var pane = view.get('pane');
130
- if(pane && pane.get('isPaneAttached')) {
131
- view._notifyDidAppendToDocument();
132
- }
129
+ view.endPropertyChanges();
133
130
 
134
- // notify views
135
- if (this.didAddChild) { this.didAddChild(view, beforeView) ; }
136
- if (view.didAddToParent) { view.didAddToParent(this, beforeView) ; }
131
+ // Make sure all notifications are delayed since the appending
132
+ // doesn't complete until the end of the RunLoop
133
+ // There may be better ways to do this than with invokeLast,
134
+ // but it's the best I can do for now - PDW
135
+ this.invokeLast(function(){
136
+ var pane = view.get('pane');
137
+ if(pane && pane.get('isPaneAttached')) {
138
+ view._notifyDidAppendToDocument();
139
+ }
137
140
 
138
- view.endPropertyChanges();
141
+ // notify views
142
+ if (this.didAddChild) { this.didAddChild(view, beforeView) ; }
143
+ if (view.didAddToParent) { view.didAddToParent(this, beforeView) ; }
144
+ });
139
145
 
140
146
  return this ;
141
147
  },
@@ -116,23 +116,6 @@ SC.View.reopen(
116
116
  }
117
117
  }.property('baseThemeName', 'parentView').cacheable(),
118
118
 
119
- /**
120
- * Returns the named property if it is specified on the view, and
121
- * otherwise returns the named constant from the view's theme.
122
- *
123
- * @param {String} property The property on the view.
124
- * @param {String} constantName The name of the constant on the theme.
125
- */
126
- getThemedProperty: function(property, constantName){
127
- var value = this.get(property);
128
- if (value !== undefined) { return value; }
129
-
130
- var theme = this.get('theme');
131
- if (!theme) { return undefined; }
132
-
133
- return theme[constantName];
134
- },
135
-
136
119
  /**
137
120
  The object to which rendering and updating the HTML representation of this
138
121
  view should be delegated.
@@ -253,9 +236,16 @@ SC.View.reopen(
253
236
  original(context);
254
237
 
255
238
  var renderDelegate = this.get('renderDelegate');
239
+ if (renderDelegate && renderDelegate.className) {
240
+ context.addClass(renderDelegate.className);
241
+ }
242
+
243
+ // @if(debug)
256
244
  if (renderDelegate && renderDelegate.name) {
257
- context.addClass(renderDelegate.name);
245
+ SC.Logger.error("Render delegates now use 'className' instead of 'name'.");
246
+ SC.Logger.error("Name '%@' will be ignored.", renderDelegate.name);
258
247
  }
248
+ // @endif
259
249
  }.enhance()
260
250
  });
261
251
 
@@ -236,7 +236,7 @@ SC.FixturesDataSource = SC.DataSource.extend(
236
236
  @param {SC.Store} store the store to load into
237
237
  @param {SC.Record} recordType the record type to load
238
238
  @param {SC.Array} ret is passed, array to add loaded storeKeys to.
239
- @returns {SC.Fixture} receiver
239
+ @returns {SC.FixturesDataSource} receiver
240
240
  */
241
241
  loadFixturesFor: function(store, recordType, ret) {
242
242
  var hashes = [],
@@ -344,7 +344,17 @@ SC.FixturesDataSource = SC.DataSource.extend(
344
344
  var ret = [], fixtures = this._fixtures[SC.guidFor(recordType)];
345
345
  return fixtures ? YES: NO;
346
346
  },
347
-
347
+
348
+ /**
349
+ Resets the fixtures to their original values.
350
+
351
+ @returns {SC.FixturesDataSource} receiver
352
+ */
353
+ reset: function(){
354
+ this._fixtures = null;
355
+ return this;
356
+ },
357
+
348
358
  /**
349
359
  Returns YES or SC.MIXED_STATE if one or more of the storeKeys can be
350
360
  handled by the fixture data source.
@@ -0,0 +1,296 @@
1
+ // ==========================================================================
2
+ // Project: SproutCore - JavaScript Application Framework
3
+ // Copyright: ©2006-2011 Strobe Inc. and contributors.
4
+ // Portions ©2008-2011 Apple Inc. All rights reserved.
5
+ // License: Licensed under MIT license (see license.js)
6
+ // ==========================================================================
7
+
8
+ /** @class
9
+ Provides support for having relationships propagate by
10
+ data provided by the data source.
11
+
12
+ This means the following interaction is now valid:
13
+
14
+ App = { store: SC.Store.create(SC.RelationshipSupport) };
15
+
16
+ App.Person = SC.Record.extend({
17
+ primaryKey: 'name',
18
+
19
+ name: SC.Record.attr(String),
20
+
21
+ power: SC.Record.toOne('App.Power', {
22
+ isMaster: NO,
23
+ inverse: 'person'
24
+ })
25
+ });
26
+
27
+ App.Power = SC.Record.extend({
28
+ person: SC.Record.toOne('App.Person', {
29
+ isMaster: YES,
30
+ inverse: 'power'
31
+ })
32
+ });
33
+
34
+ var zan = App.store.createRecord(App.Person, { name: 'Zan' }),
35
+ jayna = App.store.createRecord(App.Person, { name: 'Janya' });
36
+
37
+ // Wondertwins activate!
38
+ var glacier = App.store.loadRecords(App.Power, [{
39
+ guid: 'petunia', // Shape of...
40
+ person: 'Jayna'
41
+ }, {
42
+ guid: 'drizzle', // Form of...
43
+ person: 'Zan'
44
+ }]);
45
+
46
+
47
+ Leveraging this mixin requires your records to be unambiguously
48
+ defined. Note that this mixin does not take into account record
49
+ relationship created / destroyed on `dataSourceDidComplete`,
50
+ `writeAttribute`, etc. The only support here is for `pushRetrieve`,
51
+ `pushDestroy`, and `loadRecords` (under the hood, `loadRecord(s)` uses
52
+ `pushRetrieve`).
53
+
54
+ This mixin also supports lazily creating records when a related
55
+ record is pushed in from the store (but it doesn't exist).
56
+
57
+ This means that the previous example could have been simplified
58
+ to this:
59
+
60
+ App.Power = SC.Record.extend({
61
+ person: SC.Record.toOne('App.Person', {
62
+ isMaster: YES,
63
+ lazilyInstantiate: YES,
64
+ inverse: 'power'
65
+ })
66
+ });
67
+
68
+ // Wondertwins activate!
69
+ var glacier = App.store.loadRecords(App.Power, [{
70
+ guid: 'petunia', // Shape of...
71
+ person: 'Jayna'
72
+ }, {
73
+ guid: 'drizzle', // Form of...
74
+ person: 'Zan'
75
+ }]);
76
+
77
+
78
+ When the `loadRecords` call is done, there will be four unmaterialized
79
+ records in the store, giving the exact same result as the former
80
+ example.
81
+
82
+ As a side note, this mixin was developed primarily for use
83
+ in a real-time backend that provides data to SproutCore
84
+ as soon as it gets it (no transactions). This means streaming
85
+ APIs / protocols like the Twitter streaming API or XMPP (an IM
86
+ protocol) can be codified easier.
87
+
88
+ @since SproutCore 1.6
89
+ */
90
+ SC.RelationshipSupport = {
91
+
92
+ /** @private
93
+ Relinquish many records.
94
+
95
+ This happens when a master record (`isMaster` = `YES`) removes a reference
96
+ to related records, either through `pushRetrieve` or `pushDestroy`.
97
+ */
98
+ _srs_inverseDidRelinquishRelationships: function (recordType, ids, attr, inverseId) {
99
+ ids.forEach(function (id) {
100
+ this._srs_inverseDidRelinquishRelationship(recordType, id, attr, inverseId);
101
+ },this);
102
+ },
103
+
104
+ /** @private
105
+ Relinquish the record, removing the reference of the record being
106
+ destroyed on any related records.
107
+ */
108
+ _srs_inverseDidRelinquishRelationship: function (recordType, id, toAttr, relativeID) {
109
+ var storeKey = recordType.storeKeyFor(id),
110
+ dataHash = this.readDataHash(storeKey),
111
+ key = toAttr.inverse,
112
+ proto = recordType.prototype;
113
+
114
+ if (!dataHash || !key) return;
115
+
116
+ if (SC.instanceOf(proto[key], SC.SingleAttribute)) {
117
+ delete dataHash[key];
118
+ } else if (SC.instanceOf(proto[key], SC.ManyAttribute)
119
+ && SC.typeOf(dataHash[key]) === SC.T_ARRAY) {
120
+ dataHash[key].removeObject(relativeID);
121
+ }
122
+
123
+ this.pushRetrieve(recordType, id, dataHash, undefined, true);
124
+ },
125
+
126
+ /** @private
127
+ Add a relationship to many inverse records.
128
+
129
+ This happens when a master record (`isMaster` = `YES`) adds a reference
130
+ to another record on a `pushRetrieve`.
131
+ */
132
+ _srs_inverseDidAddRelationships: function (recordType, ids, attr, inverseId) {
133
+ ids.forEach(function (id) {
134
+ this._srs_inverseDidAddRelationship(recordType, id, attr, inverseId);
135
+ },this);
136
+ },
137
+
138
+
139
+ /** @private
140
+ Add a relationship to an inverse record.
141
+
142
+ If the flag lazilyInstantiate is set to YES, then the inverse record will be
143
+ created lazily.
144
+
145
+ @param {SC.Record} recordType The inverse record type.
146
+ @param {String} id The id of the recordType to add.
147
+ @param {SC.RecordAttribute} toAttr The record attribute that represents
148
+ the relationship being created.
149
+ @param {String} relativeID The ID of the model that needs to have it's
150
+ relationship updated.
151
+ */
152
+ _srs_inverseDidAddRelationship: function (recordType, id, toAttr, relativeID) {
153
+ var storeKey = recordType.storeKeyFor(id),
154
+ dataHash = this.readDataHash(storeKey),
155
+ status = this.peekStatus(storeKey),
156
+ proto = recordType.prototype,
157
+ key = toAttr.inverse,
158
+ hashKey = proto[key],
159
+ primaryAttr = proto[proto.primaryKey],
160
+ shouldRecurse = false;
161
+
162
+ // in case the SC.RecordAttribute defines a `key` field, we need to use that
163
+ hashKey = (hashKey && hashKey.get && hashKey.get('key') || hashKey.key) || key;
164
+
165
+ if ((status === SC.Record.EMPTY) &&
166
+ (SC.typeOf(toAttr.lazilyInstantiate) === SC.T_FUNCTION && toAttr.lazilyInstantiate() ||
167
+ SC.typeOf(toAttr.lazilyInstantiate) !== SC.T_FUNCTION && toAttr.lazilyInstantiate)) {
168
+
169
+ if (!SC.none(primaryAttr) && primaryAttr.typeClass &&
170
+ SC.typeOf(primaryAttr.typeClass()) === SC.T_CLASS) {
171
+
172
+ // Recurse to create the record that this primaryKey points to iff it
173
+ // also should be created if the record is empty.
174
+ // Identifies chained relationships where the object up the chain
175
+ // doesn't exist yet.
176
+
177
+ // TODO: this can lead to an infinite recursion if the relationship
178
+ // graph is cyclic
179
+ shouldRecurse = true;
180
+ }
181
+ dataHash = {};
182
+ dataHash[proto.primaryKey] = id;
183
+ }
184
+
185
+ if (!dataHash || !key) return;
186
+
187
+ if (SC.instanceOf(proto[key], SC.SingleAttribute)) {
188
+ dataHash[hashKey] = relativeID;
189
+ } else if (SC.instanceOf(proto[key], SC.ManyAttribute)) {
190
+ dataHash[hashKey] = dataHash[hashKey] || [];
191
+
192
+ if (dataHash[key].indexOf(relativeID) < 0) {
193
+ dataHash[hashKey].push(relativeID);
194
+ }
195
+ }
196
+
197
+ this.pushRetrieve(recordType, id, dataHash, undefined, !shouldRecurse);
198
+ },
199
+
200
+ // ..........................................................
201
+ // ASYNCHRONOUS RECORD RELATIONSHIPS
202
+ //
203
+
204
+ /** @private
205
+ Iterates over keys on the recordType prototype, looking for RecordAttributes
206
+ that have relationships (toOne or toMany).
207
+
208
+ @param {SC.Record} recordType The record type to do introspection on to see
209
+ if it has any RecordAttributes that have relationships to other records.
210
+ @param {String} id The id of the record being pushed in.
211
+ @param {Number} storeKey The storeKey
212
+ */
213
+ _srs_pushIterator: function (recordType, id, storeKey, lambda) {
214
+ var proto = recordType.prototype,
215
+ attr, currentHash, key, inverseType;
216
+
217
+ if (typeof storeKey === "undefined") {
218
+ storeKey = recordType.storeKeyFor(id);
219
+ }
220
+
221
+ currentHash = this.readDataHash(storeKey) || {};
222
+
223
+ for (key in proto) {
224
+ attr = proto[key];
225
+ if (attr && attr.typeClass && attr.inverse && attr.isMaster) {
226
+ inverseType = attr.typeClass();
227
+
228
+ if (SC.typeOf(inverseType) !== SC.T_CLASS) continue;
229
+
230
+ lambda.apply(this, [inverseType, currentHash, attr,
231
+ attr.get && attr.get('key') || key]);
232
+ }
233
+ }
234
+ },
235
+
236
+
237
+ /**
238
+ Disassociate records that are related to the one being destroyed iff this
239
+ record has `isMaster` set to `YES`.
240
+ */
241
+ pushDestroy: function (original, recordType, id, storeKey) {
242
+ var existingIDs;
243
+
244
+ this._srs_pushIterator(recordType, id, storeKey,
245
+ function (inverseType, currentHash, toAttr, keyValue) {
246
+ // update old relationships
247
+ existingIDs = [currentHash[keyValue] || null].flatten().compact().uniq();
248
+ this._srs_inverseDidRelinquishRelationships(inverseType, existingIDs, toAttr, id);
249
+ });
250
+
251
+ return original(recordType, id, storeKey);
252
+ }.enhance(),
253
+
254
+ /**
255
+ Associate records that are added via a pushRetrieve, and update subsequent
256
+ relationships to ensure that the master-slave relationship is kept intact.
257
+
258
+ For use cases, see the test for pushRelationships.
259
+
260
+ The `ignore` argument is only set to true when adding the inverse
261
+ relationship (to prevent recursion).
262
+ */
263
+ pushRetrieve: function (original, recordType, id, dataHash, storeKey, ignore) {
264
+ // avoid infinite recursions when additional changes are propogated
265
+ // from `_srs_inverseDidAddRelationship`
266
+ if (!ignore) {
267
+ var existingIDs, inverseIDs;
268
+
269
+ this._srs_pushIterator(recordType, id, storeKey,
270
+ /** @ignore
271
+ @param {SC.Record} inverseType - in a Master-Slave
272
+ relationship when pushing master, Slave is the inverse type
273
+ @param {Object} currentHash - the hash in the data store (data to be replaced by `dataHash`)
274
+ @param {SC.RecordAttribute} toAttr - key in `recordType.prototype` that names isMaster and has an inverse
275
+ @param {String} keyValue - the property name in the datahash that defines this foreign key relationship
276
+ */
277
+ function (inverseType, currentHash, toAttr, keyValue) {
278
+ // update old relationships
279
+ existingIDs = [currentHash[keyValue] || null].flatten().compact().uniq();
280
+
281
+ // update new relationships
282
+ inverseIDs = [dataHash[keyValue] || null].flatten().compact().uniq();
283
+
284
+ this._srs_inverseDidRelinquishRelationships(inverseType, existingIDs.filter(
285
+ function (el) {
286
+ return inverseIDs.indexOf(el) === -1;
287
+ }), toAttr, id);
288
+
289
+ this._srs_inverseDidAddRelationships(inverseType, inverseIDs, toAttr, id);
290
+ });
291
+ }
292
+
293
+ storeKey = storeKey || recordType.storeKeyFor(id);
294
+ return original(recordType, id, dataHash, storeKey);
295
+ }.enhance()
296
+ };