/test/unit/item-view.spec.js

https://gitlab.com/CodeYellowBV/backbone.marionette · JavaScript · 431 lines · 342 code · 89 blank · 0 comment · 3 complexity · 50389e19fa844b19c0f20d91f62c7b81 MD5 · raw file

  1. describe('item view', function() {
  2. 'use strict';
  3. beforeEach(function() {
  4. this.modelData = {foo: 'bar'};
  5. this.collectionData = [{foo: 'bar'}, {foo: 'baz'}];
  6. this.model = new Backbone.Model(this.modelData);
  7. this.collection = new Backbone.Collection(this.collectionData);
  8. this.template = 'foobar';
  9. this.templateStub = this.sinon.stub().returns(this.template);
  10. });
  11. describe('when rendering without a valid template', function() {
  12. beforeEach(function() {
  13. this.view = new Marionette.ItemView();
  14. });
  15. it('should throw an exception because there was no valid template', function() {
  16. expect(this.view.render).to.throw('Cannot render the template since it is null or undefined.');
  17. });
  18. });
  19. describe('when rendering with a false template', function() {
  20. beforeEach(function() {
  21. this.onBeforeRenderStub = this.sinon.stub();
  22. this.onRenderStub = this.sinon.stub();
  23. this.View = Marionette.ItemView.extend({
  24. template: false,
  25. onBeforeRender: this.onBeforeRenderStub,
  26. onRender: this.onRenderStub
  27. });
  28. this.view = new this.View();
  29. this.marionetteRendererSpy = this.sinon.spy(Marionette.Renderer, 'render');
  30. this.triggerSpy = this.sinon.spy(this.view, 'trigger');
  31. this.serializeDataSpy = this.sinon.spy(this.view, 'serializeData');
  32. this.mixinTemplateHelpersSpy = this.sinon.spy(this.view, 'mixinTemplateHelpers');
  33. this.attachElContentSpy = this.sinon.spy(this.view, 'attachElContent');
  34. this.view.render();
  35. });
  36. it('should not throw an exception for a false template', function() {
  37. expect(this.view.render).to.not.throw('Cannot render the template since it is null or undefined.');
  38. });
  39. it('should call an "onBeforeRender" method on the view', function() {
  40. expect(this.onBeforeRenderStub).to.have.been.calledOnce;
  41. });
  42. it('should call an "onRender" method on the view', function() {
  43. expect(this.onRenderStub).to.have.been.calledOnce;
  44. });
  45. it('should trigger a before:render event', function() {
  46. expect(this.triggerSpy).to.have.been.calledWith('before:render', this.view);
  47. });
  48. it('should trigger a rendered event', function() {
  49. expect(this.triggerSpy).to.have.been.calledWith('render', this.view);
  50. });
  51. it('should not add in data or template helpers', function() {
  52. expect(this.serializeDataSpy).to.not.have.been.called;
  53. expect(this.mixinTemplateHelpersSpy).to.not.have.been.called;
  54. });
  55. it('should not render a template', function() {
  56. expect(this.marionetteRendererSpy).to.not.have.been.called;
  57. });
  58. it('should not attach any html content', function() {
  59. expect(this.attachElContentSpy).to.not.have.been.called;
  60. });
  61. it('should claim isRendered', function() {
  62. expect(this.view.isRendered).to.be.true;
  63. });
  64. });
  65. describe('when rendering with a overridden attachElContent', function() {
  66. beforeEach(function() {
  67. this.attachElContentStub = this.sinon.stub();
  68. this.ItemView = Marionette.ItemView.extend({
  69. template : this.templateStub,
  70. attachElContent : this.attachElContentStub
  71. });
  72. this.sinon.spy(Marionette.Renderer, 'render');
  73. this.itemView = new this.ItemView();
  74. this.itemView.render();
  75. });
  76. it('should render according to the custom attachElContent logic', function() {
  77. expect(this.attachElContentStub).to.have.been.calledOnce.and.calledWith(this.template);
  78. });
  79. it('should pass template stub, data and view instance to `Marionette.Renderer.Render`', function() {
  80. expect(Marionette.Renderer.render).to.have.been.calledWith(this.templateStub, {}, this.itemView);
  81. });
  82. });
  83. describe('when rendering', function() {
  84. beforeEach(function() {
  85. this.onBeforeRenderStub = this.sinon.spy(function() {
  86. return this.isRendered;
  87. });
  88. this.onRenderStub = this.sinon.spy(function() {
  89. return this.isRendered;
  90. });
  91. this.View = Marionette.ItemView.extend({
  92. template : this.templateStub,
  93. onBeforeRender : this.onBeforeRenderStub,
  94. onRender : this.onRenderStub
  95. });
  96. this.view = new this.View();
  97. this.triggerSpy = this.sinon.spy(this.view, 'trigger');
  98. this.view.render();
  99. });
  100. it('should call a "onBeforeRender" method on the view', function() {
  101. expect(this.onBeforeRenderStub).to.have.been.calledOnce;
  102. });
  103. it('should call an "onRender" method on the view', function() {
  104. expect(this.onRenderStub).to.have.been.calledOnce;
  105. });
  106. it('should call "onBeforeRender" before "onRender"', function() {
  107. expect(this.onBeforeRenderStub).to.have.been.calledBefore(this.onRenderStub);
  108. });
  109. it('should not be rendered when "onBeforeRender" is called', function() {
  110. expect(this.onBeforeRenderStub.lastCall.returnValue).not.to.be.ok;
  111. });
  112. it('should be rendered when "onRender" is called', function() {
  113. expect(this.onRenderStub.lastCall.returnValue).to.be.true;
  114. });
  115. it('should trigger a before:render event', function() {
  116. expect(this.triggerSpy).to.have.been.calledWith('before:render', this.view);
  117. });
  118. it('should trigger a rendered event', function() {
  119. expect(this.triggerSpy).to.have.been.calledWith('render', this.view);
  120. });
  121. it('should mark as rendered', function() {
  122. expect(this.view).to.have.property('isRendered', true);
  123. });
  124. });
  125. describe('when an item view has a model and is rendered', function() {
  126. beforeEach(function() {
  127. this.view = new Marionette.ItemView({
  128. template : this.templateStub,
  129. model : this.model
  130. });
  131. this.serializeDataSpy = this.sinon.spy(this.view, 'serializeData');
  132. this.view.render();
  133. });
  134. it('should serialize the model', function() {
  135. expect(this.serializeDataSpy).to.have.been.calledOnce;
  136. });
  137. it('should render the template with the serialized model', function() {
  138. expect(this.templateStub).to.have.been.calledWith(this.modelData);
  139. });
  140. });
  141. describe('when an item view has a collection and is rendered', function() {
  142. beforeEach(function() {
  143. this.view = new Marionette.ItemView({
  144. template : this.templateStub,
  145. collection : this.collection
  146. });
  147. this.serializeDataSpy = this.sinon.spy(this.view, 'serializeData');
  148. this.view.render();
  149. });
  150. it('should serialize the collection', function() {
  151. expect(this.serializeDataSpy).to.have.been.calledOnce;
  152. });
  153. it('should render the template with the serialized collection', function() {
  154. expect(this.templateStub).to.have.been.calledWith({items: this.collectionData});
  155. });
  156. });
  157. describe('when an item view has a model and collection, and is rendered', function() {
  158. beforeEach(function() {
  159. this.view = new Marionette.ItemView({
  160. template : this.templateStub,
  161. model : this.model,
  162. collection : this.collection
  163. });
  164. this.serializeDataSpy = this.sinon.spy(this.view, 'serializeData');
  165. this.view.render();
  166. });
  167. it('should serialize the model', function() {
  168. expect(this.serializeDataSpy).to.have.been.calledOnce;
  169. });
  170. it('should render the template with the serialized model', function() {
  171. expect(this.templateStub).to.have.been.calledWith(this.modelData);
  172. });
  173. });
  174. describe('when destroying an item view', function() {
  175. beforeEach(function() {
  176. this.onBeforeDestroyStub = this.sinon.spy(function() {
  177. return {
  178. isRendered: this.isRendered,
  179. isDestroyed: this.isDestroyed
  180. };
  181. });
  182. this.onDestroyStub = this.sinon.spy(function() {
  183. return {
  184. isRendered: this.isRendered,
  185. isDestroyed: this.isDestroyed
  186. };
  187. });
  188. this.View = Marionette.ItemView.extend({
  189. template : this.templateStub,
  190. onBeforeDestroy : this.onBeforeDestroyStub,
  191. onDestroy : this.onDestroyStub
  192. });
  193. this.view = new this.View();
  194. this.view.render();
  195. this.removeSpy = this.sinon.spy(this.view, 'remove');
  196. this.stopListeningSpy = this.sinon.spy(this.view, 'stopListening');
  197. this.triggerSpy = this.sinon.spy(this.view, 'trigger');
  198. this.sinon.spy(this.view, 'destroy');
  199. this.view.destroy();
  200. });
  201. it('should remove the views EL from the DOM', function() {
  202. expect(this.removeSpy).to.have.been.calledOnce;
  203. });
  204. it('should unbind any listener to custom view events', function() {
  205. expect(this.stopListeningSpy).to.have.been.calledOnce;
  206. });
  207. it('should trigger "before:destroy"', function() {
  208. expect(this.triggerSpy).to.have.been.calledWith('before:destroy');
  209. });
  210. it('should trigger "destroy"', function() {
  211. expect(this.triggerSpy).to.have.been.calledWith('destroy');
  212. });
  213. it('should call "onBeforeDestroy" if provided', function() {
  214. expect(this.onBeforeDestroyStub).to.have.been.called;
  215. });
  216. it('should call "onDestroy" if provided', function() {
  217. expect(this.onDestroyStub).to.have.been.called;
  218. });
  219. it('should return the view', function() {
  220. expect(this.view.destroy).to.have.returned(this.view);
  221. });
  222. it('should not be destroyed when "onBeforeDestroy" is called', function() {
  223. expect(this.onBeforeDestroyStub.lastCall.returnValue.isDestroyed).not.to.be.ok;
  224. });
  225. it('should be destroyed when "onDestroy" is called', function() {
  226. expect(this.onDestroyStub.lastCall.returnValue.isDestroyed).to.be.true;
  227. });
  228. it('should be rendered when "onDestroy" is called', function() {
  229. expect(this.onDestroyStub.lastCall.returnValue.isRendered).to.be.true;
  230. });
  231. it('should be marked destroyed', function() {
  232. expect(this.view).to.have.property('isDestroyed', true);
  233. });
  234. it('should be marked not rendered', function() {
  235. expect(this.view).to.have.property('isRendered', false);
  236. });
  237. });
  238. describe('when re-rendering an ItemView that is already shown', function() {
  239. beforeEach(function() {
  240. this.onDomRefreshStub = this.sinon.stub();
  241. this.View = Marionette.ItemView.extend({
  242. template : this.templateStub,
  243. onDomRefresh : this.onDomRefreshStub
  244. });
  245. this.view = new this.View();
  246. this.setFixtures(this.view.$el);
  247. this.view.render();
  248. this.view.triggerMethod('show');
  249. this.view.render();
  250. });
  251. it('should trigger a dom:refresh event', function() {
  252. expect(this.onDomRefreshStub).to.have.been.calledTwice;
  253. });
  254. });
  255. describe('has a valid inheritance chain back to Marionette.View', function() {
  256. beforeEach(function() {
  257. this.constructorSpy = this.sinon.spy(Marionette, 'View');
  258. this.itemView = new Marionette.ItemView();
  259. });
  260. it('calls the parent Marionette.Views constructor function on instantiation', function() {
  261. expect(this.constructorSpy).to.have.been.called;
  262. });
  263. });
  264. describe('when serializing view data', function() {
  265. beforeEach(function() {
  266. this.modelData = {foo: 'bar'};
  267. this.collectionData = [{foo: 'bar'}, {foo: 'baz'}];
  268. this.itemView = new Marionette.ItemView();
  269. this.sinon.spy(this.itemView, 'serializeModel');
  270. this.sinon.spy(this.itemView, 'serializeCollection');
  271. });
  272. it('should return an empty object without data', function() {
  273. expect(this.itemView.serializeData()).to.deep.equal({});
  274. });
  275. describe('and the view has a model', function() {
  276. beforeEach(function() {
  277. this.itemView.model = new Backbone.Model(this.modelData);
  278. this.itemView.serializeData();
  279. });
  280. it('should call serializeModel', function() {
  281. expect(this.itemView.serializeModel).to.have.been.calledOnce;
  282. });
  283. it('should not call serializeCollection', function() {
  284. expect(this.itemView.serializeCollection).to.not.have.been.called;
  285. });
  286. });
  287. describe('and the view only has a collection', function() {
  288. beforeEach(function() {
  289. this.itemView.collection = new Backbone.Collection(this.collectionData);
  290. this.itemView.serializeData(1, 2, 3);
  291. });
  292. it('should call serializeCollection', function() {
  293. expect(this.itemView.serializeCollection).to.have.been.calledOnce;
  294. });
  295. it('and the serialize function should be called with the provided arguments', function() {
  296. expect(this.itemView.serializeCollection).to.have.been.calledWith(this.itemView.collection, 1, 2, 3);
  297. });
  298. it('should not call serializeModel', function() {
  299. expect(this.itemView.serializeModel).to.not.have.been.called;
  300. });
  301. });
  302. describe('and the view has a collection and a model', function() {
  303. beforeEach(function() {
  304. this.itemView.model = new Backbone.Model(this.modelData);
  305. this.itemView.collection = new Backbone.Collection(this.collectionData);
  306. this.itemView.serializeData(1, 2, 3);
  307. });
  308. it('should call serializeModel', function() {
  309. expect(this.itemView.serializeModel).to.have.been.calledOnce;
  310. });
  311. it('and the serialize function should be called with the provided arguments', function() {
  312. expect(this.itemView.serializeModel).to.have.been.calledWith(this.itemView.model, 1, 2, 3);
  313. });
  314. it('should not call serializeCollection', function() {
  315. expect(this.itemView.serializeCollection).to.not.have.been.called;
  316. });
  317. });
  318. });
  319. describe('when serializing a collection', function() {
  320. beforeEach(function() {
  321. this.collectionData = [{foo: 'bar'}, {foo: 'baz'}];
  322. this.itemView = new Marionette.ItemView({
  323. collection: new Backbone.Collection(this.collectionData)
  324. });
  325. });
  326. it('should serialize to an items attribute', function() {
  327. expect(this.itemView.serializeData().items).to.be.defined;
  328. });
  329. it('should serialize all models', function() {
  330. expect(this.itemView.serializeData().items).to.deep.equal(this.collectionData);
  331. });
  332. });
  333. describe('has a valid inheritance chain back to Marionette.View', function() {
  334. beforeEach(function() {
  335. this.constructor = this.sinon.spy(Marionette, 'View');
  336. this.collectionView = new Marionette.ItemView();
  337. });
  338. it('calls the parent Marionette.View\'s constructor function on instantiation', function() {
  339. expect(this.constructor).to.have.been.calledOnce;
  340. });
  341. });
  342. });