bookmarks.js 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467
  1. /*global waitUntilPromise */
  2. (function (root, factory) {
  3. define([
  4. "jquery",
  5. "converse-core",
  6. "utils",
  7. "mock",
  8. "test-utils"
  9. ], factory);
  10. } (this, function ($, converse, utils, mock, test_utils) {
  11. "use strict";
  12. var $iq = converse.env.$iq,
  13. Strophe = converse.env.Strophe,
  14. _ = converse.env._;
  15. describe("A chat room", function () {
  16. it("can be bookmarked", mock.initConverse(function (_converse) {
  17. var sent_stanza, IQ_id;
  18. var sendIQ = _converse.connection.sendIQ;
  19. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  20. sent_stanza = iq;
  21. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  22. });
  23. spyOn(_converse.connection, 'getUniqueId').and.callThrough();
  24. test_utils.openChatRoom(_converse, 'theplay', 'conference.shakespeare.lit', 'JC');
  25. var jid = 'theplay@conference.shakespeare.lit';
  26. var view = _converse.chatboxviews.get(jid);
  27. spyOn(view, 'renderBookmarkForm').and.callThrough();
  28. spyOn(view, 'cancelConfiguration').and.callThrough();
  29. var $bookmark = view.$el.find('.icon-pushpin');
  30. $bookmark.click();
  31. expect(view.renderBookmarkForm).toHaveBeenCalled();
  32. view.$el.find('.button-cancel').click();
  33. expect(view.cancelConfiguration).toHaveBeenCalled();
  34. expect($bookmark.hasClass('on-button'), false);
  35. $bookmark.click();
  36. expect(view.renderBookmarkForm).toHaveBeenCalled();
  37. /* Client uploads data:
  38. * --------------------
  39. * <iq from='juliet@capulet.lit/balcony' type='set' id='pip1'>
  40. * <pubsub xmlns='http://jabber.org/protocol/pubsub'>
  41. * <publish node='storage:bookmarks'>
  42. * <item id='current'>
  43. * <storage xmlns='storage:bookmarks'>
  44. * <conference name='The Play&apos;s the Thing'
  45. * autojoin='true'
  46. * jid='theplay@conference.shakespeare.lit'>
  47. * <nick>JC</nick>
  48. * </conference>
  49. * </storage>
  50. * </item>
  51. * </publish>
  52. * <publish-options>
  53. * <x xmlns='jabber:x:data' type='submit'>
  54. * <field var='FORM_TYPE' type='hidden'>
  55. * <value>http://jabber.org/protocol/pubsub#publish-options</value>
  56. * </field>
  57. * <field var='pubsub#persist_items'>
  58. * <value>true</value>
  59. * </field>
  60. * <field var='pubsub#access_model'>
  61. * <value>whitelist</value>
  62. * </field>
  63. * </x>
  64. * </publish-options>
  65. * </pubsub>
  66. * </iq>
  67. */
  68. expect(view.model.get('bookmarked')).toBeFalsy();
  69. var $form = view.$el.find('.chatroom-form');
  70. $form.find('input[name="name"]').val('Play&apos;s the Thing');
  71. $form.find('input[name="autojoin"]').prop('checked', true);
  72. $form.find('input[name="nick"]').val('JC');
  73. $form.submit();
  74. expect(view.model.get('bookmarked')).toBeTruthy();
  75. expect($bookmark.hasClass('on-button'), true);
  76. expect(sent_stanza.toLocaleString()).toBe(
  77. "<iq type='set' from='dummy@localhost/resource' xmlns='jabber:client' id='"+IQ_id+"'>"+
  78. "<pubsub xmlns='http://jabber.org/protocol/pubsub'>"+
  79. "<publish node='storage:bookmarks'>"+
  80. "<item id='current'>"+
  81. "<storage xmlns='storage:bookmarks'>"+
  82. "<conference name='Play&amp;apos;s the Thing' autojoin='true' jid='theplay@conference.shakespeare.lit'>"+
  83. "<nick>JC</nick>"+
  84. "</conference>"+
  85. "</storage>"+
  86. "</item>"+
  87. "</publish>"+
  88. "<publish-options>"+
  89. "<x xmlns='jabber:x:data' type='submit'>"+
  90. "<field var='FORM_TYPE' type='hidden'>"+
  91. "<value>http://jabber.org/protocol/pubsub#publish-options</value>"+
  92. "</field>"+
  93. "<field var='pubsub#persist_items'>"+
  94. "<value>true</value>"+
  95. "</field>"+
  96. "<field var='pubsub#access_model'>"+
  97. "<value>whitelist</value>"+
  98. "</field>"+
  99. "</x>"+
  100. "</publish-options>"+
  101. "</pubsub>"+
  102. "</iq>"
  103. );
  104. /* Server acknowledges successful storage
  105. *
  106. * <iq to='juliet@capulet.lit/balcony' type='result' id='pip1'/>
  107. */
  108. var stanza = $iq({
  109. 'to':_converse.connection.jid,
  110. 'type':'result',
  111. 'id':IQ_id
  112. });
  113. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  114. // We ignore this IQ stanza... (unless it's an error stanza), so
  115. // nothing to test for here.
  116. }));
  117. it("will be automatically opened if 'autojoin' is set on the bookmark", mock.initConverse(function (_converse) {
  118. var jid = 'lounge@localhost';
  119. _converse.bookmarks.create({
  120. 'jid': jid,
  121. 'autojoin': false,
  122. 'name': 'The Lounge',
  123. 'nick': ' Othello'
  124. });
  125. expect(_.isUndefined(_converse.chatboxviews.get(jid))).toBeTruthy();
  126. jid = 'theplay@conference.shakespeare.lit';
  127. _converse.bookmarks.create({
  128. 'jid': jid,
  129. 'autojoin': true,
  130. 'name': 'The Play',
  131. 'nick': ' Othello'
  132. });
  133. expect(_.isUndefined(_converse.chatboxviews.get(jid))).toBeFalsy();
  134. }));
  135. describe("when bookmarked", function () {
  136. it("displays that it's bookmarked through its bookmark icon", mock.initConverse(function (_converse) {
  137. test_utils.openChatRoom(_converse, 'lounge', 'localhost', 'dummy');
  138. var view = _converse.chatboxviews.get('lounge@localhost');
  139. var $bookmark_icon = view.$('.icon-pushpin');
  140. expect($bookmark_icon.hasClass('button-on')).toBeFalsy();
  141. view.model.set('bookmarked', true);
  142. expect($bookmark_icon.hasClass('button-on')).toBeTruthy();
  143. view.model.set('bookmarked', false);
  144. expect($bookmark_icon.hasClass('button-on')).toBeFalsy();
  145. }));
  146. it("can be unbookmarked", mock.initConverse(function (_converse) {
  147. var sent_stanza, IQ_id;
  148. var sendIQ = _converse.connection.sendIQ;
  149. test_utils.openChatRoom(_converse, 'theplay', 'conference.shakespeare.lit', 'JC');
  150. var jid = 'theplay@conference.shakespeare.lit';
  151. var view = _converse.chatboxviews.get(jid);
  152. spyOn(view, 'toggleBookmark').and.callThrough();
  153. spyOn(_converse.bookmarks, 'sendBookmarkStanza').and.callThrough();
  154. view.delegateEvents();
  155. _converse.bookmarks.create({
  156. 'jid': view.model.get('jid'),
  157. 'autojoin': false,
  158. 'name': 'The Play',
  159. 'nick': ' Othello'
  160. });
  161. expect(_converse.bookmarks.length).toBe(1);
  162. expect(view.model.get('bookmarked')).toBeTruthy();
  163. var $bookmark_icon = view.$('.icon-pushpin');
  164. expect($bookmark_icon.hasClass('button-on')).toBeTruthy();
  165. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  166. sent_stanza = iq;
  167. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  168. });
  169. spyOn(_converse.connection, 'getUniqueId').and.callThrough();
  170. $bookmark_icon.click();
  171. expect(view.toggleBookmark).toHaveBeenCalled();
  172. expect($bookmark_icon.hasClass('button-on')).toBeFalsy();
  173. expect(_converse.bookmarks.length).toBe(0);
  174. // Check that an IQ stanza is sent out, containing no
  175. // conferences to bookmark (since we removed the one and
  176. // only bookmark).
  177. expect(sent_stanza.toLocaleString()).toBe(
  178. "<iq type='set' from='dummy@localhost/resource' xmlns='jabber:client' id='"+IQ_id+"'>"+
  179. "<pubsub xmlns='http://jabber.org/protocol/pubsub'>"+
  180. "<publish node='storage:bookmarks'>"+
  181. "<item id='current'>"+
  182. "<storage xmlns='storage:bookmarks'/>"+
  183. "</item>"+
  184. "</publish>"+
  185. "<publish-options>"+
  186. "<x xmlns='jabber:x:data' type='submit'>"+
  187. "<field var='FORM_TYPE' type='hidden'>"+
  188. "<value>http://jabber.org/protocol/pubsub#publish-options</value>"+
  189. "</field>"+
  190. "<field var='pubsub#persist_items'>"+
  191. "<value>true</value>"+
  192. "</field>"+
  193. "<field var='pubsub#access_model'>"+
  194. "<value>whitelist</value>"+
  195. "</field>"+
  196. "</x>"+
  197. "</publish-options>"+
  198. "</pubsub>"+
  199. "</iq>"
  200. );
  201. }));
  202. });
  203. describe("and when autojoin is set", function () {
  204. it("will be be opened and joined automatically upon login", mock.initConverse(function (_converse) {
  205. spyOn(_converse.api.rooms, 'open');
  206. var jid = 'theplay@conference.shakespeare.lit';
  207. var model = _converse.bookmarks.create({
  208. 'jid': jid,
  209. 'autojoin': false,
  210. 'name': 'The Play',
  211. 'nick': ''
  212. });
  213. expect(_converse.api.rooms.open).not.toHaveBeenCalled();
  214. _converse.bookmarks.remove(model);
  215. _converse.bookmarks.create({
  216. 'jid': jid,
  217. 'autojoin': true,
  218. 'name': 'Hamlet',
  219. 'nick': ''
  220. });
  221. expect(_converse.api.rooms.open).toHaveBeenCalled();
  222. }));
  223. });
  224. });
  225. describe("Bookmarks", function () {
  226. xit("can be pushed from the XMPP server", mock.initConverse(function (_converse) {
  227. // TODO
  228. /* The stored data is automatically pushed to all of the user's
  229. * connected resources.
  230. *
  231. * Publisher receives event notification
  232. * -------------------------------------
  233. * <message from='juliet@capulet.lit'
  234. * to='juliet@capulet.lit/balcony'
  235. * type='headline'
  236. * id='rnfoo1'>
  237. * <event xmlns='http://jabber.org/protocol/pubsub#event'>
  238. * <items node='storage:bookmarks'>
  239. * <item id='current'>
  240. * <storage xmlns='storage:bookmarks'>
  241. * <conference name='The Play&apos;s the Thing'
  242. * autojoin='true'
  243. * jid='theplay@conference.shakespeare.lit'>
  244. * <nick>JC</nick>
  245. * </conference>
  246. * </storage>
  247. * </item>
  248. * </items>
  249. * </event>
  250. * </message>
  251. * <message from='juliet@capulet.lit'
  252. * to='juliet@capulet.lit/chamber'
  253. * type='headline'
  254. * id='rnfoo2'>
  255. * <event xmlns='http://jabber.org/protocol/pubsub#event'>
  256. * <items node='storage:bookmarks'>
  257. * <item id='current'>
  258. * <storage xmlns='storage:bookmarks'>
  259. * <conference name='The Play&apos;s the Thing'
  260. * autojoin='true'
  261. * jid='theplay@conference.shakespeare.lit'>
  262. * <nick>JC</nick>
  263. * </conference>
  264. * </storage>
  265. * </item>
  266. * </items>
  267. * </event>
  268. * </message>
  269. */
  270. }));
  271. it("can be retrieved from the XMPP server",
  272. mock.initConverseWithConnectionSpies(['send'], function (_converse) {
  273. /* Client requests all items
  274. * -------------------------
  275. *
  276. * <iq from='juliet@capulet.lit/randomID' type='get' id='retrieve1'>
  277. * <pubsub xmlns='http://jabber.org/protocol/pubsub'>
  278. * <items node='storage:bookmarks'/>
  279. * </pubsub>
  280. * </iq>
  281. */
  282. var IQ_id;
  283. expect(_.filter(_converse.connection.send.calls.all(), function (call) {
  284. var stanza = call.args[0]
  285. if (!(stanza instanceof Element) || stanza.nodeName !== 'iq') {
  286. return;
  287. }
  288. // XXX: Wrapping in a div is a workaround for PhantomJS
  289. var div = document.createElement('div');
  290. div.appendChild(stanza);
  291. if (div.innerHTML ===
  292. '<iq from="dummy@localhost/resource" type="get" '+
  293. 'xmlns="jabber:client" id="'+stanza.getAttribute('id')+'">'+
  294. '<pubsub xmlns="http://jabber.org/protocol/pubsub">'+
  295. '<items node="storage:bookmarks"></items>'+
  296. '</pubsub>'+
  297. '</iq>') {
  298. IQ_id = stanza.getAttribute('id');
  299. return true;
  300. }
  301. }).length).toBe(1);
  302. /*
  303. * Server returns all items
  304. * ------------------------
  305. * <iq type='result'
  306. * to='juliet@capulet.lit/randomID'
  307. * id='retrieve1'>
  308. * <pubsub xmlns='http://jabber.org/protocol/pubsub'>
  309. * <items node='storage:bookmarks'>
  310. * <item id='current'>
  311. * <storage xmlns='storage:bookmarks'>
  312. * <conference name='The Play&apos;s the Thing'
  313. * autojoin='true'
  314. * jid='theplay@conference.shakespeare.lit'>
  315. * <nick>JC</nick>
  316. * </conference>
  317. * </storage>
  318. * </item>
  319. * </items>
  320. * </pubsub>
  321. * </iq>
  322. */
  323. expect(_converse.bookmarks.models.length).toBe(0);
  324. var stanza = $iq({'to': _converse.connection.jid, 'type':'result', 'id':IQ_id})
  325. .c('pubsub', {'xmlns': Strophe.NS.PUBSUB})
  326. .c('items', {'node': 'storage:bookmarks'})
  327. .c('item', {'id': 'current'})
  328. .c('storage', {'xmlns': 'storage:bookmarks'})
  329. .c('conference', {
  330. 'name': 'The Play&apos;s the Thing',
  331. 'autojoin': 'true',
  332. 'jid': 'theplay@conference.shakespeare.lit'
  333. }).c('nick').t('JC').up().up()
  334. .c('conference', {
  335. 'name': 'Another room',
  336. 'autojoin': 'false',
  337. 'jid': 'another@conference.shakespeare.lit'
  338. }).c('nick').t('JC').up().up();
  339. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  340. expect(_converse.bookmarks.models.length).toBe(2);
  341. expect(_converse.bookmarks.findWhere({'jid': 'theplay@conference.shakespeare.lit'}).get('autojoin')).toBe(true);
  342. expect(_converse.bookmarks.findWhere({'jid': 'another@conference.shakespeare.lit'}).get('autojoin')).toBe(false);
  343. }));
  344. describe("The rooms panel", function () {
  345. it("shows a list of bookmarks", mock.initConverseWithConnectionSpies(['send'], function (_converse) {
  346. var IQ_id;
  347. expect(_.filter(_converse.connection.send.calls.all(), function (call) {
  348. var stanza = call.args[0]
  349. if (!(stanza instanceof Element) || stanza.nodeName !== 'iq') {
  350. return;
  351. }
  352. // XXX: Wrapping in a div is a workaround for PhantomJS
  353. var div = document.createElement('div');
  354. div.appendChild(stanza);
  355. if (div.innerHTML ===
  356. '<iq from="dummy@localhost/resource" type="get" '+
  357. 'xmlns="jabber:client" id="'+stanza.getAttribute('id')+'">'+
  358. '<pubsub xmlns="http://jabber.org/protocol/pubsub">'+
  359. '<items node="storage:bookmarks"></items>'+
  360. '</pubsub>'+
  361. '</iq>') {
  362. IQ_id = stanza.getAttribute('id');
  363. return true;
  364. }
  365. }).length).toBe(1);
  366. _converse.chatboxviews.get('controlbox').$('#chatrooms dl.bookmarks').html('');
  367. var stanza = $iq({'to': _converse.connection.jid, 'type':'result', 'id':IQ_id})
  368. .c('pubsub', {'xmlns': Strophe.NS.PUBSUB})
  369. .c('items', {'node': 'storage:bookmarks'})
  370. .c('item', {'id': 'current'})
  371. .c('storage', {'xmlns': 'storage:bookmarks'})
  372. .c('conference', {
  373. 'name': 'The Play&apos;s the Thing',
  374. 'autojoin': 'false',
  375. 'jid': 'theplay@conference.shakespeare.lit'
  376. }).c('nick').t('JC').up().up()
  377. .c('conference', {
  378. 'name': 'Bookmark with a very very long name that will be shortened',
  379. 'autojoin': 'false',
  380. 'jid': 'longname@conference.shakespeare.lit'
  381. }).c('nick').t('JC').up().up()
  382. .c('conference', {
  383. 'name': 'Another room',
  384. 'autojoin': 'false',
  385. 'jid': 'another@conference.shakespeare.lit'
  386. }).c('nick').t('JC').up().up();
  387. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  388. expect($('#chatrooms dl.bookmarks dd').length).toBe(3);
  389. }));
  390. it("remembers the toggle state of the bookmarks list",
  391. mock.initConverseWithConnectionSpies(['send'], function (_converse) {
  392. var IQ_id;
  393. expect(_.filter(_converse.connection.send.calls.all(), function (call) {
  394. var stanza = call.args[0]
  395. if (!(stanza instanceof Element) || stanza.nodeName !== 'iq') {
  396. return;
  397. }
  398. // XXX: Wrapping in a div is a workaround for PhantomJS
  399. var div = document.createElement('div');
  400. div.appendChild(stanza);
  401. if (div.innerHTML ===
  402. '<iq from="dummy@localhost/resource" type="get" '+
  403. 'xmlns="jabber:client" id="'+stanza.getAttribute('id')+'">'+
  404. '<pubsub xmlns="http://jabber.org/protocol/pubsub">'+
  405. '<items node="storage:bookmarks"></items>'+
  406. '</pubsub>'+
  407. '</iq>') {
  408. IQ_id = stanza.getAttribute('id');
  409. return true;
  410. }
  411. }).length).toBe(1);
  412. _converse.chatboxviews.get('controlbox').$('#chatrooms dl.bookmarks').html('');
  413. var stanza = $iq({'to': _converse.connection.jid, 'type':'result', 'id':IQ_id})
  414. .c('pubsub', {'xmlns': Strophe.NS.PUBSUB})
  415. .c('items', {'node': 'storage:bookmarks'})
  416. .c('item', {'id': 'current'})
  417. .c('storage', {'xmlns': 'storage:bookmarks'});
  418. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  419. _converse.bookmarks.create({
  420. 'jid': 'theplay@conference.shakespeare.lit',
  421. 'autojoin': false,
  422. 'name': 'The Play',
  423. 'nick': ''
  424. });
  425. test_utils.openControlBox().openRoomsPanel(_converse);
  426. expect($('#chatrooms dl.bookmarks dd:visible').length).toBe(1);
  427. expect(_converse.bookmarksview.list_model.get('toggle-state')).toBe(_converse.OPENED);
  428. $('#chatrooms .bookmarks-toggle').click();
  429. expect($('#chatrooms dl.bookmarks dd:visible').length).toBe(0);
  430. expect(_converse.bookmarksview.list_model.get('toggle-state')).toBe(_converse.CLOSED);
  431. $('#chatrooms .bookmarks-toggle').click();
  432. expect($('#chatrooms dl.bookmarks dd:visible').length).toBe(1);
  433. expect(_converse.bookmarksview.list_model.get('toggle-state')).toBe(_converse.OPENED);
  434. }));
  435. });
  436. });
  437. }));