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
@@ -133,6 +133,27 @@ SC.RecordAttribute = SC.Object.extend(
133
133
  @default NO
134
134
  */
135
135
  aggregate: NO,
136
+
137
+
138
+ /**
139
+ Can only be used for toOne or toMany relationship attributes. If YES,
140
+ this flag will lazily create the related record that was pushed in
141
+ from the data source (via pushRetrieve) if the related record does
142
+ not exist yet.
143
+
144
+ Useful when you have a record used as a join table. Assumptions then
145
+ can be made that the record exists at all times (even if it doesn't).
146
+ For instance, if you have a contact that is a member of groups,
147
+ a group will be created automatically when a contact pushes a new
148
+ group.
149
+
150
+ Note that you will have to take care of destroying the created record
151
+ once all relationships are removed from it.
152
+
153
+ @property {Boolean}
154
+ @default NO
155
+ */
156
+ lazilyInstantiate: NO,
136
157
 
137
158
  // ..........................................................
138
159
  // HELPER PROPERTIES
@@ -149,7 +170,7 @@ SC.RecordAttribute = SC.Object.extend(
149
170
  */
150
171
  typeClass: function() {
151
172
  var ret = this.get('type');
152
- if (SC.typeOf(ret) === SC.T_STRING) ret = SC.objectForPropertyPath(ret);
173
+ if (SC.typeOf(ret) === SC.T_STRING) ret = SC.requiredObjectForPropertyPath(ret);
153
174
  return ret ;
154
175
  }.property('type').cacheable(),
155
176
 
@@ -318,7 +318,7 @@ SC.NestedStore = SC.Store.extend(
318
318
  if (!editables) editables = this.editables = [];
319
319
  editables[storeKey] = 1 ; // mark as editable
320
320
 
321
- } else this.dataHashes[storeKey] = this.dataHashes[storeKey];
321
+ } else this.dataHashes[storeKey] = pstore.dataHashes[storeKey];
322
322
 
323
323
  // also copy the status + revision
324
324
  this.statuses[storeKey] = this.statuses[storeKey];
@@ -11,14 +11,14 @@ sc_require('models/record');
11
11
  @class
12
12
 
13
13
 
14
- The Store is where you can find all of your dataHashes. Stores can be
15
- chained for editing purposes and committed back one chain level at a time
14
+ The Store is where you can find all of your dataHashes. Stores can be
15
+ chained for editing purposes and committed back one chain level at a time
16
16
  all the way back to a persistent data source.
17
-
17
+
18
18
  Every application you create should generally have its own store objects.
19
19
  Once you create the store, you will rarely need to work with the store
20
- directly except to retrieve records and collections.
21
-
20
+ directly except to retrieve records and collections.
21
+
22
22
  Internally, the store will keep track of changes to your json data hashes
23
23
  and manage syncing those changes with your data source. A data source may
24
24
  be a server, local storage, or any other persistent code.
@@ -27,19 +27,19 @@ sc_require('models/record');
27
27
  @since SproutCore 1.0
28
28
  */
29
29
  SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
30
-
30
+
31
31
  /**
32
32
  An (optional) name of the store, which can be useful during debugging,
33
33
  especially if you have multiple nested stores.
34
-
34
+
35
35
  @type String
36
36
  */
37
37
  name: null,
38
38
 
39
39
  /**
40
- An array of all the chained stores that current rely on the receiver
40
+ An array of all the chained stores that current rely on the receiver
41
41
  store.
42
-
42
+
43
43
  @type Array
44
44
  */
45
45
  nestedStores: null,
@@ -48,39 +48,39 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
48
48
  The data source is the persistent storage that will provide data to the
49
49
  store and save changes. You normally will set your data source when you
50
50
  first create your store in your application.
51
-
51
+
52
52
  @type SC.DataSource
53
53
  */
54
54
  dataSource: null,
55
-
55
+
56
56
  /**
57
57
  This type of store is not nested.
58
-
58
+
59
59
  @default NO
60
60
  @type Boolean
61
61
  */
62
62
  isNested: NO,
63
-
63
+
64
64
  /**
65
65
  This type of store is not nested.
66
-
66
+
67
67
  @default NO
68
68
  @type Boolean
69
69
  */
70
70
  commitRecordsAutomatically: NO,
71
-
71
+
72
72
  // ..........................................................
73
73
  // DATA SOURCE SUPPORT
74
- //
75
-
74
+ //
75
+
76
76
  /**
77
77
  Convenience method. Sets the current data source to the passed property.
78
78
  This will also set the store property on the dataSource to the receiver.
79
-
79
+
80
80
  If you are using this from the `core.js` method of your app, you may need to
81
81
  just pass a string naming your data source class. If this is the case,
82
82
  then your data source will be instantiated the first time it is requested.
83
-
83
+
84
84
  @param {SC.DataSource|String} dataSource the data source
85
85
  @returns {SC.Store} receiver
86
86
  */
@@ -88,48 +88,48 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
88
88
  this.set('dataSource', dataSource);
89
89
  return this ;
90
90
  },
91
-
91
+
92
92
  // lazily convert data source to real object
93
93
  _getDataSource: function() {
94
94
  var ret = this.get('dataSource');
95
95
  if (typeof ret === SC.T_STRING) {
96
- ret = SC.objectForPropertyPath(ret);
97
- if (ret && ret.isClass) ret = ret.create();
98
- if (ret) this.set('dataSource', ret);
96
+ ret = SC.requiredObjectForPropertyPath(ret);
97
+ if (ret.isClass) ret = ret.create();
98
+ this.set('dataSource', ret);
99
99
  }
100
100
  return ret;
101
101
  },
102
-
102
+
103
103
  /**
104
- Convenience method. Creates a `CascadeDataSource` with the passed
105
- data source arguments and sets the `CascadeDataSource` as the data source
104
+ Convenience method. Creates a `CascadeDataSource` with the passed
105
+ data source arguments and sets the `CascadeDataSource` as the data source
106
106
  for the receiver.
107
-
107
+
108
108
  @param {SC.DataSource...} dataSource one or more data source arguments
109
- @returns {SC.Store} reciever
109
+ @returns {SC.Store} receiver
110
110
  */
111
111
  cascade: function(dataSource) {
112
112
  var dataSources = SC.A(arguments) ;
113
113
  dataSource = SC.CascadeDataSource.create({
114
- dataSources: dataSources
114
+ dataSources: dataSources
115
115
  });
116
116
  return this.from(dataSource);
117
117
  },
118
-
118
+
119
119
  // ..........................................................
120
120
  // STORE CHAINING
121
- //
122
-
123
- /**
121
+ //
122
+
123
+ /**
124
124
  Returns a new nested store instance that can be used to buffer changes
125
- until you are ready to commit them. When you are ready to commit your
125
+ until you are ready to commit them. When you are ready to commit your
126
126
  changes, call `commitChanges()` or `destroyChanges()` and then `destroy()`
127
127
  when you are finished with the chained store altogether.
128
-
128
+
129
129
  store = MyApp.store.chain();
130
130
  .. edit edit edit
131
131
  store.commitChanges().destroy();
132
-
132
+
133
133
  @param {Hash} attrs optional attributes to set on new store
134
134
  @param {Class} newStoreClass optional the class of the newly-created nested store (defaults to SC.NestedStore)
135
135
  @returns {SC.NestedStore} new nested store chained to receiver
@@ -137,7 +137,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
137
137
  chain: function(attrs, newStoreClass) {
138
138
  if (!attrs) attrs = {};
139
139
  attrs.parentStore = this;
140
-
140
+
141
141
  if (newStoreClass) {
142
142
  // Ensure the passed-in class is a type of nested store.
143
143
  if (SC.typeOf(newStoreClass) !== 'class') throw new Error("%@ is not a valid class".fmt(newStoreClass));
@@ -146,24 +146,24 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
146
146
  else {
147
147
  newStoreClass = SC.NestedStore;
148
148
  }
149
-
149
+
150
150
  // Replicate parent records references
151
151
  attrs.childRecords = this.childRecords ? SC.clone(this.childRecords) : {};
152
152
  attrs.parentRecords = this.parentRecords ? SC.clone(this.parentRecords) : {};
153
-
153
+
154
154
  var ret = newStoreClass.create(attrs),
155
155
  nested = this.nestedStores;
156
-
156
+
157
157
  if (!nested) nested = this.nestedStores = [];
158
158
  nested.push(ret);
159
159
  return ret ;
160
160
  },
161
-
161
+
162
162
  /** @private
163
-
163
+
164
164
  Called by a nested store just before it is destroyed so that the parent
165
165
  can remove the store from its list of nested stores.
166
-
166
+
167
167
  @returns {SC.Store} receiver
168
168
  */
169
169
  willDestroyNestedStore: function(nestedStore) {
@@ -176,7 +176,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
176
176
  /**
177
177
  Used to determine if a nested store belongs directly or indirectly to the
178
178
  receiver.
179
-
179
+
180
180
  @param {SC.Store} store store instance
181
181
  @returns {Boolean} YES if belongs
182
182
  */
@@ -186,110 +186,110 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
186
186
  },
187
187
 
188
188
  // ..........................................................
189
- // SHARED DATA STRUCTURES
190
- //
191
-
189
+ // SHARED DATA STRUCTURES
190
+ //
191
+
192
192
  /** @private
193
- JSON data hashes indexed by store key.
194
-
193
+ JSON data hashes indexed by store key.
194
+
195
195
  *IMPORTANT: Property is not observable*
196
196
 
197
197
  Shared by a store and its child stores until you make edits to it.
198
-
198
+
199
199
  @type Hash
200
200
  */
201
201
  dataHashes: null,
202
202
 
203
203
  /** @private
204
204
  The current status of a data hash indexed by store key.
205
-
205
+
206
206
  *IMPORTANT: Property is not observable*
207
207
 
208
208
  Shared by a store and its child stores until you make edits to it.
209
-
209
+
210
210
  @type Hash
211
211
  */
212
212
  statuses: null,
213
-
213
+
214
214
  /** @private
215
- This array contains the revisions for the attributes indexed by the
216
- storeKey.
217
-
215
+ This array contains the revisions for the attributes indexed by the
216
+ storeKey.
217
+
218
218
  *IMPORTANT: Property is not observable*
219
-
220
- Revisions are used to keep track of when an attribute hash has been
221
- changed. A store shares the revisions data with its parent until it
219
+
220
+ Revisions are used to keep track of when an attribute hash has been
221
+ changed. A store shares the revisions data with its parent until it
222
222
  starts to make changes to it.
223
-
223
+
224
224
  @type Hash
225
225
  */
226
226
  revisions: null,
227
227
 
228
228
  /**
229
- Array indicates whether a data hash is possibly in use by an external
229
+ Array indicates whether a data hash is possibly in use by an external
230
230
  record for editing. If a data hash is editable then it may be modified
231
- at any time and therefore chained stores may need to clone the
231
+ at any time and therefore chained stores may need to clone the
232
232
  attributes before keeping a copy of them.
233
-
234
- Note that this is kept as an array because it will be stored as a dense
233
+
234
+ Note that this is kept as an array because it will be stored as a dense
235
235
  array on some browsers, making it faster.
236
-
236
+
237
237
  @type Array
238
238
  */
239
239
  editables: null,
240
-
240
+
241
241
  /**
242
242
  A set of storeKeys that need to be committed back to the data source. If
243
243
  you call `commitRecords()` without passing any other parameters, the keys
244
244
  in this set will be committed instead.
245
-
245
+
246
246
  @type SC.Set
247
247
  */
248
248
  changelog: null,
249
-
249
+
250
250
  /**
251
251
  An array of `SC.Error` objects associated with individual records in the
252
252
  store (indexed by store keys).
253
-
253
+
254
254
  Errors passed form the data source in the call to dataSourceDidError() are
255
255
  stored here.
256
-
256
+
257
257
  @type Array
258
258
  */
259
259
  recordErrors: null,
260
-
260
+
261
261
  /**
262
262
  A hash of `SC.Error` objects associated with queries (indexed by the GUID
263
263
  of the query).
264
-
264
+
265
265
  Errors passed from the data source in the call to
266
266
  `dataSourceDidErrorQuery()` are stored here.
267
-
267
+
268
268
  @type Hash
269
269
  */
270
270
  queryErrors: null,
271
-
271
+
272
272
  /**
273
273
  A hash of child Records and there immediate parents
274
274
  */
275
275
  childRecords: null,
276
-
276
+
277
277
  /**
278
278
  A hash of parent records with registered children
279
279
  */
280
280
  parentRecords: null,
281
-
281
+
282
282
  // ..........................................................
283
283
  // CORE ATTRIBUTE API
284
- //
284
+ //
285
285
  // The methods in this layer work on data hashes in the store. They do not
286
- // perform any changes that can impact records. Usually you will not need
286
+ // perform any changes that can impact records. Usually you will not need
287
287
  // to use these methods.
288
-
288
+
289
289
  /**
290
290
  Returns the current edit status of a storekey. May be one of
291
291
  `EDITABLE` or `LOCKED`. Used mostly for unit testing.
292
-
292
+
293
293
  @param {Number} storeKey the store key
294
294
  @returns {Number} edit status
295
295
  */
@@ -297,27 +297,27 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
297
297
  var editables = this.editables, locks = this.locks;
298
298
  return (editables && editables[storeKey]) ? SC.Store.EDITABLE : SC.Store.LOCKED ;
299
299
  },
300
-
301
- /**
300
+
301
+ /**
302
302
  Returns the data hash for the given `storeKey`. This will also 'lock'
303
- the hash so that further edits to the parent store will no
303
+ the hash so that further edits to the parent store will no
304
304
  longer be reflected in this store until you reset.
305
-
305
+
306
306
  @param {Number} storeKey key to retrieve
307
307
  @returns {Hash} data hash or null
308
308
  */
309
309
  readDataHash: function(storeKey) {
310
310
  return this.dataHashes[storeKey];
311
311
  },
312
-
313
- /**
312
+
313
+ /**
314
314
  Returns the data hash for the `storeKey`, cloned so that you can edit
315
315
  the contents of the attributes if you like. This will do the extra work
316
- to make sure that you only clone the attributes one time.
317
-
318
- If you use this method to modify data hash, be sure to call
316
+ to make sure that you only clone the attributes one time.
317
+
318
+ If you use this method to modify data hash, be sure to call
319
319
  `dataHashDidChange()` when you make edits to record the change.
320
-
320
+
321
321
  @param {Number} storeKey the store key to retrieve
322
322
  @returns {Hash} the attributes hash
323
323
  */
@@ -335,48 +335,48 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
335
335
  }
336
336
  return ret;
337
337
  },
338
-
338
+
339
339
  /**
340
- Reads a property from the hash - cloning it if needed so you can modify
340
+ Reads a property from the hash - cloning it if needed so you can modify
341
341
  it independently of any parent store. This method is really only well
342
- tested for use with toMany relationships. Although it is public you
342
+ tested for use with toMany relationships. Although it is public you
343
343
  generally should not call it directly.
344
-
345
- @param {Number} storeKey storeKey of data hash
344
+
345
+ @param {Number} storeKey storeKey of data hash
346
346
  @param {String} propertyName property to read
347
347
  @returns {Object} editable property value
348
348
  */
349
349
  readEditableProperty: function(storeKey, propertyName) {
350
- var hash = this.readEditableDataHash(storeKey),
350
+ var hash = this.readEditableDataHash(storeKey),
351
351
  editables = this.editables[storeKey], // get editable info...
352
352
  ret = hash[propertyName];
353
-
353
+
354
354
  // editables must be made into a hash so that we can keep track of which
355
355
  // properties have already been made editable
356
356
  if (editables === 1) editables = this.editables[storeKey] = {};
357
-
357
+
358
358
  // clone if needed
359
359
  if (!editables[propertyName]) {
360
360
  ret = hash[propertyName];
361
361
  if (ret && ret.isCopyable) ret = hash[propertyName] = ret.copy(YES);
362
362
  editables[propertyName] = YES ;
363
363
  }
364
-
364
+
365
365
  return ret ;
366
366
  },
367
-
367
+
368
368
  /**
369
369
  Replaces the data hash for the `storeKey`. This will lock the data hash
370
370
  and mark them as cloned. This will also call `dataHashDidChange()` for
371
371
  you.
372
-
373
- Note that the hash you set here must be a different object from the
372
+
373
+ Note that the hash you set here must be a different object from the
374
374
  original data hash. Once you make a change here, you must also call
375
375
  `dataHashDidChange()` to register the changes.
376
376
 
377
377
  If the data hash does not yet exist in the store, this method will add it.
378
378
  Pass the optional status to edit the status as well.
379
-
379
+
380
380
  @param {Number} storeKey the store key to write
381
381
  @param {Hash} hash the new hash
382
382
  @param {String} status the new hash status
@@ -387,53 +387,53 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
387
387
  // update dataHashes and optionally status.
388
388
  if (hash) this.dataHashes[storeKey] = hash;
389
389
  if (status) this.statuses[storeKey] = status ;
390
-
390
+
391
391
  // also note that this hash is now editable
392
392
  var editables = this.editables;
393
393
  if (!editables) editables = this.editables = [];
394
394
  editables[storeKey] = 1 ; // use number for dense array support
395
-
395
+
396
396
  var that = this;
397
397
  this._propagateToChildren(storeKey, function(storeKey){
398
398
  that.writeDataHash(storeKey, null, status);
399
399
  });
400
-
400
+
401
401
  return this ;
402
402
  },
403
403
 
404
404
  /**
405
405
  Removes the data hash from the store. This does not imply a deletion of
406
- the record. You could be simply unloading the record. Either way,
407
- removing the dataHash will be synced back to the parent store but not to
406
+ the record. You could be simply unloading the record. Either way,
407
+ removing the dataHash will be synced back to the parent store but not to
408
408
  the server.
409
-
409
+
410
410
  Note that you can optionally pass a new status to go along with this. If
411
411
  you do not pass a status, it will change the status to `SC.RECORD_EMPTY`
412
412
  (assuming you just unloaded the record). If you are deleting the record
413
413
  you may set it to `SC.Record.DESTROYED_CLEAN`.
414
-
414
+
415
415
  Be sure to also call `dataHashDidChange()` to register this change.
416
-
416
+
417
417
  @param {Number} storeKey
418
418
  @param {String} status optional new status
419
- @returns {SC.Store} reciever
419
+ @returns {SC.Store} receiver
420
420
  */
421
421
  removeDataHash: function(storeKey, status) {
422
422
  // don't use delete -- that will allow parent dataHash to come through
423
- this.dataHashes[storeKey] = null;
423
+ this.dataHashes[storeKey] = null;
424
424
  this.statuses[storeKey] = status || SC.Record.EMPTY;
425
-
425
+
426
426
  // hash is gone and therefore no longer editable
427
427
  var editables = this.editables;
428
428
  if (editables) editables[storeKey] = 0 ;
429
-
430
- return this ;
429
+
430
+ return this ;
431
431
  },
432
-
432
+
433
433
  /**
434
- Reads the current status for a storeKey. This will also lock the data
434
+ Reads the current status for a storeKey. This will also lock the data
435
435
  hash. If no status is found, returns `SC.RECORD_EMPTY`.
436
-
436
+
437
437
  @param {Number} storeKey the store key
438
438
  @returns {Number} status
439
439
  */
@@ -443,41 +443,41 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
443
443
  this.readDataHash(storeKey);
444
444
  return this.statuses[storeKey] || SC.Record.EMPTY;
445
445
  },
446
-
446
+
447
447
  /**
448
- Reads the current status for the storeKey without actually locking the
448
+ Reads the current status for the storeKey without actually locking the
449
449
  record. Usually you won't need to use this method. It is mostly used
450
450
  internally.
451
-
451
+
452
452
  @param {Number} storeKey the store key
453
453
  @returns {Number} status
454
454
  */
455
455
  peekStatus: function(storeKey) {
456
- return this.statuses[storeKey] || SC.Record.EMPTY;
456
+ return this.statuses[storeKey] || SC.Record.EMPTY;
457
457
  },
458
-
458
+
459
459
  /**
460
- Writes the current status for a storeKey. If the new status is
461
- `SC.Record.ERROR`, you may also pass an optional error object. Otherwise
460
+ Writes the current status for a storeKey. If the new status is
461
+ `SC.Record.ERROR`, you may also pass an optional error object. Otherwise
462
462
  this param is ignored.
463
-
463
+
464
464
  @param {Number} storeKey the store key
465
465
  @param {String} newStatus the new status
466
466
  @param {SC.Error} error optional error object
467
467
  @returns {SC.Store} receiver
468
468
  */
469
469
  writeStatus: function(storeKey, newStatus) {
470
- // use writeDataHash for now to handle optimistic lock. maximize code
470
+ // use writeDataHash for now to handle optimistic lock. maximize code
471
471
  // reuse.
472
472
  return this.writeDataHash(storeKey, null, newStatus);
473
473
  },
474
-
474
+
475
475
  /**
476
476
  Call this method whenever you modify some editable data hash to register
477
477
  with the Store that the attribute values have actually changed. This will
478
- do the book-keeping necessary to track the change across stores including
478
+ do the book-keeping necessary to track the change across stores including
479
479
  managing locks.
480
-
480
+
481
481
  @param {Number|Array} storeKeys one or more store keys that changed
482
482
  @param {Number} rev optional new revision number. normally leave null
483
483
  @param {Boolean} statusOnly (optional) YES if only status changed
@@ -485,13 +485,13 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
485
485
  @returns {SC.Store} receiver
486
486
  */
487
487
  dataHashDidChange: function(storeKeys, rev, statusOnly, key) {
488
-
488
+
489
489
  // update the revision for storeKey. Use generateStoreKey() because that
490
- // gaurantees a universally (to this store hierarchy anyway) unique
490
+ // gaurantees a universally (to this store hierarchy anyway) unique
491
491
  // key value.
492
492
  if (!rev) rev = SC.Store.generateStoreKey();
493
493
  var isArray, len, idx, storeKey;
494
-
494
+
495
495
  isArray = SC.typeOf(storeKeys) === SC.T_ARRAY;
496
496
  if (isArray) {
497
497
  len = storeKeys.length;
@@ -499,39 +499,39 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
499
499
  len = 1;
500
500
  storeKey = storeKeys;
501
501
  }
502
-
502
+
503
503
  var that = this;
504
504
  for(idx=0;idx<len;idx++) {
505
505
  if (isArray) storeKey = storeKeys[idx];
506
506
  this.revisions[storeKey] = rev;
507
507
  this._notifyRecordPropertyChange(storeKey, statusOnly, key);
508
-
508
+
509
509
  this._propagateToChildren(storeKey, function(storeKey){
510
510
  that.dataHashDidChange(storeKey, null, statusOnly, key);
511
511
  });
512
512
  }
513
-
513
+
514
514
  return this ;
515
515
  },
516
516
 
517
- /** @private
517
+ /** @private
518
518
  Will push all changes to a the recordPropertyChanges property
519
519
  and execute `flush()` once at the end of the runloop.
520
520
  */
521
521
  _notifyRecordPropertyChange: function(storeKey, statusOnly, key) {
522
-
523
- var records = this.records,
522
+
523
+ var records = this.records,
524
524
  nestedStores = this.get('nestedStores'),
525
525
  K = SC.Store,
526
526
  rec, editState, len, idx, store, status, keys;
527
-
527
+
528
528
  // pass along to nested stores
529
529
  len = nestedStores ? nestedStores.length : 0 ;
530
530
  for(idx=0;idx<len;idx++) {
531
531
  store = nestedStores[idx];
532
532
  status = store.peekStatus(storeKey); // important: peek avoids read-lock
533
533
  editState = store.storeKeyEditState(storeKey);
534
-
534
+
535
535
  // when store needs to propagate out changes in the parent store
536
536
  // to nested stores
537
537
  if (editState === K.INHERITED) {
@@ -543,29 +543,29 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
543
543
  store.reset();
544
544
  }
545
545
  }
546
-
546
+
547
547
  // store info in changes hash and schedule notification if needed.
548
548
  var changes = this.recordPropertyChanges;
549
549
  if (!changes) {
550
- changes = this.recordPropertyChanges =
550
+ changes = this.recordPropertyChanges =
551
551
  { storeKeys: SC.CoreSet.create(),
552
552
  records: SC.CoreSet.create(),
553
553
  hasDataChanges: SC.CoreSet.create(),
554
554
  propertyForStoreKeys: {} };
555
555
  }
556
-
556
+
557
557
  changes.storeKeys.add(storeKey);
558
558
 
559
559
  if (records && (rec=records[storeKey])) {
560
560
  changes.records.push(storeKey);
561
-
561
+
562
562
  // If there are changes other than just the status we need to record
563
563
  // that information so we do the right thing during the next flush.
564
564
  // Note that if we're called multiple times before flush and one call
565
565
  // has `statusOnly=true` and another has `statusOnly=false`, the flush
566
566
  // will (correctly) operate in `statusOnly=false` mode.
567
567
  if (!statusOnly) changes.hasDataChanges.push(storeKey);
568
-
568
+
569
569
  // If this is a key specific change, make sure that only those
570
570
  // properties/keys are notified. However, if a previous invocation of
571
571
  // `_notifyRecordPropertyChange` specified that all keys have changed, we
@@ -574,7 +574,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
574
574
  if (!(keys = changes.propertyForStoreKeys[storeKey])) {
575
575
  keys = changes.propertyForStoreKeys[storeKey] = SC.CoreSet.create();
576
576
  }
577
-
577
+
578
578
  // If it's '*' instead of a set, then that means there was a previous
579
579
  // invocation that said all keys have changed.
580
580
  if (keys !== '*') {
@@ -586,22 +586,22 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
586
586
  changes.propertyForStoreKeys[storeKey] = '*';
587
587
  }
588
588
  }
589
-
589
+
590
590
  this.invokeOnce(this.flush);
591
591
  return this;
592
592
  },
593
593
 
594
594
  /**
595
- Delivers any pending changes to materialized records. Normally this
595
+ Delivers any pending changes to materialized records. Normally this
596
596
  happens once, automatically, at the end of the RunLoop. If you have
597
- updated some records and need to update records immediately, however,
597
+ updated some records and need to update records immediately, however,
598
598
  you may call this manually.
599
599
 
600
600
  @returns {SC.Store} receiver
601
601
  */
602
602
  flush: function() {
603
603
  if (!this.recordPropertyChanges) return this;
604
-
604
+
605
605
  var changes = this.recordPropertyChanges,
606
606
  storeKeys = changes.storeKeys,
607
607
  hasDataChanges = changes.hasDataChanges,
@@ -609,26 +609,26 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
609
609
  propertyForStoreKeys = changes.propertyForStoreKeys,
610
610
  recordTypes = SC.CoreSet.create(),
611
611
  rec, recordType, statusOnly, idx, len, storeKey, keys;
612
-
612
+
613
613
  storeKeys.forEach(function(storeKey) {
614
614
  if (records.contains(storeKey)) {
615
615
  statusOnly = hasDataChanges.contains(storeKey) ? NO : YES;
616
616
  rec = this.records[storeKey];
617
617
  keys = propertyForStoreKeys ? propertyForStoreKeys[storeKey] : null;
618
-
618
+
619
619
  // Are we invalidating all keys? If so, don't pass any to
620
620
  // storeDidChangeProperties.
621
621
  if (keys === '*') keys = null;
622
-
622
+
623
623
  // remove it so we don't trigger this twice
624
624
  records.remove(storeKey);
625
-
625
+
626
626
  if (rec) rec.storeDidChangeProperties(statusOnly, keys);
627
627
  }
628
-
628
+
629
629
  recordType = SC.Store.recordTypeFor(storeKey);
630
630
  recordTypes.add(recordType);
631
-
631
+
632
632
  }, this);
633
633
 
634
634
  if (storeKeys.get('length') > 0) this._notifyRecordArrays(storeKeys, recordTypes);
@@ -638,19 +638,19 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
638
638
  records.clear();
639
639
  // Provide full reference to overwrite
640
640
  this.recordPropertyChanges.propertyForStoreKeys = {};
641
-
641
+
642
642
  return this;
643
643
  },
644
-
644
+
645
645
  /**
646
646
  Resets the store content. This will clear all internal data for all
647
647
  records, resetting them to an EMPTY state. You generally do not want
648
648
  to call this method yourself, though you may override it.
649
-
649
+
650
650
  @returns {SC.Store} receiver
651
651
  */
652
652
  reset: function() {
653
-
653
+
654
654
  // create a new empty data store
655
655
  this.dataHashes = {} ;
656
656
  this.revisions = {} ;
@@ -662,6 +662,9 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
662
662
  this.recordErrors = null;
663
663
  this.queryErrors = null;
664
664
 
665
+ var dataSource = this.get('dataSource');
666
+ if (dataSource && dataSource.reset) { dataSource.reset(); }
667
+
665
668
  var records = this.records, storeKey;
666
669
  if (records) {
667
670
  for(storeKey in records) {
@@ -669,21 +672,21 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
669
672
  this._notifyRecordPropertyChange(parseInt(storeKey, 10), NO);
670
673
  }
671
674
  }
672
-
675
+
673
676
  this.set('hasChanges', NO);
674
677
  },
675
-
678
+
676
679
  /** @private
677
680
  Called by a nested store on a parent store to commit any changes from the
678
- store. This will copy any changed dataHashes as well as any persistant
681
+ store. This will copy any changed dataHashes as well as any persistant
679
682
  change logs.
680
-
683
+
681
684
  If the parentStore detects a conflict with the optimistic locking, it will
682
- raise an exception before it makes any changes. If you pass the
685
+ raise an exception before it makes any changes. If you pass the
683
686
  force flag then this detection phase will be skipped and the changes will
684
687
  be applied even if another resource has modified the store in the mean
685
688
  time.
686
-
689
+
687
690
  @param {SC.Store} nestedStore the child store
688
691
  @param {SC.Set} changes the set of changed store keys
689
692
  @param {Boolean} force
@@ -692,20 +695,20 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
692
695
  commitChangesFromNestedStore: function(nestedStore, changes, force) {
693
696
  // first, check for optimistic locking problems
694
697
  if (!force) this._verifyLockRevisions(changes, nestedStore.locks);
695
-
696
- // OK, no locking issues. So let's just copy them changes.
698
+
699
+ // OK, no locking issues. So let's just copy them changes.
697
700
  // get local reference to values.
698
- var len = changes.length, i, storeKey, myDataHashes, myStatuses,
699
- myEditables, myRevisions, myParentRecords, myChildRecords,
701
+ var len = changes.length, i, storeKey, myDataHashes, myStatuses,
702
+ myEditables, myRevisions, myParentRecords, myChildRecords,
700
703
  chDataHashes, chStatuses, chRevisions, chParentRecords, chChildRecords;
701
-
704
+
702
705
  myRevisions = this.revisions ;
703
706
  myDataHashes = this.dataHashes;
704
707
  myStatuses = this.statuses;
705
708
  myEditables = this.editables ;
706
709
  myParentRecords = this.parentRecords ? this.parentRecords : this.parentRecords ={} ;
707
710
  myChildRecords = this.childRecords ? this.childRecords : this.childRecords = {} ;
708
-
711
+
709
712
  // setup some arrays if needed
710
713
  if (!myEditables) myEditables = this.editables = [] ;
711
714
  chDataHashes = nestedStore.dataHashes;
@@ -713,7 +716,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
713
716
  chStatuses = nestedStore.statuses;
714
717
  chParentRecords = nestedStore.parentRecords || {};
715
718
  chChildRecords = nestedStore.childRecords || {};
716
-
719
+
717
720
  for(i=0;i<len;i++) {
718
721
  storeKey = changes[i];
719
722
 
@@ -723,32 +726,32 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
723
726
  myRevisions[storeKey] = chRevisions[storeKey];
724
727
  myParentRecords[storeKey] = chParentRecords[storeKey];
725
728
  myChildRecords[storeKey] = chChildRecords[storeKey];
726
-
729
+
727
730
  myEditables[storeKey] = 0 ; // always make dataHash no longer editable
728
-
731
+
729
732
  this._notifyRecordPropertyChange(storeKey, NO);
730
733
  }
731
-
734
+
732
735
  // add any records to the changelog for commit handling
733
736
  var myChangelog = this.changelog, chChangelog = nestedStore.changelog;
734
737
  if (chChangelog) {
735
738
  if (!myChangelog) myChangelog = this.changelog = SC.CoreSet.create();
736
739
  myChangelog.addEach(chChangelog);
737
- }
740
+ }
738
741
  this.changelog = myChangelog;
739
-
742
+
740
743
  // immediately flush changes to notify records - nested stores will flush
741
744
  // later on.
742
745
  if (!this.get('parentStore')) this.flush();
743
-
746
+
744
747
  return this ;
745
748
  },
746
749
 
747
750
  /** @private
748
- Verifies that the passed lock revisions match the current revisions
749
- in the receiver store. If the lock revisions do not match, then the
751
+ Verifies that the passed lock revisions match the current revisions
752
+ in the receiver store. If the lock revisions do not match, then the
750
753
  store is in a conflict and an exception will be raised.
751
-
754
+
752
755
  @param {Array} changes set of changes we are trying to apply
753
756
  @param {SC.Set} locks the locks to verify
754
757
  @returns {SC.Store} receiver
@@ -763,85 +766,85 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
763
766
 
764
767
  // if the save revision for the item does not match the current rev
765
768
  // the someone has changed the data hash in this store and we have
766
- // a conflict.
769
+ // a conflict.
767
770
  if (lock < rev) throw SC.Store.CHAIN_CONFLICT_ERROR;
768
- }
771
+ }
769
772
  }
770
773
  return this ;
771
774
  },
772
-
775
+
773
776
  // ..........................................................
774
777
  // HIGH-LEVEL RECORD API
775
- //
776
-
778
+ //
779
+
777
780
  /**
778
781
  Finds a single record instance with the specified `recordType` and id or
779
782
  an array of records matching some query conditions.
780
-
783
+
781
784
  Finding a Single Record
782
785
  ---
783
-
786
+
784
787
  If you pass a single `recordType` and id, this method will return an
785
788
  actual record instance. If the record has not been loaded into the store
786
789
  yet, this method will ask the data source to retrieve it. If no data
787
790
  source indicates that it can retrieve the record, then this method will
788
791
  return `null`.
789
-
792
+
790
793
  Note that if the record needs to be retrieved from the server, then the
791
- record instance returned from this method will not have any data yet.
794
+ record instance returned from this method will not have any data yet.
792
795
  Instead it will have a status of `SC.Record.READY_LOADING`. You can
793
796
  monitor the status property to be notified when the record data is
794
797
  available for you to use it.
795
-
798
+
796
799
  Find a Collection of Records
797
800
  ---
798
-
799
- If you pass only a record type or a query object, you can instead find
801
+
802
+ If you pass only a record type or a query object, you can instead find
800
803
  all records matching a specified set of conditions. When you call
801
804
  `find()` in this way, it will create a query if needed and pass it to the
802
805
  data source to fetch the results.
803
-
806
+
804
807
  If this is the first time you have fetched the query, then the store will
805
- automatically ask the data source to fetch any records related to it as
808
+ automatically ask the data source to fetch any records related to it as
806
809
  well. Otherwise you can refresh the query results at anytime by calling
807
810
  `refresh()` on the returned `RecordArray`.
808
811
 
809
- You can detect whether a RecordArray is fetching from the server by
812
+ You can detect whether a RecordArray is fetching from the server by
810
813
  checking its status.
811
-
814
+
812
815
  Examples
813
816
  ---
814
-
817
+
815
818
  Finding a single record:
816
-
819
+
817
820
  MyApp.store.find(MyApp.Contact, "23"); // returns MyApp.Contact
818
-
821
+
819
822
  Finding all records of a particular type:
820
-
823
+
821
824
  MyApp.store.find(MyApp.Contact); // returns SC.RecordArray of contacts
822
-
823
-
825
+
826
+
824
827
  Finding all contacts with first name John:
825
-
828
+
826
829
  var query = SC.Query.local(MyApp.Contact, "firstName = %@", "John");
827
830
  MyApp.store.find(query); // returns SC.RecordArray of contacts
828
-
831
+
829
832
  Finding all contacts using a remote query:
830
-
833
+
831
834
  var query = SC.Query.remote(MyApp.Contact);
832
835
  MyApp.store.find(query); // returns SC.RecordArray filled by server
833
-
836
+
834
837
  @param {SC.Record|String} recordType the expected record type
835
838
  @param {String} id the id to load
836
839
  @returns {SC.Record} record instance or null
837
840
  */
838
841
  find: function(recordType, id) {
839
-
842
+
840
843
  // if recordType is passed as string, find object
841
844
  if (SC.typeOf(recordType)===SC.T_STRING) {
842
845
  recordType = SC.objectForPropertyPath(recordType);
843
846
  }
844
-
847
+
845
848
  // handle passing a query...
846
849
  if ((arguments.length === 1) && !(recordType && recordType.get && recordType.get('isRecord'))) {
847
850
  if (!recordType) throw new Error("SC.Store#find() must pass recordType or query");
@@ -849,7 +852,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
849
852
  recordType = SC.Query.local(recordType);
850
853
  }
851
854
  return this._findQuery(recordType, YES, YES);
852
-
855
+
853
856
  // handle finding a single record
854
857
  } else {
855
858
  return this._findRecord(recordType, id);
@@ -858,36 +861,35 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
858
861
 
859
862
  /** @private
860
863
  DEPRECATED used find() instead.
861
-
864
+
862
865
  This method will accept a record type or query and return a record array
863
- matching the results. This method was commonly used prior to SproutCore
866
+ matching the results. This method was commonly used prior to SproutCore
864
867
  1.0. It has been deprecated in favor of a single `find()` method instead.
865
-
868
+
866
869
  For compatibility, this method will continue to work in SproutCore 1.0 but
867
- it will raise a warning. It will be removed in a future version of
870
+ it will raise a warning. It will be removed in a future version of
868
871
  SproutCore.
869
872
  */
870
873
  findAll: function(recordType, conditions, params) {
871
874
  SC.Logger.warn("SC.Store#findAll() will be removed in a future version of SproutCore. Use SC.Store#find() instead");
872
-
873
875
 
874
876
  if (!recordType || !recordType.isQuery) {
875
877
  recordType = SC.Query.local(recordType, conditions, params);
876
878
  }
877
-
879
+
878
880
  return this._findQuery(recordType, YES, YES);
879
881
  },
880
-
881
-
882
+
883
+
882
884
  _findQuery: function(query, createIfNeeded, refreshIfNew) {
883
885
 
884
886
  // lookup the local RecordArray for this query.
885
- var cache = this._scst_recordArraysByQuery,
887
+ var cache = this._scst_recordArraysByQuery,
886
888
  key = SC.guidFor(query),
887
889
  ret, ra ;
888
890
  if (!cache) cache = this._scst_recordArraysByQuery = {};
889
891
  ret = cache[key];
890
-
892
+
891
893
  // if a RecordArray was not found, then create one and also add it to the
892
894
  // list of record arrays to update.
893
895
  if (!ret && createIfNeeded) {
@@ -899,51 +901,51 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
899
901
 
900
902
  if (refreshIfNew) this.refreshQuery(query);
901
903
  }
902
-
904
+
903
905
  this.flush();
904
906
  return ret ;
905
907
  },
906
-
908
+
907
909
  _findRecord: function(recordType, id) {
908
910
 
909
- var storeKey ;
910
-
911
- // if a record instance is passed, simply use the storeKey. This allows
911
+ var storeKey ;
912
+
913
+ // if a record instance is passed, simply use the storeKey. This allows
912
914
  // you to pass a record from a chained store to get the same record in the
913
915
  // current store.
914
916
  if (recordType && recordType.get && recordType.get('isRecord')) {
915
917
  storeKey = recordType.get('storeKey');
916
-
917
- // otherwise, lookup the storeKey for the passed id. look in subclasses
918
+
919
+ // otherwise, lookup the storeKey for the passed id. look in subclasses
918
920
  // as well.
919
921
  } else storeKey = id ? recordType.storeKeyFor(id) : null;
920
-
922
+
921
923
  if (storeKey && (this.readStatus(storeKey) === SC.Record.EMPTY)) {
922
924
  storeKey = this.retrieveRecord(recordType, id);
923
925
  }
924
-
926
+
925
927
  // now we have the storeKey, materialize the record and return it.
926
928
  return storeKey ? this.materializeRecord(storeKey) : null ;
927
929
  },
928
930
 
929
931
  // ..........................................................
930
932
  // RECORD ARRAY OPERATIONS
931
- //
933
+ //
932
934
 
933
935
  /**
934
- Called by the record array just before it is destroyed. This will
936
+ Called by the record array just before it is destroyed. This will
935
937
  de-register it from receiving future notifications.
936
938
 
937
939
  You should never call this method yourself. Instead call `destroy()` on
938
940
  the `RecordArray` directly.
939
-
941
+
940
942
  @param {SC.RecordArray} recordArray the record array
941
943
  @returns {SC.Store} receiver
942
944
  */
943
945
  recordArrayWillDestroy: function(recordArray) {
944
946
  var cache = this._scst_recordArraysByQuery,
945
947
  set = this.get('recordArrays');
946
-
948
+
947
949
  if (cache) delete cache[SC.guidFor(recordArray.get('query'))];
948
950
  if (set) set.remove(recordArray);
949
951
  return this ;
@@ -956,7 +958,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
956
958
 
957
959
  You should never call this method yourself. Instead call `refresh()` on
958
960
  the `RecordArray` directly.
959
-
961
+
960
962
  @param {SC.Query} query the record array query to refresh
961
963
  @returns {SC.Store} receiver
962
964
  */
@@ -964,21 +966,21 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
964
966
  if (!query) throw new Error("refreshQuery() requires a query");
965
967
 
966
968
  var cache = this._scst_recordArraysByQuery,
967
- recArray = cache ? cache[SC.guidFor(query)] : null,
969
+ recArray = cache ? cache[SC.guidFor(query)] : null,
968
970
  source = this._getDataSource();
969
-
971
+
970
972
  if (source && source.fetch) {
971
973
  if (recArray) recArray.storeWillFetchQuery(query);
972
974
  source.fetch.call(source, this, query);
973
975
  }
974
-
975
- return this ;
976
+
977
+ return this ;
976
978
  },
977
-
978
- /** @private
979
+
980
+ /** @private
979
981
  Will ask all record arrays that have been returned from `findAll`
980
982
  with an `SC.Query` to check their arrays with the new `storeKey`s
981
-
983
+
982
984
  @param {SC.IndexSet} storeKeys set of storeKeys that changed
983
985
  @param {SC.Set} recordTypes
984
986
  @returns {SC.Store} receiver
@@ -990,29 +992,29 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
990
992
  recordArrays.forEach(function(recArray) {
991
993
  if (recArray) recArray.storeDidChangeStoreKeys(storeKeys, recordTypes);
992
994
  }, this);
993
-
995
+
994
996
  return this ;
995
997
  },
996
-
997
-
998
+
999
+
998
1000
  // ..........................................................
999
1001
  // LOW-LEVEL HELPERS
1000
- //
1001
-
1002
+ //
1003
+
1002
1004
  /**
1003
1005
  Array of all records currently in the store with the specified
1004
1006
  type. This method only reflects the actual records loaded into memory and
1005
1007
  therefore is not usually needed at runtime. However you will often use
1006
1008
  this method for testing.
1007
-
1009
+
1008
1010
  @param {SC.Record} recordType the record type
1009
1011
  @returns {SC.Array} array instance - usually SC.RecordArray
1010
1012
  */
1011
1013
  recordsFor: function(recordType) {
1012
- var storeKeys = [],
1014
+ var storeKeys = [],
1013
1015
  storeKeysById = recordType.storeKeysById(),
1014
1016
  id, storeKey, ret;
1015
-
1017
+
1016
1018
  // collect all non-empty store keys
1017
1019
  for(id in storeKeysById) {
1018
1020
  storeKey = storeKeysById[id]; // get the storeKey
@@ -1020,23 +1022,23 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1020
1022
  storeKeys.push(storeKey);
1021
1023
  }
1022
1024
  }
1023
-
1025
+
1024
1026
  if (storeKeys.length>0) {
1025
1027
  ret = SC.RecordArray.create({ store: this, storeKeys: storeKeys });
1026
1028
  } else ret = storeKeys; // empty array
1027
1029
  return ret ;
1028
1030
  },
1029
-
1031
+
1030
1032
  _TMP_REC_ATTRS: {},
1031
-
1032
- /**
1033
+
1034
+ /**
1033
1035
  Given a `storeKey`, return a materialized record. You will not usually
1034
1036
  call this method yourself. Instead it will used by other methods when
1035
1037
  you find records by id or perform other searches.
1036
1038
 
1037
1039
  If a `recordType` has been mapped to the storeKey, then a record instance
1038
1040
  will be returned even if the data hash has not been requested yet.
1039
-
1041
+
1040
1042
  Each Store instance returns unique record instances for each storeKey.
1041
1043
 
1042
1044
  @param {Number} storeKey The storeKey for the dataHash.
@@ -1044,33 +1046,33 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1044
1046
  */
1045
1047
  materializeRecord: function(storeKey) {
1046
1048
  var records = this.records, ret, recordType, attrs;
1047
-
1049
+
1048
1050
  // look up in cached records
1049
1051
  if (!records) records = this.records = {}; // load cached records
1050
1052
  ret = records[storeKey];
1051
1053
  if (ret) return ret;
1052
-
1054
+
1053
1055
  // not found -- OK, create one then.
1054
1056
  recordType = SC.Store.recordTypeFor(storeKey);
1055
1057
  if (!recordType) return null; // not recordType registered, nothing to do
1056
-
1058
+
1057
1059
  attrs = this._TMP_REC_ATTRS ;
1058
1060
  attrs.storeKey = storeKey ;
1059
1061
  attrs.store = this ;
1060
1062
  ret = records[storeKey] = recordType.create(attrs);
1061
-
1063
+
1062
1064
  return ret ;
1063
1065
  },
1064
1066
 
1065
1067
  // ..........................................................
1066
1068
  // CORE RECORDS API
1067
- //
1068
- // The methods in this section can be used to manipulate records without
1069
+ //
1070
+ // The methods in this section can be used to manipulate records without
1069
1071
  // actually creating record instances.
1070
-
1072
+
1071
1073
  /**
1072
1074
  Creates a new record instance with the passed `recordType` and `dataHash`.
1073
- You can also optionally specify an id or else it will be pulled from the
1075
+ You can also optionally specify an id or else it will be pulled from the
1074
1076
  data hash.
1075
1077
 
1076
1078
  Note that the record will not yet be saved back to the server. To save
@@ -1085,8 +1087,8 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1085
1087
  createRecord: function(recordType, dataHash, id) {
1086
1088
  var primaryKey, storeKey, status, K = SC.Record, changelog, defaultVal,
1087
1089
  ret;
1088
-
1089
- // First, try to get an id. If no id is passed, look it up in the
1090
+
1091
+ // First, try to get an id. If no id is passed, look it up in the
1090
1092
  // dataHash.
1091
1093
  if (!id && (primaryKey = recordType.prototype.primaryKey)) {
1092
1094
  id = dataHash[primaryKey];
@@ -1097,57 +1099,57 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1097
1099
  id = dataHash[primaryKey] = defaultVal();
1098
1100
  }
1099
1101
  }
1100
-
1102
+
1101
1103
  // Next get the storeKey - base on id if available
1102
1104
  storeKey = id ? recordType.storeKeyFor(id) : SC.Store.generateStoreKey();
1103
-
1105
+
1104
1106
  // now, check the state and do the right thing.
1105
1107
  status = this.readStatus(storeKey);
1106
-
1108
+
1107
1109
  // check state
1108
1110
  // any busy or ready state or destroyed dirty state is not allowed
1109
- if ((status & K.BUSY) ||
1110
- (status & K.READY) ||
1111
- (status === K.DESTROYED_DIRTY)) {
1111
+ if ((status & K.BUSY) ||
1112
+ (status & K.READY) ||
1113
+ (status === K.DESTROYED_DIRTY)) {
1112
1114
  throw id ? K.RECORD_EXISTS_ERROR : K.BAD_STATE_ERROR;
1113
-
1115
+
1114
1116
  // allow error or destroyed state only with id
1115
1117
  } else if (!id && (status===SC.DESTROYED_CLEAN || status===SC.ERROR)) {
1116
1118
  throw K.BAD_STATE_ERROR;
1117
1119
  }
1118
-
1120
+
1119
1121
  // add dataHash and setup initial status -- also save recordType
1120
1122
  this.writeDataHash(storeKey, (dataHash ? dataHash : {}), K.READY_NEW);
1121
-
1123
+
1122
1124
  SC.Store.replaceRecordTypeFor(storeKey, recordType);
1123
1125
  this.dataHashDidChange(storeKey);
1124
-
1126
+
1125
1127
  // Record is now in a committable state -- add storeKey to changelog
1126
1128
  changelog = this.changelog;
1127
1129
  if (!changelog) changelog = SC.Set.create();
1128
1130
  changelog.add(storeKey);
1129
1131
  this.changelog = changelog;
1130
-
1132
+
1131
1133
  // if commit records is enabled
1132
1134
  if(this.get('commitRecordsAutomatically')){
1133
1135
  this.invokeLast(this.commitRecords);
1134
1136
  }
1135
-
1137
+
1136
1138
  // Finally return materialized record, after we propagate the status to
1137
1139
  // any aggregrate records.
1138
1140
  ret = this.materializeRecord(storeKey);
1139
1141
  if (ret) ret.propagateToAggregates();
1140
1142
  return ret;
1141
- },
1142
-
1143
+ },
1144
+
1143
1145
  /**
1144
- Creates an array of new records. You must pass an array of `dataHash`es
1146
+ Creates an array of new records. You must pass an array of `dataHash`es
1145
1147
  plus a `recordType` and, optionally, an array of ids. This will create an
1146
1148
  array of record instances with the same record type.
1147
-
1149
+
1148
1150
  If you need to instead create a bunch of records with different data types
1149
1151
  you can instead pass an array of `recordType`s, one for each data hash.
1150
-
1152
+
1151
1153
  @param {SC.Record|Array} recordTypes class or array of classes
1152
1154
  @param {Array} dataHashes array of data hashes
1153
1155
  @param {Array} ids (optional) ids to assign to records
@@ -1164,14 +1166,14 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1164
1166
  }
1165
1167
  return ret ;
1166
1168
  },
1167
-
1168
-
1169
+
1170
+
1169
1171
  /**
1170
- Unloads a record, removing the data hash from the store. If you try to
1171
- unload a record that is already destroyed then this method will have no effect.
1172
- If you unload a record that does not exist or an error then an exception
1172
+ Unloads a record, removing the data hash from the store. If you try to
1173
+ unload a record that is already destroyed then this method will have no effect.
1174
+ If you unload a record that does not exist or an error then an exception
1173
1175
  will be raised.
1174
-
1176
+
1175
1177
  @param {SC.Record} recordType the recordType
1176
1178
  @param {String} id the record id
1177
1179
  @param {Number} storeKey (optional) if passed, ignores recordType and id
@@ -1184,35 +1186,35 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1184
1186
  // handle status - ignore if destroying or destroyed
1185
1187
  if ((status === K.BUSY_DESTROYING) || (status & K.DESTROYED)) {
1186
1188
  return this; // nothing to do
1187
-
1189
+
1188
1190
  // error out if empty
1189
1191
  } else if (status & K.BUSY) {
1190
1192
  throw K.BUSY_ERROR ;
1191
-
1193
+
1192
1194
  // otherwise, destroy in dirty state
1193
1195
  } else status = newStatus ;
1194
-
1196
+
1195
1197
  // remove the data hash, set new status
1196
1198
  this.removeDataHash(storeKey, status);
1197
1199
  this.dataHashDidChange(storeKey);
1198
-
1200
+
1199
1201
  // Handle all the child Records
1200
1202
  var that = this;
1201
1203
  this._propagateToChildren(storeKey, function(storeKey){
1202
1204
  that.unloadRecord(null, null, storeKey, newStatus);
1203
1205
  });
1204
-
1206
+
1205
1207
  return this ;
1206
1208
  },
1207
-
1209
+
1208
1210
  /**
1209
1211
  Unloads a group of records. If you have a set of record ids, unloading
1210
- them this way can be faster than retrieving each record and unloading
1212
+ them this way can be faster than retrieving each record and unloading
1211
1213
  it individually.
1212
1214
 
1213
1215
  You can pass either a single `recordType` or an array of `recordType`s. If
1214
1216
  you pass a single `recordType`, then the record type will be used for each
1215
- record. If you pass an array, then each id must have a matching record
1217
+ record. If you pass an array, then each id must have a matching record
1216
1218
  type in the array.
1217
1219
 
1218
1220
  You can optionally pass an array of `storeKey`s instead of the `recordType`
@@ -1259,10 +1261,10 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1259
1261
 
1260
1262
  /**
1261
1263
  Destroys a record, removing the data hash from the store and adding the
1262
- record to the destroyed changelog. If you try to destroy a record that is
1263
- already destroyed then this method will have no effect. If you destroy a
1264
+ record to the destroyed changelog. If you try to destroy a record that is
1265
+ already destroyed then this method will have no effect. If you destroy a
1264
1266
  record that does not exist or an error then an exception will be raised.
1265
-
1267
+
1266
1268
  @param {SC.Record} recordType the recordType
1267
1269
  @param {String} id the record id
1268
1270
  @param {Number} storeKey (optional) if passed, ignores recordType and id
@@ -1275,22 +1277,22 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1275
1277
  // handle status - ignore if destroying or destroyed
1276
1278
  if ((status === K.BUSY_DESTROYING) || (status & K.DESTROYED)) {
1277
1279
  return this; // nothing to do
1278
-
1280
+
1279
1281
  // error out if empty
1280
1282
  } else if (status === K.EMPTY) {
1281
1283
  throw K.NOT_FOUND_ERROR ;
1282
-
1284
+
1283
1285
  // error out if busy
1284
1286
  } else if (status & K.BUSY) {
1285
1287
  throw K.BUSY_ERROR ;
1286
-
1288
+
1287
1289
  // if new status, destroy but leave in clean state
1288
1290
  } else if (status === K.READY_NEW) {
1289
1291
  status = K.DESTROYED_CLEAN ;
1290
-
1292
+
1291
1293
  // otherwise, destroy in dirty state
1292
1294
  } else status = K.DESTROYED_DIRTY ;
1293
-
1295
+
1294
1296
  // remove the data hash, set new status
1295
1297
  this.writeStatus(storeKey, status);
1296
1298
  this.dataHashDidChange(storeKey);
@@ -1306,30 +1308,30 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1306
1308
  if(this.get('commitRecordsAutomatically')){
1307
1309
  this.invokeLast(this.commitRecords);
1308
1310
  }
1309
-
1311
+
1310
1312
  var that = this;
1311
1313
  this._propagateToChildren(storeKey, function(storeKey){
1312
1314
  that.destroyRecord(null, null, storeKey);
1313
1315
  });
1314
-
1316
+
1315
1317
  return this ;
1316
1318
  },
1317
-
1319
+
1318
1320
  /**
1319
1321
  Destroys a group of records. If you have a set of record ids, destroying
1320
- them this way can be faster than retrieving each record and destroying
1322
+ them this way can be faster than retrieving each record and destroying
1321
1323
  it individually.
1322
-
1324
+
1323
1325
  You can pass either a single `recordType` or an array of `recordType`s. If
1324
1326
  you pass a single `recordType`, then the record type will be used for each
1325
- record. If you pass an array, then each id must have a matching record
1327
+ record. If you pass an array, then each id must have a matching record
1326
1328
  type in the array.
1327
1329
 
1328
1330
  You can optionally pass an array of `storeKey`s instead of the `recordType`
1329
1331
  and ids. In this case the first two parameters will be ignored. This
1330
1332
  is usually only used by low-level internal methods. You will not usually
1331
1333
  destroy records this way.
1332
-
1334
+
1333
1335
  @param {SC.Record|Array} recordTypes class or array of classes
1334
1336
  @param {Array} ids ids to destroy
1335
1337
  @param {Array} storeKeys (optional) store keys to destroy
@@ -1355,7 +1357,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1355
1357
  }
1356
1358
  return this ;
1357
1359
  },
1358
-
1360
+
1359
1361
  /**
1360
1362
  register a Child Record to the parent
1361
1363
  */
@@ -1380,7 +1382,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1380
1382
  this.childRecords = crs;
1381
1383
  this.parentRecords = prs;
1382
1384
  },
1383
-
1385
+
1384
1386
  /**
1385
1387
  materialize the parent when passing in a store key for the child
1386
1388
  */
@@ -1390,10 +1392,10 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1390
1392
  crs = this.childRecords;
1391
1393
  pk = crs ? this.childRecords[childStoreKey] : null ;
1392
1394
  if (SC.none(pk)) return null;
1393
-
1395
+
1394
1396
  return this.materializeRecord(pk);
1395
1397
  },
1396
-
1398
+
1397
1399
  /**
1398
1400
  function for retrieving a parent record key
1399
1401
 
@@ -1404,7 +1406,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1404
1406
  var crs = this.childRecords || {};
1405
1407
  return crs[storeKey];
1406
1408
  },
1407
-
1409
+
1408
1410
  /**
1409
1411
  function that propagates a function call to all children
1410
1412
  */
@@ -1417,12 +1419,12 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1417
1419
  if (children.hasOwnProperty(key)) func(key);
1418
1420
  }
1419
1421
  },
1420
-
1422
+
1421
1423
  /**
1422
1424
  Notes that the data for the given record id has changed. The record will
1423
1425
  be committed to the server the next time you commit the root store. Only
1424
1426
  call this method on a record in a READY state of some type.
1425
-
1427
+
1426
1428
  @param {SC.Record} recordType the recordType
1427
1429
  @param {String} id the record id
1428
1430
  @param {Number} storeKey (optional) if passed, ignores recordType and id
@@ -1433,55 +1435,55 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1433
1435
  recordDidChange: function(recordType, id, storeKey, key, statusOnly) {
1434
1436
  if (storeKey === undefined) storeKey = recordType.storeKeyFor(id);
1435
1437
  var status = this.readStatus(storeKey), changelog, K = SC.Record;
1436
-
1438
+
1437
1439
  // BUSY_LOADING, BUSY_CREATING, BUSY_COMMITTING, BUSY_REFRESH_CLEAN
1438
1440
  // BUSY_REFRESH_DIRTY, BUSY_DESTROYING
1439
1441
  if (status & K.BUSY) {
1440
1442
  throw K.BUSY_ERROR ;
1441
-
1443
+
1442
1444
  // if record is not in ready state, then it is not found.
1443
1445
  // ERROR, EMPTY, DESTROYED_CLEAN, DESTROYED_DIRTY
1444
1446
  } else if (!(status & K.READY)) {
1445
1447
  throw K.NOT_FOUND_ERROR ;
1446
-
1448
+
1447
1449
  // otherwise, make new status READY_DIRTY unless new.
1448
1450
  // K.READY_CLEAN, K.READY_DIRTY, ignore K.READY_NEW
1449
1451
  } else {
1450
1452
  if (status != K.READY_NEW) this.writeStatus(storeKey, K.READY_DIRTY);
1451
1453
  }
1452
-
1454
+
1453
1455
  // record data hash change
1454
1456
  this.dataHashDidChange(storeKey, null, statusOnly, key);
1455
-
1457
+
1456
1458
  // record in changelog
1457
1459
  changelog = this.changelog ;
1458
1460
  if (!changelog) changelog = this.changelog = SC.Set.create() ;
1459
1461
  changelog.add(storeKey);
1460
1462
  this.changelog = changelog;
1461
-
1463
+
1462
1464
  // if commit records is enabled
1463
1465
  if(this.get('commitRecordsAutomatically')){
1464
1466
  this.invokeLast(this.commitRecords);
1465
1467
  }
1466
-
1468
+
1467
1469
  return this ;
1468
1470
  },
1469
1471
 
1470
1472
  /**
1471
1473
  Mark a group of records as dirty. The records will be committed to the
1472
- server the next time you commit changes on the root store. If you have a
1473
- set of record ids, marking them dirty this way can be faster than
1474
+ server the next time you commit changes on the root store. If you have a
1475
+ set of record ids, marking them dirty this way can be faster than
1474
1476
  retrieving each record and destroying it individually.
1475
-
1477
+
1476
1478
  You can pass either a single `recordType` or an array of `recordType`s. If
1477
1479
  you pass a single `recordType`, then the record type will be used for each
1478
- record. If you pass an array, then each id must have a matching record
1480
+ record. If you pass an array, then each id must have a matching record
1479
1481
  type in the array.
1480
1482
 
1481
1483
  You can optionally pass an array of `storeKey`s instead of the `recordType`
1482
1484
  and ids. In this case the first two parameters will be ignored. This
1483
- is usually only used by low-level internal methods.
1484
-
1485
+ is usually only used by low-level internal methods.
1486
+
1485
1487
  @param {SC.Record|Array} recordTypes class or array of classes
1486
1488
  @param {Array} ids ids to destroy
1487
1489
  @param {Array} storeKeys (optional) store keys to destroy
@@ -1506,20 +1508,20 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1506
1508
  this.recordDidChange(undefined, undefined, storeKey);
1507
1509
  }
1508
1510
  }
1509
- return this ;
1511
+ return this ;
1510
1512
  },
1511
1513
 
1512
1514
  /**
1513
- Retrieves a set of records from the server. If the records has
1514
- already been loaded in the store, then this method will simply return.
1515
- Otherwise if your store has a `dataSource`, this will call the
1516
- `dataSource` to retrieve the record. Generally you will not need to
1515
+ Retrieves a set of records from the server. If the records has
1516
+ already been loaded in the store, then this method will simply return.
1517
+ Otherwise if your store has a `dataSource`, this will call the
1518
+ `dataSource` to retrieve the record. Generally you will not need to
1517
1519
  call this method yourself. Instead you can just use `find()`.
1518
-
1519
- This will not actually create a record instance but it will initiate a
1520
- load of the record from the server. You can subsequently get a record
1520
+
1521
+ This will not actually create a record instance but it will initiate a
1522
+ load of the record from the server. You can subsequently get a record
1521
1523
  instance itself using `materializeRecord()`.
1522
-
1524
+
1523
1525
  @param {SC.Record|Array} recordTypes class or array of classes
1524
1526
  @param {Array} ids ids to retrieve
1525
1527
  @param {Array} storeKeys (optional) store keys to retrieve
@@ -1528,7 +1530,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1528
1530
  @returns {Array} storeKeys to be retrieved
1529
1531
  */
1530
1532
  retrieveRecords: function(recordTypes, ids, storeKeys, isRefresh, callbacks) {
1531
-
1533
+
1532
1534
  var source = this._getDataSource(),
1533
1535
  isArray = SC.typeOf(recordTypes) === SC.T_ARRAY,
1534
1536
  hasCallbackArray = SC.typeOf(callbacks) === SC.T_ARRAY,
@@ -1537,12 +1539,12 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1537
1539
  rev = SC.Store.generateStoreKey(),
1538
1540
  K = SC.Record,
1539
1541
  recordType, idx, storeKey, status, ok, callback;
1540
-
1542
+
1541
1543
  if (!isArray) recordType = recordTypes;
1542
-
1544
+
1543
1545
  // if no storeKeys were passed, map recordTypes + ids
1544
1546
  for(idx=0;idx<len;idx++) {
1545
-
1547
+
1546
1548
  // collect store key
1547
1549
  if (storeKeys) {
1548
1550
  storeKey = storeKeys[idx];
@@ -1552,10 +1554,10 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1552
1554
  }
1553
1555
  //collect the callback
1554
1556
  callback = hasCallbackArray ? callbacks[idx] : callbacks;
1555
-
1557
+
1556
1558
  // collect status and process
1557
1559
  status = this.readStatus(storeKey);
1558
-
1560
+
1559
1561
  // K.EMPTY, K.ERROR, K.DESTROYED_CLEAN - initial retrieval
1560
1562
  if ((status == K.EMPTY) || (status == K.ERROR) || (status == K.DESTROYED_CLEAN)) {
1561
1563
  this.writeStatus(storeKey, K.BUSY_LOADING);
@@ -1577,19 +1579,19 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1577
1579
  // K.DESTROY_DIRTY, bad state...
1578
1580
  } else if (status == K.DESTROYED_DIRTY) {
1579
1581
  throw K.BAD_STATE_ERROR ;
1580
-
1582
+
1581
1583
  // ignore K.BUSY_LOADING, K.BUSY_REFRESH_CLEAN, K.BUSY_REFRESH_DIRTY
1582
1584
  }
1583
1585
  }
1584
1586
  }
1585
-
1587
+
1586
1588
  // now retrieve storekeys from dataSource. if there is no dataSource,
1587
1589
  // then act as if we couldn't retrieve.
1588
1590
  ok = NO;
1589
1591
  if (source) ok = source.retrieveRecords.call(source, this, ret, ids);
1590
1592
 
1591
1593
  // if the data source could not retrieve or if there is no source, then
1592
- // simulate the data source calling dataSourceDidError on those we are
1594
+ // simulate the data source calling dataSourceDidError on those we are
1593
1595
  // loading for the first time or dataSourceDidComplete on refreshes.
1594
1596
  if (!ok) {
1595
1597
  len = ret.length;
@@ -1600,7 +1602,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1600
1602
  if (status === K.BUSY_LOADING) {
1601
1603
  this.writeStatus(storeKey, K.ERROR);
1602
1604
  this.dataHashDidChange(storeKey, rev, YES);
1603
-
1605
+
1604
1606
  } else if (status & K.BUSY_REFRESH) {
1605
1607
  this.writeStatus(storeKey, K.READY | (status & 0x03));
1606
1608
  this.dataHashDidChange(storeKey, rev, YES);
@@ -1610,11 +1612,11 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1610
1612
  }
1611
1613
  return ret ;
1612
1614
  },
1613
-
1615
+
1614
1616
  _TMP_RETRIEVE_ARRAY: [],
1615
-
1617
+
1616
1618
  _callback_queue: {},
1617
-
1619
+
1618
1620
  /**
1619
1621
  @private
1620
1622
  stores the callbacks for the storeKeys that are inflight
@@ -1651,14 +1653,14 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1651
1653
  delete queue[key];
1652
1654
  });
1653
1655
  }
1654
-
1656
+
1655
1657
  }
1656
1658
  }
1657
1659
  },
1658
-
1660
+
1659
1661
  /*
1660
1662
  @private
1661
-
1663
+
1662
1664
  */
1663
1665
  _cancelCallback: function(storeKey){
1664
1666
  var queue = this._callback_queue;
@@ -1666,17 +1668,17 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1666
1668
  delete queue[storeKey];
1667
1669
  }
1668
1670
  },
1669
-
1670
-
1671
+
1672
+
1671
1673
  /**
1672
1674
  Retrieves a record from the server. If the record has already been loaded
1673
- in the store, then this method will simply return. Otherwise if your
1674
- store has a `dataSource`, this will call the `dataSource` to retrieve the
1675
- record. Generally you will not need to call this method yourself.
1675
+ in the store, then this method will simply return. Otherwise if your
1676
+ store has a `dataSource`, this will call the `dataSource` to retrieve the
1677
+ record. Generally you will not need to call this method yourself.
1676
1678
  Instead you can just use `find()`.
1677
-
1678
- This will not actually create a record instance but it will initiate a
1679
- load of the record from the server. You can subsequently get a record
1679
+
1680
+ This will not actually create a record instance but it will initiate a
1681
+ load of the record from the server. You can subsequently get a record
1680
1682
  instance itself using `materializeRecord()`.
1681
1683
 
1682
1684
  @param {SC.Record} recordType class
@@ -1684,12 +1686,12 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1684
1686
  @param {Number} storeKey (optional) store key
1685
1687
  @param {Boolean} isRefresh
1686
1688
  @param {Function} callback (optional)
1687
- @returns {Number} storeKey that was retrieved
1689
+ @returns {Number} storeKey that was retrieved
1688
1690
  */
1689
1691
  retrieveRecord: function(recordType, id, storeKey, isRefresh, callback) {
1690
1692
  var array = this._TMP_RETRIEVE_ARRAY,
1691
1693
  ret;
1692
-
1694
+
1693
1695
  if (storeKey) {
1694
1696
  array[0] = storeKey;
1695
1697
  storeKey = array;
@@ -1698,7 +1700,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1698
1700
  array[0] = id;
1699
1701
  id = array;
1700
1702
  }
1701
-
1703
+
1702
1704
  ret = this.retrieveRecords(recordType, id, storeKey, isRefresh, callback);
1703
1705
  array.length = 0 ;
1704
1706
  return ret[0];
@@ -1708,7 +1710,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1708
1710
  Refreshes a record from the server. If the record has already been loaded
1709
1711
  in the store, then this method will request a refresh from the
1710
1712
  `dataSource`. Otherwise it will attempt to retrieve the record.
1711
-
1713
+
1712
1714
  @param {String} id to id of the record to load
1713
1715
  @param {SC.Record} recordType the expected record type
1714
1716
  @param {Number} storeKey (optional) optional store key
@@ -1723,7 +1725,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1723
1725
  Refreshes a set of records from the server. If the records has already been loaded
1724
1726
  in the store, then this method will request a refresh from the
1725
1727
  `dataSource`. Otherwise it will attempt to retrieve them.
1726
-
1728
+
1727
1729
  @param {SC.Record|Array} recordTypes class or array of classes
1728
1730
  @param {Array} ids ids to destroy
1729
1731
  @param {Array} storeKeys (optional) store keys to destroy
@@ -1734,29 +1736,29 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1734
1736
  var ret = this.retrieveRecords(recordTypes, ids, storeKeys, YES, callback);
1735
1737
  return ret && ret.length>0;
1736
1738
  },
1737
-
1739
+
1738
1740
  /**
1739
1741
  Commits the passed store keys or ids. If no `storeKey`s are given,
1740
- it will commit any records in the changelog.
1741
-
1742
- Based on the current state of the record, this will ask the data
1742
+ it will commit any records in the changelog.
1743
+
1744
+ Based on the current state of the record, this will ask the data
1743
1745
  source to perform the appropriate actions
1744
1746
  on the store keys.
1745
-
1747
+
1746
1748
  @param {Array} recordTypes the expected record types (SC.Record)
1747
1749
  @param {Array} ids to commit
1748
1750
  @param {SC.Set} storeKeys to commit
1749
1751
  @param {Hash} params optional additional parameters to pass along to the
1750
1752
  data source
1751
1753
  @param {Function|Array} callback function or array of callbacks
1752
-
1754
+
1753
1755
  @returns {Boolean} if the action was succesful.
1754
1756
  */
1755
1757
  commitRecords: function(recordTypes, ids, storeKeys, params, callbacks) {
1756
1758
  var source = this._getDataSource(),
1757
1759
  isArray = SC.typeOf(recordTypes) === SC.T_ARRAY,
1758
- hasCallbackArray = SC.typeOf(callbacks) === SC.T_ARRAY,
1759
- retCreate= [], retUpdate= [], retDestroy = [],
1760
+ hasCallbackArray = SC.typeOf(callbacks) === SC.T_ARRAY,
1761
+ retCreate= [], retUpdate= [], retDestroy = [],
1760
1762
  rev = SC.Store.generateStoreKey(),
1761
1763
  K = SC.Record,
1762
1764
  recordType, idx, storeKey, status, key, ret, len, callback;
@@ -1769,9 +1771,9 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1769
1771
  }
1770
1772
 
1771
1773
  len = storeKeys ? storeKeys.get('length') : (ids ? ids.get('length') : 0);
1772
-
1774
+
1773
1775
  for(idx=0;idx<len;idx++) {
1774
-
1776
+
1775
1777
  // collect store key
1776
1778
  if (storeKeys) {
1777
1779
  storeKey = storeKeys[idx];
@@ -1780,16 +1782,16 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1780
1782
  else recordType = recordTypes;
1781
1783
  storeKey = recordType.storeKeyFor(ids[idx]);
1782
1784
  }
1783
-
1785
+
1784
1786
  //collect the callback
1785
1787
  callback = hasCallbackArray ? callbacks[idx] : callbacks;
1786
-
1788
+
1787
1789
  // collect status and process
1788
1790
  status = this.readStatus(storeKey);
1789
-
1791
+
1790
1792
  if ((status == K.EMPTY) || (status == K.ERROR)) {
1791
1793
  throw K.NOT_FOUND_ERROR ;
1792
- }
1794
+ }
1793
1795
  else {
1794
1796
  if(status==K.READY_NEW) {
1795
1797
  this.writeStatus(storeKey, K.BUSY_CREATING);
@@ -1809,16 +1811,16 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1809
1811
  } else if (status==K.DESTROYED_CLEAN) {
1810
1812
  this.dataHashDidChange(storeKey, rev, YES);
1811
1813
  }
1812
- // ignore K.READY_CLEAN, K.BUSY_LOADING, K.BUSY_CREATING, K.BUSY_COMMITTING,
1814
+ // ignore K.READY_CLEAN, K.BUSY_LOADING, K.BUSY_CREATING, K.BUSY_COMMITTING,
1813
1815
  // K.BUSY_REFRESH_CLEAN, K_BUSY_REFRESH_DIRTY, KBUSY_DESTROYING
1814
1816
  }
1815
1817
  }
1816
-
1818
+
1817
1819
  // now commit storekeys to dataSource
1818
1820
  if (source && (len>0 || params)) {
1819
1821
  ret = source.commitRecords.call(source, this, retCreate, retUpdate, retDestroy, params);
1820
1822
  }
1821
-
1823
+
1822
1824
  //remove all commited changes from changelog
1823
1825
  if (ret && !recordTypes && !ids) {
1824
1826
  if (storeKeys === this.changelog) {
@@ -1832,13 +1834,13 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1832
1834
  },
1833
1835
 
1834
1836
  /**
1835
- Commits the passed store key or id. Based on the current state of the
1837
+ Commits the passed store key or id. Based on the current state of the
1836
1838
  record, this will ask the data source to perform the appropriate action
1837
1839
  on the store key.
1838
-
1839
- You have to pass either the id or the storeKey otherwise it will return
1840
+
1841
+ You have to pass either the id or the storeKey otherwise it will return
1840
1842
  NO.
1841
-
1843
+
1842
1844
  @param {SC.Record} recordType the expected record type
1843
1845
  @param {String} id the id of the record to commit
1844
1846
  @param {Number} storeKey the storeKey of the record to commit
@@ -1859,17 +1861,17 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1859
1861
  array[0] = id;
1860
1862
  id = array;
1861
1863
  }
1862
-
1864
+
1863
1865
  ret = this.commitRecords(recordType, id, storeKey, params, callback);
1864
1866
  array.length = 0 ;
1865
1867
  return ret;
1866
1868
  },
1867
-
1869
+
1868
1870
  /**
1869
- Cancels an inflight request for the passed records. Depending on the
1870
- server implementation, this could cancel an entire request, causing
1871
+ Cancels an inflight request for the passed records. Depending on the
1872
+ server implementation, this could cancel an entire request, causing
1871
1873
  other records to also transition their current state.
1872
-
1874
+
1873
1875
  @param {SC.Record|Array} recordTypes class or array of classes
1874
1876
  @param {Array} ids ids to destroy
1875
1877
  @param {Array} storeKeys (optional) store keys to destroy
@@ -1881,22 +1883,22 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1881
1883
  K = SC.Record,
1882
1884
  ret = [],
1883
1885
  status, len, idx, id, recordType, storeKey;
1884
-
1886
+
1885
1887
  len = (storeKeys === undefined) ? ids.length : storeKeys.length;
1886
1888
  for(idx=0;idx<len;idx++) {
1887
1889
  if (isArray) recordType = recordTypes[idx] || SC.Record;
1888
1890
  else recordType = recordTypes || SC.Record;
1889
-
1891
+
1890
1892
  id = ids ? ids[idx] : undefined ;
1891
-
1893
+
1892
1894
  if(storeKeys===undefined){
1893
1895
  storeKey = recordType.storeKeyFor(id);
1894
1896
  }else{
1895
- storeKey = storeKeys ? storeKeys[idx] : undefined ;
1897
+ storeKey = storeKeys ? storeKeys[idx] : undefined ;
1896
1898
  }
1897
1899
  if(storeKey) {
1898
1900
  status = this.readStatus(storeKey);
1899
-
1901
+
1900
1902
  if ((status == K.EMPTY) || (status == K.ERROR)) {
1901
1903
  throw K.NOT_FOUND_ERROR ;
1902
1904
  }
@@ -1904,17 +1906,17 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1904
1906
  this._cancelCallback(storeKey);
1905
1907
  }
1906
1908
  }
1907
-
1909
+
1908
1910
  if (source) source.cancel.call(source, this, ret);
1909
-
1911
+
1910
1912
  return this ;
1911
1913
  },
1912
1914
 
1913
1915
  /**
1914
- Cancels an inflight request for the passed record. Depending on the
1915
- server implementation, this could cancel an entire request, causing
1916
+ Cancels an inflight request for the passed record. Depending on the
1917
+ server implementation, this could cancel an entire request, causing
1916
1918
  other records to also transition their current state.
1917
-
1919
+
1918
1920
  @param {SC.Record|Array} recordTypes class or array of classes
1919
1921
  @param {Array} ids ids to destroy
1920
1922
  @param {Array} storeKeys (optional) store keys to destroy
@@ -1923,7 +1925,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1923
1925
  cancelRecord: function(recordType, id, storeKey) {
1924
1926
  var array = this._TMP_RETRIEVE_ARRAY,
1925
1927
  ret ;
1926
-
1928
+
1927
1929
  if (storeKey !== undefined) {
1928
1930
  array[0] = storeKey;
1929
1931
  storeKey = array;
@@ -1932,32 +1934,32 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1932
1934
  array[0] = id;
1933
1935
  id = array;
1934
1936
  }
1935
-
1937
+
1936
1938
  ret = this.cancelRecords(recordType, id, storeKey);
1937
1939
  array.length = 0 ;
1938
1940
  return this;
1939
1941
  },
1940
1942
 
1941
- /**
1942
- Convenience method can be called by the store or other parts of your
1943
+ /**
1944
+ Convenience method can be called by the store or other parts of your
1943
1945
  application to load a record into the store. This method will take a
1944
- recordType and a data hashes and either add or update the
1945
- record in the store.
1946
-
1946
+ recordType and a data hashes and either add or update the
1947
+ record in the store.
1948
+
1947
1949
  The loaded records will be in an `SC.Record.READY_CLEAN` state, indicating
1948
- they were loaded from the data source and do not need to be committed
1950
+ they were loaded from the data source and do not need to be committed
1949
1951
  back before changing.
1950
-
1951
- This method will check the state of the storeKey and call either
1952
- `pushRetrieve()` or `dataSourceDidComplete()`. The standard state constraints
1952
+
1953
+ This method will check the state of the storeKey and call either
1954
+ `pushRetrieve()` or `dataSourceDidComplete()`. The standard state constraints
1953
1955
  for these methods apply here.
1954
-
1956
+
1955
1957
  The return value will be the `storeKey` used for the push. This is often
1956
1958
  convenient to pass into `loadQuery()`, if you are fetching a remote query.
1957
-
1958
- If you are upgrading from a pre SproutCore 1.0 application, this method
1959
+
1960
+ If you are upgrading from a pre SproutCore 1.0 application, this method
1959
1961
  is the closest to the old `updateRecord()`.
1960
-
1962
+
1961
1963
  @param {SC.Record} recordType the record type
1962
1964
  @param {Array} dataHash to update
1963
1965
  @param {Array} id optional. if not passed lookup on the hash
@@ -1966,44 +1968,44 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
1966
1968
  loadRecord: function(recordType, dataHash, id) {
1967
1969
  var K = SC.Record,
1968
1970
  ret, primaryKey, storeKey;
1969
-
1971
+
1970
1972
  // save lookup info
1971
1973
  recordType = recordType || SC.Record;
1972
1974
  primaryKey = recordType.prototype.primaryKey;
1973
-
1974
-
1975
+
1976
+
1975
1977
  // push each record
1976
1978
  id = id || dataHash[primaryKey];
1977
1979
  ret = storeKey = recordType.storeKeyFor(id); // needed to cache
1978
-
1980
+
1979
1981
  if (this.readStatus(storeKey) & K.BUSY) {
1980
1982
  this.dataSourceDidComplete(storeKey, dataHash, id);
1981
1983
  } else this.pushRetrieve(recordType, id, dataHash, storeKey);
1982
-
1984
+
1983
1985
  // return storeKey
1984
1986
  return ret ;
1985
1987
  },
1986
-
1987
- /**
1988
- Convenience method can be called by the store or other parts of your
1988
+
1989
+ /**
1990
+ Convenience method can be called by the store or other parts of your
1989
1991
  application to load records into the store. This method will take a
1990
- recordType and an array of data hashes and either add or update the
1991
- record in the store.
1992
-
1992
+ recordType and an array of data hashes and either add or update the
1993
+ record in the store.
1994
+
1993
1995
  The loaded records will be in an `SC.Record.READY_CLEAN` state, indicating
1994
- they were loaded from the data source and do not need to be committed
1996
+ they were loaded from the data source and do not need to be committed
1995
1997
  back before changing.
1996
-
1997
- This method will check the state of each storeKey and call either
1998
- `pushRetrieve()` or `dataSourceDidComplete()`. The standard state
1998
+
1999
+ This method will check the state of each storeKey and call either
2000
+ `pushRetrieve()` or `dataSourceDidComplete()`. The standard state
1999
2001
  constraints for these methods apply here.
2000
-
2002
+
2001
2003
  The return value will be the storeKeys used for each push. This is often
2002
2004
  convenient to pass into `loadQuery()`, if you are fetching a remote query.
2003
-
2004
- If you are upgrading from a pre SproutCore 1.0 application, this method
2005
+
2006
+ If you are upgrading from a pre SproutCore 1.0 application, this method
2005
2007
  is the closest to the old `updateRecords()`.
2006
-
2008
+
2007
2009
  @param {SC.Record} recordTypes the record type or array of record types
2008
2010
  @param {Array} dataHashes array of data hashes to update
2009
2011
  @param {Array} ids optional array of ids. if not passed lookup on hashes
@@ -2015,13 +2017,13 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2015
2017
  ret = [],
2016
2018
  K = SC.Record,
2017
2019
  recordType, id, primaryKey, idx, dataHash, storeKey;
2018
-
2020
+
2019
2021
  // save lookup info
2020
2022
  if (!isArray) {
2021
2023
  recordType = recordTypes || SC.Record;
2022
2024
  primaryKey = recordType.prototype.primaryKey ;
2023
2025
  }
2024
-
2026
+
2025
2027
  // push each record
2026
2028
  for(idx=0;idx<len;idx++) {
2027
2029
  dataHash = dataHashes.objectAt(idx);
@@ -2031,9 +2033,9 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2031
2033
  }
2032
2034
  id = (ids) ? ids.objectAt(idx) : dataHash[primaryKey];
2033
2035
  ret[idx] = this.loadRecord(recordType, dataHash, id);
2034
-
2036
+
2035
2037
  }
2036
-
2038
+
2037
2039
  // return storeKeys
2038
2040
  return ret ;
2039
2041
  },
@@ -2042,7 +2044,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2042
2044
  Returns the `SC.Error` object associated with a specific record.
2043
2045
 
2044
2046
  @param {Number} storeKey The store key of the record.
2045
-
2047
+
2046
2048
  @returns {SC.Error} SC.Error or undefined if no error associated with the record.
2047
2049
  */
2048
2050
  readError: function(storeKey) {
@@ -2054,91 +2056,91 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2054
2056
  Returns the `SC.Error` object associated with a specific query.
2055
2057
 
2056
2058
  @param {SC.Query} query The SC.Query with which the error is associated.
2057
-
2059
+
2058
2060
  @returns {SC.Error} SC.Error or undefined if no error associated with the query.
2059
2061
  */
2060
2062
  readQueryError: function(query) {
2061
2063
  var errors = this.queryErrors ;
2062
2064
  return errors ? errors[SC.guidFor(query)] : undefined ;
2063
2065
  },
2064
-
2066
+
2065
2067
  // ..........................................................
2066
2068
  // DATA SOURCE CALLBACKS
2067
- //
2069
+ //
2068
2070
  // Mathods called by the data source on the store
2069
2071
 
2070
2072
  /**
2071
- Called by a `dataSource` when it cancels an inflight operation on a
2073
+ Called by a `dataSource` when it cancels an inflight operation on a
2072
2074
  record. This will transition the record back to it non-inflight state.
2073
-
2075
+
2074
2076
  @param {Number} storeKey record store key to cancel
2075
- @returns {SC.Store} reciever
2077
+ @returns {SC.Store} receiver
2076
2078
  */
2077
2079
  dataSourceDidCancel: function(storeKey) {
2078
- var status = this.readStatus(storeKey),
2080
+ var status = this.readStatus(storeKey),
2079
2081
  K = SC.Record;
2080
-
2082
+
2081
2083
  // EMPTY, ERROR, READY_CLEAN, READY_NEW, READY_DIRTY, DESTROYED_CLEAN,
2082
2084
  // DESTROYED_DIRTY
2083
2085
  if (!(status & K.BUSY)) {
2084
2086
  throw K.BAD_STATE_ERROR; // should never be called in this state
2085
2087
  }
2086
-
2088
+
2087
2089
  // otherwise, determine proper state transition
2088
2090
  switch(status) {
2089
2091
  case K.BUSY_LOADING:
2090
2092
  status = K.EMPTY;
2091
2093
  break ;
2092
-
2094
+
2093
2095
  case K.BUSY_CREATING:
2094
2096
  status = K.READY_NEW;
2095
2097
  break;
2096
-
2098
+
2097
2099
  case K.BUSY_COMMITTING:
2098
2100
  status = K.READY_DIRTY ;
2099
2101
  break;
2100
-
2102
+
2101
2103
  case K.BUSY_REFRESH_CLEAN:
2102
2104
  status = K.READY_CLEAN;
2103
2105
  break;
2104
-
2106
+
2105
2107
  case K.BUSY_REFRESH_DIRTY:
2106
2108
  status = K.READY_DIRTY ;
2107
2109
  break ;
2108
-
2110
+
2109
2111
  case K.BUSY_DESTROYING:
2110
2112
  status = K.DESTROYED_DIRTY ;
2111
2113
  break;
2112
-
2114
+
2113
2115
  default:
2114
2116
  throw K.BAD_STATE_ERROR ;
2115
- }
2117
+ }
2116
2118
  this.writeStatus(storeKey, status) ;
2117
2119
  this.dataHashDidChange(storeKey, null, YES);
2118
2120
  this._cancelCallback(storeKey);
2119
-
2121
+
2120
2122
  return this ;
2121
2123
  },
2122
-
2124
+
2123
2125
  /**
2124
2126
  Called by a data source when it creates or commits a record. Passing an
2125
- optional id will remap the `storeKey` to the new record id. This is
2127
+ optional id will remap the `storeKey` to the new record id. This is
2126
2128
  required when you commit a record that does not have an id yet.
2127
-
2129
+
2128
2130
  @param {Number} storeKey record store key to change to READY_CLEAN state
2129
2131
  @param {Hash} dataHash optional data hash to replace current hash
2130
2132
  @param {Object} newId optional new id to replace the old one
2131
- @returns {SC.Store} reciever
2133
+ @returns {SC.Store} receiver
2132
2134
  */
2133
2135
  dataSourceDidComplete: function(storeKey, dataHash, newId) {
2134
2136
  var status = this.readStatus(storeKey), K = SC.Record, statusOnly;
2135
-
2137
+
2136
2138
  // EMPTY, ERROR, READY_CLEAN, READY_NEW, READY_DIRTY, DESTROYED_CLEAN,
2137
2139
  // DESTROYED_DIRTY
2138
2140
  if (!(status & K.BUSY)) {
2139
2141
  throw K.BAD_STATE_ERROR; // should never be called in this state
2140
2142
  }
2141
-
2143
+
2142
2144
  // otherwise, determine proper state transition
2143
2145
  if(status===K.BUSY_DESTROYING) {
2144
2146
  throw K.BAD_STATE_ERROR ;
@@ -2147,7 +2149,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2147
2149
  this.writeStatus(storeKey, status) ;
2148
2150
  if (dataHash) this.writeDataHash(storeKey, dataHash, status) ;
2149
2151
  if (newId) SC.Store.replaceIdFor(storeKey, newId);
2150
-
2152
+
2151
2153
  statusOnly = dataHash || newId ? NO : YES;
2152
2154
  this.dataHashDidChange(storeKey, null, statusOnly);
2153
2155
 
@@ -2158,16 +2160,16 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2158
2160
  }
2159
2161
  //update callbacks
2160
2162
  this._retreiveCallbackForStoreKey(storeKey);
2161
-
2163
+
2162
2164
  return this ;
2163
2165
  },
2164
-
2166
+
2165
2167
  /**
2166
2168
  Called by a data source when it has destroyed a record. This will
2167
2169
  transition the record to the proper state.
2168
-
2170
+
2169
2171
  @param {Number} storeKey record store key to cancel
2170
- @returns {SC.Store} reciever
2172
+ @returns {SC.Store} receiver
2171
2173
  */
2172
2174
  dataSourceDidDestroy: function(storeKey) {
2173
2175
  var status = this.readStatus(storeKey), K = SC.Record;
@@ -2180,7 +2182,7 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2180
2182
  // otherwise, determine proper state transition
2181
2183
  else{
2182
2184
  status = K.DESTROYED_CLEAN ;
2183
- }
2185
+ }
2184
2186
  this.removeDataHash(storeKey, status) ;
2185
2187
  this.dataHashDidChange(storeKey);
2186
2188
 
@@ -2197,10 +2199,10 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2197
2199
 
2198
2200
  /**
2199
2201
  Converts the passed record into an error object.
2200
-
2202
+
2201
2203
  @param {Number} storeKey record store key to error
2202
2204
  @param {SC.Error} error [optional] an SC.Error instance to associate with storeKey
2203
- @returns {SC.Store} reciever
2205
+ @returns {SC.Store} receiver
2204
2206
  */
2205
2207
  dataSourceDidError: function(storeKey, error) {
2206
2208
  var status = this.readStatus(storeKey), errors = this.recordErrors, K = SC.Record;
@@ -2233,44 +2235,44 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2233
2235
 
2234
2236
  // ..........................................................
2235
2237
  // PUSH CHANGES FROM DATA SOURCE
2236
- //
2237
-
2238
+ //
2239
+
2238
2240
  /**
2239
- Call by the data source whenever you want to push new data out of band
2241
+ Call by the data source whenever you want to push new data out of band
2240
2242
  into the store.
2241
-
2243
+
2242
2244
  @param {Class} recordType the SC.Record subclass
2243
2245
  @param {Object} id the record id or null
2244
2246
  @param {Hash} dataHash data hash to load
2245
- @param {Number} storeKey optional store key.
2247
+ @param {Number} storeKey optional store key.
2246
2248
  @returns {Number|Boolean} storeKey if push was allowed, NO if not
2247
2249
  */
2248
2250
  pushRetrieve: function(recordType, id, dataHash, storeKey) {
2249
2251
  var K = SC.Record, status;
2250
-
2252
+
2251
2253
  if(storeKey===undefined) storeKey = recordType.storeKeyFor(id);
2252
2254
  status = this.readStatus(storeKey);
2253
2255
  if(status==K.EMPTY || status==K.ERROR || status==K.READY_CLEAN || status==K.DESTROYED_CLEAN) {
2254
-
2256
+
2255
2257
  status = K.READY_CLEAN;
2256
2258
  if(dataHash===undefined) this.writeStatus(storeKey, status) ;
2257
2259
  else this.writeDataHash(storeKey, dataHash, status) ;
2258
2260
 
2259
2261
  this.dataHashDidChange(storeKey);
2260
-
2262
+
2261
2263
  return storeKey;
2262
2264
  }
2263
2265
  //conflicted (ready)
2264
2266
  return NO;
2265
2267
  },
2266
-
2268
+
2267
2269
  /**
2268
- Call by the data source whenever you want to push a deletion into the
2270
+ Call by the data source whenever you want to push a deletion into the
2269
2271
  store.
2270
-
2272
+
2271
2273
  @param {Class} recordType the SC.Record subclass
2272
2274
  @param {Object} id the record id or null
2273
- @param {Number} storeKey optional store key.
2275
+ @param {Number} storeKey optional store key.
2274
2276
  @returns {Number|Boolean} storeKey if push was allowed, NO if not
2275
2277
  */
2276
2278
  pushDestroy: function(recordType, id, storeKey) {
@@ -2291,9 +2293,9 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2291
2293
  },
2292
2294
 
2293
2295
  /**
2294
- Call by the data source whenever you want to push an error into the
2296
+ Call by the data source whenever you want to push an error into the
2295
2297
  store.
2296
-
2298
+
2297
2299
  @param {Class} recordType the SC.Record subclass
2298
2300
  @param {Object} id the record id or null
2299
2301
  @param {SC.Error} error [optional] an SC.Error instance to associate with id or storeKey
@@ -2308,13 +2310,13 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2308
2310
 
2309
2311
  if(status==K.EMPTY || status==K.ERROR || status==K.READY_CLEAN || status==K.DESTROYED_CLEAN){
2310
2312
  status = K.ERROR;
2311
-
2313
+
2312
2314
  // Add the error to the array of record errors (for lookup later on if necessary).
2313
2315
  if (error && error.isError) {
2314
2316
  if (!errors) errors = this.recordErrors = [];
2315
2317
  errors[storeKey] = error;
2316
2318
  }
2317
-
2319
+
2318
2320
  this.writeStatus(storeKey, status) ;
2319
2321
  this.dataHashDidChange(storeKey, null, YES);
2320
2322
  return storeKey;
@@ -2322,29 +2324,29 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2322
2324
  //conflicted (error)
2323
2325
  return NO;
2324
2326
  },
2325
-
2327
+
2326
2328
  // ..........................................................
2327
2329
  // FETCH CALLBACKS
2328
- //
2329
-
2330
+ //
2331
+
2330
2332
  // **NOTE**: although these method works on RecordArray instances right now.
2331
2333
  // They could be optimized to actually share query results between nested
2332
- // stores. This is why these methods are implemented here instead of
2334
+ // stores. This is why these methods are implemented here instead of
2333
2335
  // directly on `Query` or `RecordArray` objects.
2334
-
2336
+
2335
2337
  /**
2336
2338
  Sets the passed array of storeKeys as the new data for the query. You
2337
2339
  can call this at any time for a remote query to update its content. If
2338
2340
  you want to use incremental loading, then pass a `SparseArray` object.
2339
-
2341
+
2340
2342
  If the query you pass is not a REMOTE query, then this method will raise
2341
- an exception. This will also implicitly transition the query state to
2343
+ an exception. This will also implicitly transition the query state to
2342
2344
  `SC.Record.READY`.
2343
-
2345
+
2344
2346
  If you called `loadRecords()` before to load the actual content, you can
2345
2347
  call this method with the return value of that method to actually set the
2346
2348
  storeKeys on the result.
2347
-
2349
+
2348
2350
  @param {SC.Query} query the query you are loading. must be remote.
2349
2351
  @param {SC.Array} storeKeys array of store keys
2350
2352
  @returns {SC.Store} receiver
@@ -2357,75 +2359,75 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2357
2359
  var recArray = this._findQuery(query, YES, NO);
2358
2360
  if (recArray) recArray.set('storeKeys', storeKeys);
2359
2361
  this.dataSourceDidFetchQuery(query);
2360
-
2362
+
2361
2363
  return this ;
2362
2364
  },
2363
-
2365
+
2364
2366
  /**
2365
- Called by your data source whenever you finish fetching the results of a
2367
+ Called by your data source whenever you finish fetching the results of a
2366
2368
  query. This will put the query into a READY state if it was loading.
2367
-
2368
- Note that if the query is a REMOTE query, then you must separately load
2369
- the results into the query using `loadQueryResults()`. If the query is
2370
- LOCAL, then the query will update automatically with any new records you
2369
+
2370
+ Note that if the query is a REMOTE query, then you must separately load
2371
+ the results into the query using `loadQueryResults()`. If the query is
2372
+ LOCAL, then the query will update automatically with any new records you
2371
2373
  added to the store.
2372
-
2374
+
2373
2375
  @param {SC.Query} query the query you fetched
2374
2376
  @returns {SC.Store} receiver
2375
2377
  */
2376
2378
  dataSourceDidFetchQuery: function(query) {
2377
2379
  return this._scstore_dataSourceDidFetchQuery(query, YES);
2378
2380
  },
2379
-
2381
+
2380
2382
  _scstore_dataSourceDidFetchQuery: function(query, createIfNeeded) {
2381
2383
  var recArray = this._findQuery(query, createIfNeeded, NO),
2382
2384
  nestedStores = this.get('nestedStores'),
2383
2385
  loc = nestedStores ? nestedStores.get('length') : 0;
2384
-
2386
+
2385
2387
  // fix query if needed
2386
2388
  if (recArray) recArray.storeDidFetchQuery(query);
2387
-
2389
+
2388
2390
  // notify nested stores
2389
2391
  while(--loc >= 0) {
2390
2392
  nestedStores[loc]._scstore_dataSourceDidFetchQuery(query, NO);
2391
2393
  }
2392
-
2394
+
2393
2395
  return this ;
2394
2396
  },
2395
-
2397
+
2396
2398
  /**
2397
2399
  Called by your data source if it cancels fetching the results of a query.
2398
2400
  This will put any RecordArray's back into its original state (READY or
2399
2401
  EMPTY).
2400
-
2402
+
2401
2403
  @param {SC.Query} query the query you cancelled
2402
2404
  @returns {SC.Store} receiver
2403
2405
  */
2404
2406
  dataSourceDidCancelQuery: function(query) {
2405
2407
  return this._scstore_dataSourceDidCancelQuery(query, YES);
2406
2408
  },
2407
-
2409
+
2408
2410
  _scstore_dataSourceDidCancelQuery: function(query, createIfNeeded) {
2409
2411
  var recArray = this._findQuery(query, createIfNeeded, NO),
2410
2412
  nestedStores = this.get('nestedStores'),
2411
2413
  loc = nestedStores ? nestedStores.get('length') : 0;
2412
-
2414
+
2413
2415
  // fix query if needed
2414
2416
  if (recArray) recArray.storeDidCancelQuery(query);
2415
-
2417
+
2416
2418
  // notify nested stores
2417
2419
  while(--loc >= 0) {
2418
2420
  nestedStores[loc]._scstore_dataSourceDidCancelQuery(query, NO);
2419
2421
  }
2420
-
2422
+
2421
2423
  return this ;
2422
2424
  },
2423
-
2425
+
2424
2426
  /**
2425
2427
  Called by your data source if it encountered an error loading the query.
2426
2428
  This will put the query into an error state until you try to refresh it
2427
2429
  again.
2428
-
2430
+
2429
2431
  @param {SC.Query} query the query with the error
2430
2432
  @param {SC.Error} error [optional] an SC.Error instance to associate with query
2431
2433
  @returns {SC.Store} receiver
@@ -2457,18 +2459,18 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2457
2459
 
2458
2460
  return this ;
2459
2461
  },
2460
-
2462
+
2461
2463
  // ..........................................................
2462
2464
  // INTERNAL SUPPORT
2463
- //
2464
-
2465
+ //
2466
+
2465
2467
  /** @private */
2466
2468
  init: function() {
2467
2469
  sc_super();
2468
2470
  this.reset();
2469
2471
  },
2470
-
2471
-
2472
+
2473
+
2472
2474
  toString: function() {
2473
2475
  // Include the name if the client has specified one.
2474
2476
  var name = this.get('name');
@@ -2484,32 +2486,32 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2484
2486
 
2485
2487
  // ..........................................................
2486
2488
  // PRIMARY KEY CONVENIENCE METHODS
2487
- //
2489
+ //
2488
2490
 
2489
- /**
2491
+ /**
2490
2492
  Given a `storeKey`, return the `primaryKey`.
2491
-
2493
+
2492
2494
  @param {Number} storeKey the store key
2493
2495
  @returns {String} primaryKey value
2494
2496
  */
2495
2497
  idFor: function(storeKey) {
2496
2498
  return SC.Store.idFor(storeKey);
2497
2499
  },
2498
-
2500
+
2499
2501
  /**
2500
2502
  Given a storeKey, return the recordType.
2501
-
2503
+
2502
2504
  @param {Number} storeKey the store key
2503
2505
  @returns {SC.Record} record instance
2504
2506
  */
2505
2507
  recordTypeFor: function(storeKey) {
2506
2508
  return SC.Store.recordTypeFor(storeKey) ;
2507
2509
  },
2508
-
2510
+
2509
2511
  /**
2510
2512
  Given a `recordType` and `primaryKey`, find the `storeKey`. If the
2511
2513
  `primaryKey` has not been assigned a `storeKey` yet, it will be added.
2512
-
2514
+
2513
2515
  @param {SC.Record} recordType the record type
2514
2516
  @param {String} primaryKey the primary key
2515
2517
  @returns {Number} storeKey
@@ -2517,12 +2519,12 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2517
2519
  storeKeyFor: function(recordType, primaryKey) {
2518
2520
  return recordType.storeKeyFor(primaryKey);
2519
2521
  },
2520
-
2522
+
2521
2523
  /**
2522
2524
  Given a `primaryKey` value for the record, returns the associated
2523
2525
  `storeKey`. As opposed to `storeKeyFor()` however, this method
2524
2526
  will **NOT** generate a new `storeKey` but returned `undefined`.
2525
-
2527
+
2526
2528
  @param {SC.Record} recordType the record type
2527
2529
  @param {String} primaryKey the primary key
2528
2530
  @returns {Number} a storeKey.
@@ -2530,56 +2532,56 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2530
2532
  storeKeyExists: function(recordType, primaryKey) {
2531
2533
  return recordType.storeKeyExists(primaryKey);
2532
2534
  },
2533
-
2535
+
2534
2536
  /**
2535
2537
  Finds all `storeKey`s of a certain record type in this store
2536
2538
  and returns an array.
2537
-
2539
+
2538
2540
  @param {SC.Record} recordType
2539
2541
  @returns {Array} set of storeKeys
2540
2542
  */
2541
2543
  storeKeysFor: function(recordType) {
2542
- var ret = [],
2544
+ var ret = [],
2543
2545
  isEnum = recordType && recordType.isEnumerable,
2544
2546
  recType, storeKey, isMatch ;
2545
-
2547
+
2546
2548
  if (!this.statuses) return ret;
2547
2549
  for(storeKey in SC.Store.recordTypesByStoreKey) {
2548
2550
  recType = SC.Store.recordTypesByStoreKey[storeKey];
2549
-
2551
+
2550
2552
  // if same record type and this store has it
2551
2553
  if (isEnum) isMatch = recordType.contains(recType);
2552
2554
  else isMatch = recType === recordType;
2553
-
2555
+
2554
2556
  if(isMatch && this.statuses[storeKey]) ret.push(parseInt(storeKey, 10));
2555
2557
  }
2556
-
2558
+
2557
2559
  return ret;
2558
2560
  },
2559
-
2561
+
2560
2562
  /**
2561
2563
  Finds all `storeKey`s in this store
2562
2564
  and returns an array.
2563
-
2565
+
2564
2566
  @returns {Array} set of storeKeys
2565
2567
  */
2566
2568
  storeKeys: function() {
2567
2569
  var ret = [], storeKey;
2568
2570
  if(!this.statuses) return ret;
2569
-
2571
+
2570
2572
  for(storeKey in this.statuses) {
2571
2573
  // if status is not empty
2572
2574
  if(this.statuses[storeKey] != SC.Record.EMPTY) {
2573
2575
  ret.push(parseInt(storeKey, 10));
2574
2576
  }
2575
2577
  }
2576
-
2578
+
2577
2579
  return ret;
2578
2580
  },
2579
-
2581
+
2580
2582
  /**
2581
2583
  Returns string representation of a `storeKey`, with status.
2582
-
2584
+
2583
2585
  @param {Number} storeKey
2584
2586
  @returns {String}
2585
2587
  */
@@ -2587,54 +2589,54 @@ SC.Store = SC.Object.extend( /** @scope SC.Store.prototype */ {
2587
2589
  var rec = this.materializeRecord(storeKey);
2588
2590
  return rec.statusString();
2589
2591
  }
2590
-
2592
+
2591
2593
  }) ;
2592
2594
 
2593
2595
  SC.Store.mixin(/** @scope SC.Store.prototype */{
2594
-
2596
+
2595
2597
  /**
2596
2598
  Standard error raised if you try to commit changes from a nested store
2597
2599
  and there is a conflict.
2598
-
2600
+
2599
2601
  @type Error
2600
2602
  */
2601
2603
  CHAIN_CONFLICT_ERROR: new Error("Nested Store Conflict"),
2602
-
2604
+
2603
2605
  /**
2604
- Standard error if you try to perform an operation on a nested store
2606
+ Standard error if you try to perform an operation on a nested store
2605
2607
  without a parent.
2606
-
2608
+
2607
2609
  @type Error
2608
2610
  */
2609
2611
  NO_PARENT_STORE_ERROR: new Error("Parent Store Required"),
2610
-
2612
+
2611
2613
  /**
2612
2614
  Standard error if you try to perform an operation on a nested store that
2613
2615
  is only supported in root stores.
2614
-
2616
+
2615
2617
  @type Error
2616
2618
  */
2617
2619
  NESTED_STORE_UNSUPPORTED_ERROR: new Error("Unsupported In Nested Store"),
2618
-
2620
+
2619
2621
  /**
2620
2622
  Standard error if you try to retrieve a record in a nested store that is
2621
2623
  dirty. (This is allowed on the main store, but not in nested stores.)
2622
-
2624
+
2623
2625
  @type Error
2624
2626
  */
2625
2627
  NESTED_STORE_RETRIEVE_DIRTY_ERROR: new Error("Cannot Retrieve Dirty Record in Nested Store"),
2626
2628
 
2627
2629
  /**
2628
2630
  Data hash state indicates the data hash is currently editable
2629
-
2631
+
2630
2632
  @type String
2631
2633
  */
2632
2634
  EDITABLE: 'editable',
2633
-
2635
+
2634
2636
  /**
2635
- Data hash state indicates the hash no longer tracks changes from a
2637
+ Data hash state indicates the hash no longer tracks changes from a
2636
2638
  parent store, but it is not editable.
2637
-
2639
+
2638
2640
  @type String
2639
2641
  */
2640
2642
  LOCKED: 'locked',
@@ -2642,88 +2644,88 @@ SC.Store.mixin(/** @scope SC.Store.prototype */{
2642
2644
  /**
2643
2645
  Data hash state indicates the hash is tracking changes from the parent
2644
2646
  store and is not editable.
2645
-
2647
+
2646
2648
  @type String
2647
2649
  */
2648
2650
  INHERITED: 'inherited',
2649
-
2651
+
2650
2652
  /** @private
2651
2653
  This array maps all storeKeys to primary keys. You will not normally
2652
- access this method directly. Instead use the `idFor()` and
2654
+ access this method directly. Instead use the `idFor()` and
2653
2655
  `storeKeyFor()` methods on `SC.Record`.
2654
2656
  */
2655
2657
  idsByStoreKey: [],
2656
-
2658
+
2657
2659
  /** @private
2658
2660
  Maps all `storeKey`s to a `recordType`. Once a `storeKey` is associated
2659
2661
  with a `primaryKey` and `recordType` that remains constant throughout the
2660
2662
  lifetime of the application.
2661
2663
  */
2662
2664
  recordTypesByStoreKey: {},
2663
-
2665
+
2664
2666
  /** @private
2665
2667
  Maps some `storeKeys` to query instance. Once a `storeKey` is associated
2666
- with a query instance, that remains constant through the lifetime of the
2667
- application. If a `Query` is destroyed, it will remove itself from this
2668
+ with a query instance, that remains constant through the lifetime of the
2669
+ application. If a `Query` is destroyed, it will remove itself from this
2668
2670
  list.
2669
-
2671
+
2670
2672
  Don't access this directly. Use queryFor().
2671
2673
  */
2672
2674
  queriesByStoreKey: [],
2673
-
2675
+
2674
2676
  /** @private
2675
2677
  The next store key to allocate. A storeKey must always be greater than 0
2676
2678
  */
2677
2679
  nextStoreKey: 1,
2678
-
2680
+
2679
2681
  /**
2680
2682
  Generates a new store key for use.
2681
-
2683
+
2682
2684
  @type Number
2683
2685
  */
2684
2686
  generateStoreKey: function() { return this.nextStoreKey++; },
2685
-
2686
- /**
2687
+
2688
+ /**
2687
2689
  Given a `storeKey` returns the `primaryKey` associated with the key.
2688
2690
  If no `primaryKey` is associated with the `storeKey`, returns `null`.
2689
-
2691
+
2690
2692
  @param {Number} storeKey the store key
2691
2693
  @returns {String} the primary key or null
2692
2694
  */
2693
2695
  idFor: function(storeKey) {
2694
2696
  return this.idsByStoreKey[storeKey] ;
2695
2697
  },
2696
-
2698
+
2697
2699
  /**
2698
2700
  Given a `storeKey`, returns the query object associated with the key. If
2699
2701
  no query is associated with the `storeKey`, returns `null`.
2700
-
2702
+
2701
2703
  @param {Number} storeKey the store key
2702
2704
  @returns {SC.Query} query query object
2703
2705
  */
2704
2706
  queryFor: function(storeKey) {
2705
- return this.queriesByStoreKey[storeKey];
2707
+ return this.queriesByStoreKey[storeKey];
2706
2708
  },
2707
-
2709
+
2708
2710
  /**
2709
2711
  Given a `storeKey` returns the `SC.Record` class associated with the key.
2710
2712
  If no record type is associated with the store key, returns `null`.
2711
-
2713
+
2712
2714
  The SC.Record class will only be found if you have already called
2713
2715
  storeKeyFor() on the record.
2714
-
2716
+
2715
2717
  @param {Number} storeKey the store key
2716
2718
  @returns {SC.Record} the record type
2717
2719
  */
2718
2720
  recordTypeFor: function(storeKey) {
2719
2721
  return this.recordTypesByStoreKey[storeKey];
2720
2722
  },
2721
-
2723
+
2722
2724
  /**
2723
- Swaps the `primaryKey` mapped to the given storeKey with the new
2725
+ Swaps the `primaryKey` mapped to the given storeKey with the new
2724
2726
  `primaryKey`. If the `storeKey` is not currently associated with a record
2725
2727
  this will raise an exception.
2726
-
2728
+
2727
2729
  @param {Number} storeKey the existing store key
2728
2730
  @param {String} newPrimaryKey the new primary key
2729
2731
  @returns {SC.Store} receiver
@@ -2731,7 +2733,7 @@ SC.Store.mixin(/** @scope SC.Store.prototype */{
2731
2733
  replaceIdFor: function(storeKey, newId) {
2732
2734
  var oldId = this.idsByStoreKey[storeKey],
2733
2735
  recordType, storeKeys;
2734
-
2736
+
2735
2737
  if (oldId !== newId) { // skip if id isn't changing
2736
2738
 
2737
2739
  recordType = this.recordTypeFor(storeKey);
@@ -2740,32 +2742,32 @@ SC.Store.mixin(/** @scope SC.Store.prototype */{
2740
2742
  }
2741
2743
 
2742
2744
  // map one direction...
2743
- this.idsByStoreKey[storeKey] = newId;
2745
+ this.idsByStoreKey[storeKey] = newId;
2744
2746
 
2745
2747
  // then the other...
2746
2748
  storeKeys = recordType.storeKeysById() ;
2747
2749
  delete storeKeys[oldId];
2748
- storeKeys[newId] = storeKey;
2750
+ storeKeys[newId] = storeKey;
2749
2751
  }
2750
-
2752
+
2751
2753
  return this ;
2752
2754
  },
2753
-
2755
+
2754
2756
  /**
2755
2757
  Swaps the `recordType` recorded for a given `storeKey`. Normally you
2756
2758
  should not call this method directly as it can damage the store behavior.
2757
- This method is used by other store methods to set the `recordType` for a
2759
+ This method is used by other store methods to set the `recordType` for a
2758
2760
  `storeKey`.
2759
-
2761
+
2760
2762
  @param {Integer} storeKey the store key
2761
2763
  @param {SC.Record} recordType a record class
2762
- @returns {SC.Store} reciever
2764
+ @returns {SC.Store} receiver
2763
2765
  */
2764
2766
  replaceRecordTypeFor: function(storeKey, recordType) {
2765
2767
  this.recordTypesByStoreKey[storeKey] = recordType;
2766
2768
  return this ;
2767
2769
  }
2768
-
2770
+
2769
2771
  });
2770
2772
 
2771
2773
 
@@ -2774,7 +2776,7 @@ SC.Store.prototype.nextStoreIndex = 1;
2774
2776
 
2775
2777
  // ..........................................................
2776
2778
  // COMPATIBILITY
2777
- //
2779
+ //
2778
2780
 
2779
2781
  /** @private
2780
2782
  global store is used only for deprecated compatibility methods. Don't use
@@ -2789,14 +2791,14 @@ SC.Store._getDefaultStore = function() {
2789
2791
  /** @private
2790
2792
 
2791
2793
  DEPRECATED
2792
-
2793
- Included for compatibility, loads data hashes with the named `recordType`.
2794
- If no `recordType` is passed, expects to find a `recordType` property in the
2794
+
2795
+ Included for compatibility, loads data hashes with the named `recordType`.
2796
+ If no `recordType` is passed, expects to find a `recordType` property in the
2795
2797
  data hashes. `dataSource` and `isLoaded` params are ignored.
2796
-
2797
- Calls `SC.Store#loadRecords()` on the default store. Do not use this method in
2798
- new code.
2799
-
2798
+
2799
+ Calls `SC.Store#loadRecords()` on the default store. Do not use this method in
2800
+ new code.
2801
+
2800
2802
  @param {Array} dataHashes data hashes to import
2801
2803
  @param {Object} dataSource ignored
2802
2804
  @param {SC.Record} recordType default record type
@@ -2804,37 +2806,35 @@ SC.Store._getDefaultStore = function() {
2804
2806
  @returns {Array} SC.Record instances for loaded data hashes
2805
2807
  */
2806
2808
  SC.Store.updateRecords = function(dataHashes, dataSource, recordType, isLoaded) {
2807
-
2808
2809
  SC.Logger.warn("SC.Store.updateRecords() is deprecated. Use loadRecords() instead");
2809
-
2810
2810
  var store = this._getDefaultStore(),
2811
2811
  len = dataHashes.length,
2812
2812
  idx, ret;
2813
-
2813
+
2814
2814
  // if no recordType was passed, build an array of recordTypes from hashes
2815
2815
  if (!recordType) {
2816
2816
  recordType = [];
2817
2817
  for(idx=0;idx<len;idx++) recordType[idx] = dataHashes[idx].recordType;
2818
2818
  }
2819
-
2819
+
2820
2820
  // call new API. Returns storeKeys
2821
2821
  ret = store.loadRecords(recordType, dataHashes);
2822
-
2822
+
2823
2823
  // map to SC.Record instances
2824
2824
  len = ret.length;
2825
2825
  for(idx=0;idx<len;idx++) ret[idx] = store.materializeRecord(ret[idx]);
2826
-
2826
+
2827
2827
  return ret ;
2828
2828
  };
2829
2829
 
2830
2830
  /** @private
2831
2831
 
2832
- DEPRECATED
2832
+ DEPRECATED
2833
2833
 
2834
2834
  Finds a record with the passed guid on the default store. This is included
2835
2835
  only for compatibility. You should use the newer `find()` method defined on
2836
2836
  `SC.Store` instead.
2837
-
2837
+
2838
2838
  @param {String} guid the guid
2839
2839
  @param {SC.Record} recordType expected record type
2840
2840
  @returns {SC.Record} found record
@@ -2845,12 +2845,12 @@ SC.Store.find = function(guid, recordType) {
2845
2845
 
2846
2846
  /** @private
2847
2847
 
2848
- DEPRECATED
2848
+ DEPRECATED
2849
2849
 
2850
- Passes through to `findAll` on default store. This is included only for
2850
+ Passes through to `findAll` on default store. This is included only for
2851
2851
  compatibility. You should use the newer `findAll()` defined on `SC.Store`
2852
2852
  instead.
2853
-
2853
+
2854
2854
  @param {Hash} filter search parameters
2855
2855
  @param {SC.Record} recordType type of record to find
2856
2856
  @returns {SC.RecordArray} result set