bookmarks.js 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654
  1. /*global waitUntilPromise */
  2. (function (root, factory) {
  3. define([
  4. "jasmine",
  5. "jquery",
  6. "mock",
  7. "test-utils"
  8. ], factory);
  9. } (this, function (jasmine, $, mock, test_utils) {
  10. "use strict";
  11. const $iq = converse.env.$iq,
  12. $msg = converse.env.$msg,
  13. Backbone = converse.env.Backbone,
  14. Strophe = converse.env.Strophe,
  15. sizzle = converse.env.sizzle,
  16. _ = converse.env._,
  17. u = converse.env.utils;
  18. describe("A chat room", function () {
  19. it("can be bookmarked", mock.initConverse(
  20. null, ['rosterGroupsFetched'], {},
  21. async function (done, _converse) {
  22. await test_utils.waitUntilDiscoConfirmed(
  23. _converse, _converse.bare_jid,
  24. [{'category': 'pubsub', 'type': 'pep'}],
  25. ['http://jabber.org/protocol/pubsub#publish-options']
  26. );
  27. let sent_stanza, IQ_id;
  28. const sendIQ = _converse.connection.sendIQ;
  29. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  30. sent_stanza = iq;
  31. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  32. });
  33. spyOn(_converse.connection, 'getUniqueId').and.callThrough();
  34. await test_utils.openChatRoom(_converse, 'theplay', 'conference.shakespeare.lit', 'JC');
  35. var jid = 'theplay@conference.shakespeare.lit';
  36. const view = _converse.chatboxviews.get(jid);
  37. spyOn(view, 'renderBookmarkForm').and.callThrough();
  38. spyOn(view, 'closeForm').and.callThrough();
  39. await test_utils.waitUntil(() => !_.isNull(view.el.querySelector('.toggle-bookmark')));
  40. let toggle = view.el.querySelector('.toggle-bookmark');
  41. expect(toggle.title).toBe('Bookmark this groupchat');
  42. toggle.click();
  43. expect(view.renderBookmarkForm).toHaveBeenCalled();
  44. view.el.querySelector('.button-cancel').click();
  45. expect(view.closeForm).toHaveBeenCalled();
  46. expect(u.hasClass('on-button', toggle), false);
  47. expect(toggle.title).toBe('Bookmark this groupchat');
  48. toggle.click();
  49. expect(view.renderBookmarkForm).toHaveBeenCalled();
  50. /* Client uploads data:
  51. * --------------------
  52. * <iq from='juliet@capulet.lit/balcony' type='set' id='pip1'>
  53. * <pubsub xmlns='http://jabber.org/protocol/pubsub'>
  54. * <publish node='storage:bookmarks'>
  55. * <item id='current'>
  56. * <storage xmlns='storage:bookmarks'>
  57. * <conference name='The Play&apos;s the Thing'
  58. * autojoin='true'
  59. * jid='theplay@conference.shakespeare.lit'>
  60. * <nick>JC</nick>
  61. * </conference>
  62. * </storage>
  63. * </item>
  64. * </publish>
  65. * <publish-options>
  66. * <x xmlns='jabber:x:data' type='submit'>
  67. * <field var='FORM_TYPE' type='hidden'>
  68. * <value>http://jabber.org/protocol/pubsub#publish-options</value>
  69. * </field>
  70. * <field var='pubsub#persist_items'>
  71. * <value>true</value>
  72. * </field>
  73. * <field var='pubsub#access_model'>
  74. * <value>whitelist</value>
  75. * </field>
  76. * </x>
  77. * </publish-options>
  78. * </pubsub>
  79. * </iq>
  80. */
  81. expect(view.model.get('bookmarked')).toBeFalsy();
  82. const form = view.el.querySelector('.chatroom-form');
  83. form.querySelector('input[name="name"]').value = 'Play&apos;s the Thing';
  84. form.querySelector('input[name="autojoin"]').checked = 'checked';
  85. form.querySelector('input[name="nick"]').value = 'JC';
  86. _converse.connection.IQ_stanzas = [];
  87. view.el.querySelector('.btn-primary').click();
  88. await test_utils.waitUntil(() => sent_stanza);
  89. expect(sent_stanza.toLocaleString()).toBe(
  90. `<iq from="dummy@localhost/resource" id="${IQ_id}" type="set" xmlns="jabber:client">`+
  91. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  92. `<publish node="storage:bookmarks">`+
  93. `<item id="current">`+
  94. `<storage xmlns="storage:bookmarks">`+
  95. `<conference autojoin="true" jid="theplay@conference.shakespeare.lit" name="Play&amp;apos;s the Thing">`+
  96. `<nick>JC</nick>`+
  97. `</conference>`+
  98. `</storage>`+
  99. `</item>`+
  100. `</publish>`+
  101. `<publish-options>`+
  102. `<x type="submit" xmlns="jabber:x:data">`+
  103. `<field type="hidden" var="FORM_TYPE">`+
  104. `<value>http://jabber.org/protocol/pubsub#publish-options</value>`+
  105. `</field>`+
  106. `<field var="pubsub#persist_items">`+
  107. `<value>true</value>`+
  108. `</field>`+
  109. `<field var="pubsub#access_model">`+
  110. `<value>whitelist</value>`+
  111. `</field>`+
  112. `</x>`+
  113. `</publish-options>`+
  114. `</pubsub>`+
  115. `</iq>`
  116. );
  117. /* Server acknowledges successful storage
  118. *
  119. * <iq to='juliet@capulet.lit/balcony' type='result' id='pip1'/>
  120. */
  121. const stanza = $iq({
  122. 'to':_converse.connection.jid,
  123. 'type':'result',
  124. 'id':IQ_id
  125. });
  126. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  127. await test_utils.waitUntil(() => view.model.get('bookmarked'));
  128. toggle = view.el.querySelector('.toggle-bookmark');
  129. expect(view.model.get('bookmarked')).toBeTruthy();
  130. expect(toggle.title).toBe('Unbookmark this groupchat');
  131. expect(u.hasClass('on-button', toggle), true);
  132. // We ignore this IQ stanza... (unless it's an error stanza), so
  133. // nothing to test for here.
  134. done();
  135. }));
  136. it("will be automatically opened if 'autojoin' is set on the bookmark", mock.initConverse(
  137. null, ['rosterGroupsFetched'], {},
  138. async function (done, _converse) {
  139. await test_utils.waitUntilDiscoConfirmed(
  140. _converse, _converse.bare_jid,
  141. [{'category': 'pubsub', 'type': 'pep'}],
  142. ['http://jabber.org/protocol/pubsub#publish-options']
  143. );
  144. let jid = 'lounge@localhost';
  145. _converse.bookmarks.create({
  146. 'jid': jid,
  147. 'autojoin': false,
  148. 'name': 'The Lounge',
  149. 'nick': ' Othello'
  150. });
  151. expect(_.isUndefined(_converse.chatboxviews.get(jid))).toBeTruthy();
  152. jid = 'theplay@conference.shakespeare.lit';
  153. _converse.bookmarks.create({
  154. 'jid': jid,
  155. 'autojoin': true,
  156. 'name': 'The Play',
  157. 'nick': ' Othello'
  158. });
  159. expect(_.isUndefined(_converse.chatboxviews.get(jid))).toBeFalsy();
  160. done();
  161. }));
  162. describe("when bookmarked", function () {
  163. it("displays that it's bookmarked through its bookmark icon", mock.initConverse(
  164. null, ['rosterGroupsFetched'], {},
  165. async function (done, _converse) {
  166. test_utils.waitUntilDiscoConfirmed(
  167. _converse, _converse.bare_jid,
  168. [{'category': 'pubsub', 'type': 'pep'}],
  169. ['http://jabber.org/protocol/pubsub#publish-options']
  170. );
  171. await test_utils.openChatRoom(_converse, 'lounge', 'localhost', 'dummy');
  172. const view = _converse.chatboxviews.get('lounge@localhost');
  173. await test_utils.waitUntil(() => !_.isNull(view.el.querySelector('.toggle-bookmark')));
  174. var bookmark_icon = view.el.querySelector('.toggle-bookmark');
  175. expect(_.includes(bookmark_icon.classList, 'button-on')).toBeFalsy();
  176. view.model.set('bookmarked', true);
  177. expect(_.includes(bookmark_icon.classList, 'button-on')).toBeTruthy();
  178. view.model.set('bookmarked', false);
  179. expect(_.includes(bookmark_icon.classList, 'button-on')).toBeFalsy();
  180. done();
  181. }));
  182. it("can be unbookmarked", mock.initConverse(
  183. null, ['rosterGroupsFetched'], {},
  184. async function (done, _converse) {
  185. let sent_stanza, IQ_id;
  186. await test_utils.waitUntilDiscoConfirmed(
  187. _converse, _converse.bare_jid,
  188. [{'category': 'pubsub', 'type': 'pep'}],
  189. ['http://jabber.org/protocol/pubsub#publish-options']
  190. );
  191. const sendIQ = _converse.connection.sendIQ;
  192. await test_utils.openChatRoom(_converse, 'theplay', 'conference.shakespeare.lit', 'JC');
  193. const jid = 'theplay@conference.shakespeare.lit';
  194. const view = _converse.chatboxviews.get(jid);
  195. await test_utils.waitUntil(() => !_.isNull(view.el.querySelector('.toggle-bookmark')));
  196. spyOn(view, 'toggleBookmark').and.callThrough();
  197. spyOn(_converse.bookmarks, 'sendBookmarkStanza').and.callThrough();
  198. view.delegateEvents();
  199. _converse.bookmarks.create({
  200. 'jid': view.model.get('jid'),
  201. 'autojoin': false,
  202. 'name': 'The Play',
  203. 'nick': ' Othello'
  204. });
  205. expect(_converse.bookmarks.length).toBe(1);
  206. expect(view.model.get('bookmarked')).toBeTruthy();
  207. var bookmark_icon = view.el.querySelector('.toggle-bookmark');
  208. expect(u.hasClass('button-on', bookmark_icon)).toBeTruthy();
  209. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  210. sent_stanza = iq;
  211. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  212. });
  213. spyOn(_converse.connection, 'getUniqueId').and.callThrough();
  214. bookmark_icon.click();
  215. expect(view.toggleBookmark).toHaveBeenCalled();
  216. expect(u.hasClass('button-on', bookmark_icon)).toBeFalsy();
  217. expect(_converse.bookmarks.length).toBe(0);
  218. // Check that an IQ stanza is sent out, containing no
  219. // conferences to bookmark (since we removed the one and
  220. // only bookmark).
  221. expect(sent_stanza.toLocaleString()).toBe(
  222. `<iq from="dummy@localhost/resource" id="${IQ_id}" type="set" xmlns="jabber:client">`+
  223. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  224. `<publish node="storage:bookmarks">`+
  225. `<item id="current">`+
  226. `<storage xmlns="storage:bookmarks"/>`+
  227. `</item>`+
  228. `</publish>`+
  229. `<publish-options>`+
  230. `<x type="submit" xmlns="jabber:x:data">`+
  231. `<field type="hidden" var="FORM_TYPE">`+
  232. `<value>http://jabber.org/protocol/pubsub#publish-options</value>`+
  233. `</field>`+
  234. `<field var="pubsub#persist_items">`+
  235. `<value>true</value>`+
  236. `</field>`+
  237. `<field var="pubsub#access_model">`+
  238. `<value>whitelist</value>`+
  239. `</field>`+
  240. `</x>`+
  241. `</publish-options>`+
  242. `</pubsub>`+
  243. `</iq>`
  244. );
  245. done();
  246. }));
  247. });
  248. describe("and when autojoin is set", function () {
  249. it("will be be opened and joined automatically upon login", mock.initConverse(
  250. null, ['rosterGroupsFetched'], {},
  251. async function (done, _converse) {
  252. await test_utils.waitUntilDiscoConfirmed(
  253. _converse, _converse.bare_jid,
  254. [{'category': 'pubsub', 'type': 'pep'}],
  255. ['http://jabber.org/protocol/pubsub#publish-options']
  256. );
  257. spyOn(_converse.api.rooms, 'create').and.callThrough();
  258. const jid = 'theplay@conference.shakespeare.lit';
  259. const model = _converse.bookmarks.create({
  260. 'jid': jid,
  261. 'autojoin': false,
  262. 'name': 'The Play',
  263. 'nick': ''
  264. });
  265. expect(_converse.api.rooms.create).not.toHaveBeenCalled();
  266. _converse.bookmarks.remove(model);
  267. _converse.bookmarks.create({
  268. 'jid': jid,
  269. 'autojoin': true,
  270. 'name': 'Hamlet',
  271. 'nick': ''
  272. });
  273. expect(_converse.api.rooms.create).toHaveBeenCalled();
  274. done();
  275. }));
  276. });
  277. });
  278. describe("Bookmarks", function () {
  279. it("can be pushed from the XMPP server", mock.initConverse(
  280. ['send'], ['rosterGroupsFetched', 'connected'], {},
  281. async function (done, _converse) {
  282. await test_utils.waitUntilDiscoConfirmed(
  283. _converse, _converse.bare_jid,
  284. [{'category': 'pubsub', 'type': 'pep'}],
  285. ['http://jabber.org/protocol/pubsub#publish-options']
  286. );
  287. await test_utils.waitUntil(() => _converse.bookmarks);
  288. // Emit here instead of mocking fetching of bookmarks.
  289. _converse.emit('bookmarksInitialized');
  290. /* The stored data is automatically pushed to all of the user's
  291. * connected resources.
  292. *
  293. * Publisher receives event notification
  294. * -------------------------------------
  295. * <message from='juliet@capulet.lit'
  296. * to='juliet@capulet.lit/balcony'
  297. * type='headline'
  298. * id='rnfoo1'>
  299. * <event xmlns='http://jabber.org/protocol/pubsub#event'>
  300. * <items node='storage:bookmarks'>
  301. * <item id='current'>
  302. * <storage xmlns='storage:bookmarks'>
  303. * <conference name='The Play&apos;s the Thing'
  304. * autojoin='true'
  305. * jid='theplay@conference.shakespeare.lit'>
  306. * <nick>JC</nick>
  307. * </conference>
  308. * </storage>
  309. * </item>
  310. * </items>
  311. * </event>
  312. * </message>
  313. */
  314. var stanza = $msg({
  315. 'from': 'dummy@localhost',
  316. 'to': 'dummy@localhost/resource',
  317. 'type': 'headline',
  318. 'id': 'rnfoo1'
  319. }).c('event', {'xmlns': 'http://jabber.org/protocol/pubsub#event'})
  320. .c('items', {'node': 'storage:bookmarks'})
  321. .c('item', {'id': 'current'})
  322. .c('storage', {'xmlns': 'storage:bookmarks'})
  323. .c('conference', {'name': 'The Play&apos;s the Thing',
  324. 'autojoin': 'true',
  325. 'jid':'theplay@conference.shakespeare.lit'})
  326. .c('nick').t('JC');
  327. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  328. await test_utils.waitUntil(() => _converse.bookmarks.length);
  329. expect(_converse.bookmarks.length).toBe(1);
  330. expect(_converse.chatboxviews.get('theplay@conference.shakespeare.lit')).not.toBeUndefined();
  331. done();
  332. }));
  333. it("can be retrieved from the XMPP server", mock.initConverse(
  334. {'connection': ['send']}, ['chatBoxesFetched', 'roomsPanelRendered', 'rosterGroupsFetched'], {},
  335. async function (done, _converse) {
  336. await test_utils.waitUntilDiscoConfirmed(
  337. _converse, _converse.bare_jid,
  338. [{'category': 'pubsub', 'type': 'pep'}],
  339. ['http://jabber.org/protocol/pubsub#publish-options']
  340. );
  341. /* Client requests all items
  342. * -------------------------
  343. *
  344. * <iq from='juliet@capulet.lit/randomID' type='get' id='retrieve1'>
  345. * <pubsub xmlns='http://jabber.org/protocol/pubsub'>
  346. * <items node='storage:bookmarks'/>
  347. * </pubsub>
  348. * </iq>
  349. */
  350. let IQ_id;
  351. const call = await test_utils.waitUntil(() =>
  352. _.filter(
  353. _converse.connection.send.calls.all(),
  354. call => {
  355. const stanza = call.args[0];
  356. if (!(stanza instanceof Element) || stanza.nodeName !== 'iq') {
  357. return;
  358. }
  359. if (sizzle('items[node="storage:bookmarks"]', stanza).length) {
  360. IQ_id = stanza.getAttribute('id');
  361. return true;
  362. }
  363. }
  364. ).pop()
  365. );
  366. expect(Strophe.serialize(call.args[0])).toBe(
  367. `<iq from="dummy@localhost/resource" id="${IQ_id}" type="get" xmlns="jabber:client">`+
  368. '<pubsub xmlns="http://jabber.org/protocol/pubsub">'+
  369. '<items node="storage:bookmarks"/>'+
  370. '</pubsub>'+
  371. '</iq>');
  372. /*
  373. * Server returns all items
  374. * ------------------------
  375. * <iq type='result'
  376. * to='juliet@capulet.lit/randomID'
  377. * id='retrieve1'>
  378. * <pubsub xmlns='http://jabber.org/protocol/pubsub'>
  379. * <items node='storage:bookmarks'>
  380. * <item id='current'>
  381. * <storage xmlns='storage:bookmarks'>
  382. * <conference name='The Play&apos;s the Thing'
  383. * autojoin='true'
  384. * jid='theplay@conference.shakespeare.lit'>
  385. * <nick>JC</nick>
  386. * </conference>
  387. * </storage>
  388. * </item>
  389. * </items>
  390. * </pubsub>
  391. * </iq>
  392. */
  393. expect(_converse.bookmarks.models.length).toBe(0);
  394. spyOn(_converse.bookmarks, 'onBookmarksReceived').and.callThrough();
  395. var stanza = $iq({'to': _converse.connection.jid, 'type':'result', 'id':IQ_id})
  396. .c('pubsub', {'xmlns': Strophe.NS.PUBSUB})
  397. .c('items', {'node': 'storage:bookmarks'})
  398. .c('item', {'id': 'current'})
  399. .c('storage', {'xmlns': 'storage:bookmarks'})
  400. .c('conference', {
  401. 'name': 'The Play&apos;s the Thing',
  402. 'autojoin': 'true',
  403. 'jid': 'theplay@conference.shakespeare.lit'
  404. }).c('nick').t('JC').up().up()
  405. .c('conference', {
  406. 'name': 'Another room',
  407. 'autojoin': 'false',
  408. 'jid': 'another@conference.shakespeare.lit'
  409. }); // Purposefully exclude the <nick> element to test #1043
  410. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  411. await test_utils.waitUntil(() => _converse.bookmarks.onBookmarksReceived.calls.count());
  412. expect(_converse.bookmarks.models.length).toBe(2);
  413. expect(_converse.bookmarks.findWhere({'jid': 'theplay@conference.shakespeare.lit'}).get('autojoin')).toBe(true);
  414. expect(_converse.bookmarks.findWhere({'jid': 'another@conference.shakespeare.lit'}).get('autojoin')).toBe(false);
  415. done();
  416. }));
  417. describe("The rooms panel", function () {
  418. it("shows a list of bookmarks", mock.initConverse(
  419. {'connection': ['send']}, ['rosterGroupsFetched'], {},
  420. async function (done, _converse) {
  421. await test_utils.waitUntilDiscoConfirmed(
  422. _converse, _converse.bare_jid,
  423. [{'category': 'pubsub', 'type': 'pep'}],
  424. ['http://jabber.org/protocol/pubsub#publish-options']
  425. );
  426. test_utils.openControlBox();
  427. let IQ_id;
  428. const call = await test_utils.waitUntil(() =>
  429. _.filter(
  430. _converse.connection.send.calls.all(),
  431. call => {
  432. const stanza = call.args[0];
  433. if (!(stanza instanceof Element) || stanza.nodeName !== 'iq') {
  434. return;
  435. }
  436. if (sizzle('items[node="storage:bookmarks"]', stanza).length) {
  437. IQ_id = stanza.getAttribute('id');
  438. return true;
  439. }
  440. }
  441. ).pop()
  442. );
  443. expect(Strophe.serialize(call.args[0])).toBe(
  444. `<iq from="dummy@localhost/resource" id="${IQ_id}" type="get" xmlns="jabber:client">`+
  445. '<pubsub xmlns="http://jabber.org/protocol/pubsub">'+
  446. '<items node="storage:bookmarks"/>'+
  447. '</pubsub>'+
  448. '</iq>'
  449. );
  450. const stanza = $iq({'to': _converse.connection.jid, 'type':'result', 'id':IQ_id})
  451. .c('pubsub', {'xmlns': Strophe.NS.PUBSUB})
  452. .c('items', {'node': 'storage:bookmarks'})
  453. .c('item', {'id': 'current'})
  454. .c('storage', {'xmlns': 'storage:bookmarks'})
  455. .c('conference', {
  456. 'name': 'The Play&apos;s the Thing',
  457. 'autojoin': 'false',
  458. 'jid': 'theplay@conference.shakespeare.lit'
  459. }).c('nick').t('JC').up().up()
  460. .c('conference', {
  461. 'name': '1st Bookmark',
  462. 'autojoin': 'false',
  463. 'jid': 'first@conference.shakespeare.lit'
  464. }).c('nick').t('JC').up().up()
  465. .c('conference', {
  466. 'autojoin': 'false',
  467. 'jid': 'noname@conference.shakespeare.lit'
  468. }).c('nick').t('JC').up().up()
  469. .c('conference', {
  470. 'name': 'Bookmark with a very very long name that will be shortened',
  471. 'autojoin': 'false',
  472. 'jid': 'longname@conference.shakespeare.lit'
  473. }).c('nick').t('JC').up().up()
  474. .c('conference', {
  475. 'name': 'Another room',
  476. 'autojoin': 'false',
  477. 'jid': 'another@conference.shakespeare.lit'
  478. }).c('nick').t('JC').up().up();
  479. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  480. await test_utils.waitUntil(() => document.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item').length);
  481. expect(document.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item').length).toBe(5);
  482. let els = document.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item a.list-item-link');
  483. expect(els[0].textContent).toBe("1st Bookmark");
  484. expect(els[1].textContent).toBe("Another room");
  485. expect(els[2].textContent).toBe("Bookmark with a very very long name that will be shortened");
  486. expect(els[3].textContent).toBe("noname@conference.shakespeare.lit");
  487. expect(els[4].textContent).toBe("The Play's the Thing");
  488. spyOn(window, 'confirm').and.returnValue(true);
  489. document.querySelector('#chatrooms .bookmarks.rooms-list .room-item:nth-child(2) a:nth-child(2)').click();
  490. expect(window.confirm).toHaveBeenCalled();
  491. await test_utils.waitUntil(() => document.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item').length === 4)
  492. els = document.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item a.list-item-link');
  493. expect(els[0].textContent).toBe("1st Bookmark");
  494. expect(els[1].textContent).toBe("Bookmark with a very very long name that will be shortened");
  495. expect(els[2].textContent).toBe("noname@conference.shakespeare.lit");
  496. expect(els[3].textContent).toBe("The Play's the Thing");
  497. done();
  498. }));
  499. it("remembers the toggle state of the bookmarks list", mock.initConverse(
  500. {'connection': ['send']}, ['rosterGroupsFetched'], {},
  501. async function (done, _converse) {
  502. test_utils.openControlBox();
  503. await test_utils.waitUntilDiscoConfirmed(
  504. _converse, _converse.bare_jid,
  505. [{'category': 'pubsub', 'type': 'pep'}],
  506. ['http://jabber.org/protocol/pubsub#publish-options']
  507. );
  508. let IQ_id;
  509. const call = await test_utils.waitUntil(() =>
  510. _.filter(
  511. _converse.connection.send.calls.all(),
  512. call => {
  513. const stanza = call.args[0];
  514. if (!(stanza instanceof Element) || stanza.nodeName !== 'iq') {
  515. return;
  516. }
  517. if (sizzle('items[node="storage:bookmarks"]', stanza).length) {
  518. IQ_id = stanza.getAttribute('id');
  519. return true;
  520. }
  521. }
  522. ).pop()
  523. );
  524. expect(Strophe.serialize(call.args[0])).toBe(
  525. `<iq from="dummy@localhost/resource" id="${IQ_id}" type="get" xmlns="jabber:client">`+
  526. '<pubsub xmlns="http://jabber.org/protocol/pubsub">'+
  527. '<items node="storage:bookmarks"/>'+
  528. '</pubsub>'+
  529. '</iq>'
  530. );
  531. const stanza = $iq({'to': _converse.connection.jid, 'type':'result', 'id':IQ_id})
  532. .c('pubsub', {'xmlns': Strophe.NS.PUBSUB})
  533. .c('items', {'node': 'storage:bookmarks'})
  534. .c('item', {'id': 'current'})
  535. .c('storage', {'xmlns': 'storage:bookmarks'});
  536. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  537. _converse.bookmarks.create({
  538. 'jid': 'theplay@conference.shakespeare.lit',
  539. 'autojoin': false,
  540. 'name': 'The Play',
  541. 'nick': ''
  542. });
  543. await test_utils.waitUntil(() => $('#chatrooms .bookmarks.rooms-list .room-item:visible').length);
  544. expect($('#chatrooms .bookmarks.rooms-list').hasClass('collapsed')).toBeFalsy();
  545. expect($('#chatrooms .bookmarks.rooms-list .room-item:visible').length).toBe(1);
  546. expect(_converse.bookmarksview.list_model.get('toggle-state')).toBe(_converse.OPENED);
  547. $('#chatrooms .bookmarks-toggle')[0].click();
  548. expect($('#chatrooms .bookmarks.rooms-list').hasClass('collapsed')).toBeTruthy();
  549. expect(_converse.bookmarksview.list_model.get('toggle-state')).toBe(_converse.CLOSED);
  550. $('#chatrooms .bookmarks-toggle')[0].click();
  551. expect($('#chatrooms .bookmarks.rooms-list').hasClass('collapsed')).toBeFalsy();
  552. expect($('#chatrooms .bookmarks.rooms-list .room-item:visible').length).toBe(1);
  553. expect(_converse.bookmarksview.list_model.get('toggle-state')).toBe(_converse.OPENED);
  554. done();
  555. }));
  556. });
  557. });
  558. describe("When hide_open_bookmarks is true and a bookmarked room is opened", function () {
  559. it("can be closed", mock.initConverse(
  560. null, ['rosterGroupsFetched'],
  561. { hide_open_bookmarks: true },
  562. async function (done, _converse) {
  563. const jid = 'room@conference.example.org';
  564. await test_utils.waitUntilDiscoConfirmed(
  565. _converse, _converse.bare_jid,
  566. [{'category': 'pubsub', 'type': 'pep'}],
  567. ['http://jabber.org/protocol/pubsub#publish-options']
  568. );
  569. // XXX Create bookmarks view here, otherwise we need to mock stanza
  570. // traffic for it to get created.
  571. _converse.bookmarksview = new _converse.BookmarksView(
  572. {'model': _converse.bookmarks}
  573. );
  574. _converse.emit('bookmarksInitialized');
  575. // Check that it's there
  576. _converse.bookmarks.create({
  577. 'jid': jid,
  578. 'autojoin': false,
  579. 'name': 'The Play',
  580. 'nick': ' Othello'
  581. });
  582. expect(_converse.bookmarks.length).toBe(1);
  583. const room_els = _converse.bookmarksview.el.querySelectorAll(".open-room");
  584. expect(room_els.length).toBe(1);
  585. // Check that it disappears once the room is opened
  586. const bookmark = _converse.bookmarksview.el.querySelector(".open-room");
  587. bookmark.click();
  588. await test_utils.waitUntil(() => _converse.chatboxviews.get(jid));
  589. expect(u.hasClass('hidden', _converse.bookmarksview.el.querySelector(".available-chatroom"))).toBeTruthy();
  590. // Check that it reappears once the room is closed
  591. const view = _converse.chatboxviews.get(jid);
  592. view.close();
  593. expect(u.hasClass('hidden', _converse.bookmarksview.el.querySelector(".available-chatroom"))).toBeFalsy();
  594. done();
  595. }));
  596. });
  597. }));