bookmarks.js 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666
  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. await test_utils.waitUntil(() => _converse.bookmarks);
  145. let jid = 'lounge@localhost';
  146. _converse.bookmarks.create({
  147. 'jid': jid,
  148. 'autojoin': false,
  149. 'name': 'The Lounge',
  150. 'nick': ' Othello'
  151. });
  152. expect(_.isUndefined(_converse.chatboxviews.get(jid))).toBeTruthy();
  153. jid = 'theplay@conference.shakespeare.lit';
  154. _converse.bookmarks.create({
  155. 'jid': jid,
  156. 'autojoin': true,
  157. 'name': 'The Play',
  158. 'nick': ' Othello'
  159. });
  160. expect(_.isUndefined(_converse.chatboxviews.get(jid))).toBeFalsy();
  161. // Check that we don't auto-join if muc_respect_autojoin is false
  162. _converse.muc_respect_autojoin = false;
  163. jid = 'balcony@conference.shakespeare.lit';
  164. _converse.bookmarks.create({
  165. 'jid': jid,
  166. 'autojoin': true,
  167. 'name': 'Balcony',
  168. 'nick': ' Othello'
  169. });
  170. expect(_.isUndefined(_converse.chatboxviews.get(jid))).toBe(true);
  171. done();
  172. }));
  173. describe("when bookmarked", function () {
  174. it("displays that it's bookmarked through its bookmark icon", mock.initConverse(
  175. null, ['rosterGroupsFetched'], {},
  176. async function (done, _converse) {
  177. test_utils.waitUntilDiscoConfirmed(
  178. _converse, _converse.bare_jid,
  179. [{'category': 'pubsub', 'type': 'pep'}],
  180. ['http://jabber.org/protocol/pubsub#publish-options']
  181. );
  182. await test_utils.openChatRoom(_converse, 'lounge', 'localhost', 'dummy');
  183. const view = _converse.chatboxviews.get('lounge@localhost');
  184. await test_utils.waitUntil(() => !_.isNull(view.el.querySelector('.toggle-bookmark')));
  185. var bookmark_icon = view.el.querySelector('.toggle-bookmark');
  186. expect(_.includes(bookmark_icon.classList, 'button-on')).toBeFalsy();
  187. view.model.set('bookmarked', true);
  188. expect(_.includes(bookmark_icon.classList, 'button-on')).toBeTruthy();
  189. view.model.set('bookmarked', false);
  190. expect(_.includes(bookmark_icon.classList, 'button-on')).toBeFalsy();
  191. done();
  192. }));
  193. it("can be unbookmarked", mock.initConverse(
  194. null, ['rosterGroupsFetched'], {},
  195. async function (done, _converse) {
  196. let sent_stanza, IQ_id;
  197. await test_utils.waitUntilDiscoConfirmed(
  198. _converse, _converse.bare_jid,
  199. [{'category': 'pubsub', 'type': 'pep'}],
  200. ['http://jabber.org/protocol/pubsub#publish-options']
  201. );
  202. const sendIQ = _converse.connection.sendIQ;
  203. await test_utils.openChatRoom(_converse, 'theplay', 'conference.shakespeare.lit', 'JC');
  204. const jid = 'theplay@conference.shakespeare.lit';
  205. const view = _converse.chatboxviews.get(jid);
  206. await test_utils.waitUntil(() => !_.isNull(view.el.querySelector('.toggle-bookmark')));
  207. spyOn(view, 'toggleBookmark').and.callThrough();
  208. spyOn(_converse.bookmarks, 'sendBookmarkStanza').and.callThrough();
  209. view.delegateEvents();
  210. _converse.bookmarks.create({
  211. 'jid': view.model.get('jid'),
  212. 'autojoin': false,
  213. 'name': 'The Play',
  214. 'nick': ' Othello'
  215. });
  216. expect(_converse.bookmarks.length).toBe(1);
  217. expect(view.model.get('bookmarked')).toBeTruthy();
  218. var bookmark_icon = view.el.querySelector('.toggle-bookmark');
  219. expect(u.hasClass('button-on', bookmark_icon)).toBeTruthy();
  220. spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
  221. sent_stanza = iq;
  222. IQ_id = sendIQ.bind(this)(iq, callback, errback);
  223. });
  224. spyOn(_converse.connection, 'getUniqueId').and.callThrough();
  225. bookmark_icon.click();
  226. expect(view.toggleBookmark).toHaveBeenCalled();
  227. expect(u.hasClass('button-on', bookmark_icon)).toBeFalsy();
  228. expect(_converse.bookmarks.length).toBe(0);
  229. // Check that an IQ stanza is sent out, containing no
  230. // conferences to bookmark (since we removed the one and
  231. // only bookmark).
  232. expect(sent_stanza.toLocaleString()).toBe(
  233. `<iq from="dummy@localhost/resource" id="${IQ_id}" type="set" xmlns="jabber:client">`+
  234. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  235. `<publish node="storage:bookmarks">`+
  236. `<item id="current">`+
  237. `<storage xmlns="storage:bookmarks"/>`+
  238. `</item>`+
  239. `</publish>`+
  240. `<publish-options>`+
  241. `<x type="submit" xmlns="jabber:x:data">`+
  242. `<field type="hidden" var="FORM_TYPE">`+
  243. `<value>http://jabber.org/protocol/pubsub#publish-options</value>`+
  244. `</field>`+
  245. `<field var="pubsub#persist_items">`+
  246. `<value>true</value>`+
  247. `</field>`+
  248. `<field var="pubsub#access_model">`+
  249. `<value>whitelist</value>`+
  250. `</field>`+
  251. `</x>`+
  252. `</publish-options>`+
  253. `</pubsub>`+
  254. `</iq>`
  255. );
  256. done();
  257. }));
  258. });
  259. describe("and when autojoin is set", function () {
  260. it("will be be opened and joined automatically upon login", mock.initConverse(
  261. null, ['rosterGroupsFetched'], {},
  262. async function (done, _converse) {
  263. await test_utils.waitUntilDiscoConfirmed(
  264. _converse, _converse.bare_jid,
  265. [{'category': 'pubsub', 'type': 'pep'}],
  266. ['http://jabber.org/protocol/pubsub#publish-options']
  267. );
  268. spyOn(_converse.api.rooms, 'create').and.callThrough();
  269. const jid = 'theplay@conference.shakespeare.lit';
  270. const model = _converse.bookmarks.create({
  271. 'jid': jid,
  272. 'autojoin': false,
  273. 'name': 'The Play',
  274. 'nick': ''
  275. });
  276. expect(_converse.api.rooms.create).not.toHaveBeenCalled();
  277. _converse.bookmarks.remove(model);
  278. _converse.bookmarks.create({
  279. 'jid': jid,
  280. 'autojoin': true,
  281. 'name': 'Hamlet',
  282. 'nick': ''
  283. });
  284. expect(_converse.api.rooms.create).toHaveBeenCalled();
  285. done();
  286. }));
  287. });
  288. });
  289. describe("Bookmarks", function () {
  290. it("can be pushed from the XMPP server", mock.initConverse(
  291. ['send'], ['rosterGroupsFetched', 'connected'], {},
  292. async function (done, _converse) {
  293. await test_utils.waitUntilDiscoConfirmed(
  294. _converse, _converse.bare_jid,
  295. [{'category': 'pubsub', 'type': 'pep'}],
  296. ['http://jabber.org/protocol/pubsub#publish-options']
  297. );
  298. await test_utils.waitUntil(() => _converse.bookmarks);
  299. // Emit here instead of mocking fetching of bookmarks.
  300. _converse.api.trigger('bookmarksInitialized');
  301. /* The stored data is automatically pushed to all of the user's
  302. * connected resources.
  303. *
  304. * Publisher receives event notification
  305. * -------------------------------------
  306. * <message from='juliet@capulet.lit'
  307. * to='juliet@capulet.lit/balcony'
  308. * type='headline'
  309. * id='rnfoo1'>
  310. * <event xmlns='http://jabber.org/protocol/pubsub#event'>
  311. * <items node='storage:bookmarks'>
  312. * <item id='current'>
  313. * <storage xmlns='storage:bookmarks'>
  314. * <conference name='The Play&apos;s the Thing'
  315. * autojoin='true'
  316. * jid='theplay@conference.shakespeare.lit'>
  317. * <nick>JC</nick>
  318. * </conference>
  319. * </storage>
  320. * </item>
  321. * </items>
  322. * </event>
  323. * </message>
  324. */
  325. var stanza = $msg({
  326. 'from': 'dummy@localhost',
  327. 'to': 'dummy@localhost/resource',
  328. 'type': 'headline',
  329. 'id': 'rnfoo1'
  330. }).c('event', {'xmlns': 'http://jabber.org/protocol/pubsub#event'})
  331. .c('items', {'node': 'storage:bookmarks'})
  332. .c('item', {'id': 'current'})
  333. .c('storage', {'xmlns': 'storage:bookmarks'})
  334. .c('conference', {'name': 'The Play&apos;s the Thing',
  335. 'autojoin': 'true',
  336. 'jid':'theplay@conference.shakespeare.lit'})
  337. .c('nick').t('JC');
  338. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  339. await test_utils.waitUntil(() => _converse.bookmarks.length);
  340. expect(_converse.bookmarks.length).toBe(1);
  341. expect(_converse.chatboxviews.get('theplay@conference.shakespeare.lit')).not.toBeUndefined();
  342. done();
  343. }));
  344. it("can be retrieved from the XMPP server", mock.initConverse(
  345. {'connection': ['send']}, ['chatBoxesFetched', 'roomsPanelRendered', 'rosterGroupsFetched'], {},
  346. async function (done, _converse) {
  347. await test_utils.waitUntilDiscoConfirmed(
  348. _converse, _converse.bare_jid,
  349. [{'category': 'pubsub', 'type': 'pep'}],
  350. ['http://jabber.org/protocol/pubsub#publish-options']
  351. );
  352. /* Client requests all items
  353. * -------------------------
  354. *
  355. * <iq from='juliet@capulet.lit/randomID' type='get' id='retrieve1'>
  356. * <pubsub xmlns='http://jabber.org/protocol/pubsub'>
  357. * <items node='storage:bookmarks'/>
  358. * </pubsub>
  359. * </iq>
  360. */
  361. let IQ_id;
  362. const call = await test_utils.waitUntil(() =>
  363. _.filter(
  364. _converse.connection.send.calls.all(),
  365. call => {
  366. const stanza = call.args[0];
  367. if (!(stanza instanceof Element) || stanza.nodeName !== 'iq') {
  368. return;
  369. }
  370. if (sizzle('items[node="storage:bookmarks"]', stanza).length) {
  371. IQ_id = stanza.getAttribute('id');
  372. return true;
  373. }
  374. }
  375. ).pop()
  376. );
  377. expect(Strophe.serialize(call.args[0])).toBe(
  378. `<iq from="dummy@localhost/resource" id="${IQ_id}" type="get" xmlns="jabber:client">`+
  379. '<pubsub xmlns="http://jabber.org/protocol/pubsub">'+
  380. '<items node="storage:bookmarks"/>'+
  381. '</pubsub>'+
  382. '</iq>');
  383. /*
  384. * Server returns all items
  385. * ------------------------
  386. * <iq type='result'
  387. * to='juliet@capulet.lit/randomID'
  388. * id='retrieve1'>
  389. * <pubsub xmlns='http://jabber.org/protocol/pubsub'>
  390. * <items node='storage:bookmarks'>
  391. * <item id='current'>
  392. * <storage xmlns='storage:bookmarks'>
  393. * <conference name='The Play&apos;s the Thing'
  394. * autojoin='true'
  395. * jid='theplay@conference.shakespeare.lit'>
  396. * <nick>JC</nick>
  397. * </conference>
  398. * </storage>
  399. * </item>
  400. * </items>
  401. * </pubsub>
  402. * </iq>
  403. */
  404. expect(_converse.bookmarks.models.length).toBe(0);
  405. spyOn(_converse.bookmarks, 'onBookmarksReceived').and.callThrough();
  406. var stanza = $iq({'to': _converse.connection.jid, 'type':'result', 'id':IQ_id})
  407. .c('pubsub', {'xmlns': Strophe.NS.PUBSUB})
  408. .c('items', {'node': 'storage:bookmarks'})
  409. .c('item', {'id': 'current'})
  410. .c('storage', {'xmlns': 'storage:bookmarks'})
  411. .c('conference', {
  412. 'name': 'The Play&apos;s the Thing',
  413. 'autojoin': 'true',
  414. 'jid': 'theplay@conference.shakespeare.lit'
  415. }).c('nick').t('JC').up().up()
  416. .c('conference', {
  417. 'name': 'Another room',
  418. 'autojoin': 'false',
  419. 'jid': 'another@conference.shakespeare.lit'
  420. }); // Purposefully exclude the <nick> element to test #1043
  421. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  422. await test_utils.waitUntil(() => _converse.bookmarks.onBookmarksReceived.calls.count());
  423. expect(_converse.bookmarks.models.length).toBe(2);
  424. expect(_converse.bookmarks.findWhere({'jid': 'theplay@conference.shakespeare.lit'}).get('autojoin')).toBe(true);
  425. expect(_converse.bookmarks.findWhere({'jid': 'another@conference.shakespeare.lit'}).get('autojoin')).toBe(false);
  426. done();
  427. }));
  428. describe("The rooms panel", function () {
  429. it("shows a list of bookmarks", mock.initConverse(
  430. {'connection': ['send']}, ['rosterGroupsFetched'], {},
  431. async function (done, _converse) {
  432. await test_utils.waitUntilDiscoConfirmed(
  433. _converse, _converse.bare_jid,
  434. [{'category': 'pubsub', 'type': 'pep'}],
  435. ['http://jabber.org/protocol/pubsub#publish-options']
  436. );
  437. test_utils.openControlBox();
  438. let IQ_id;
  439. const call = await test_utils.waitUntil(() =>
  440. _.filter(
  441. _converse.connection.send.calls.all(),
  442. call => {
  443. const stanza = call.args[0];
  444. if (!(stanza instanceof Element) || stanza.nodeName !== 'iq') {
  445. return;
  446. }
  447. if (sizzle('items[node="storage:bookmarks"]', stanza).length) {
  448. IQ_id = stanza.getAttribute('id');
  449. return true;
  450. }
  451. }
  452. ).pop()
  453. );
  454. expect(Strophe.serialize(call.args[0])).toBe(
  455. `<iq from="dummy@localhost/resource" id="${IQ_id}" type="get" xmlns="jabber:client">`+
  456. '<pubsub xmlns="http://jabber.org/protocol/pubsub">'+
  457. '<items node="storage:bookmarks"/>'+
  458. '</pubsub>'+
  459. '</iq>'
  460. );
  461. const stanza = $iq({'to': _converse.connection.jid, 'type':'result', 'id':IQ_id})
  462. .c('pubsub', {'xmlns': Strophe.NS.PUBSUB})
  463. .c('items', {'node': 'storage:bookmarks'})
  464. .c('item', {'id': 'current'})
  465. .c('storage', {'xmlns': 'storage:bookmarks'})
  466. .c('conference', {
  467. 'name': 'The Play&apos;s the Thing',
  468. 'autojoin': 'false',
  469. 'jid': 'theplay@conference.shakespeare.lit'
  470. }).c('nick').t('JC').up().up()
  471. .c('conference', {
  472. 'name': '1st Bookmark',
  473. 'autojoin': 'false',
  474. 'jid': 'first@conference.shakespeare.lit'
  475. }).c('nick').t('JC').up().up()
  476. .c('conference', {
  477. 'autojoin': 'false',
  478. 'jid': 'noname@conference.shakespeare.lit'
  479. }).c('nick').t('JC').up().up()
  480. .c('conference', {
  481. 'name': 'Bookmark with a very very long name that will be shortened',
  482. 'autojoin': 'false',
  483. 'jid': 'longname@conference.shakespeare.lit'
  484. }).c('nick').t('JC').up().up()
  485. .c('conference', {
  486. 'name': 'Another room',
  487. 'autojoin': 'false',
  488. 'jid': 'another@conference.shakespeare.lit'
  489. }).c('nick').t('JC').up().up();
  490. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  491. await test_utils.waitUntil(() => document.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item').length);
  492. expect(document.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item').length).toBe(5);
  493. let els = document.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item a.list-item-link');
  494. expect(els[0].textContent).toBe("1st Bookmark");
  495. expect(els[1].textContent).toBe("Another room");
  496. expect(els[2].textContent).toBe("Bookmark with a very very long name that will be shortened");
  497. expect(els[3].textContent).toBe("noname@conference.shakespeare.lit");
  498. expect(els[4].textContent).toBe("The Play's the Thing");
  499. spyOn(window, 'confirm').and.returnValue(true);
  500. document.querySelector('#chatrooms .bookmarks.rooms-list .room-item:nth-child(2) a:nth-child(2)').click();
  501. expect(window.confirm).toHaveBeenCalled();
  502. await test_utils.waitUntil(() => document.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item').length === 4)
  503. els = document.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item a.list-item-link');
  504. expect(els[0].textContent).toBe("1st Bookmark");
  505. expect(els[1].textContent).toBe("Bookmark with a very very long name that will be shortened");
  506. expect(els[2].textContent).toBe("noname@conference.shakespeare.lit");
  507. expect(els[3].textContent).toBe("The Play's the Thing");
  508. done();
  509. }));
  510. it("remembers the toggle state of the bookmarks list", mock.initConverse(
  511. {'connection': ['send']}, ['rosterGroupsFetched'], {},
  512. async function (done, _converse) {
  513. test_utils.openControlBox();
  514. await test_utils.waitUntilDiscoConfirmed(
  515. _converse, _converse.bare_jid,
  516. [{'category': 'pubsub', 'type': 'pep'}],
  517. ['http://jabber.org/protocol/pubsub#publish-options']
  518. );
  519. let IQ_id;
  520. const call = await test_utils.waitUntil(() =>
  521. _.filter(
  522. _converse.connection.send.calls.all(),
  523. call => {
  524. const stanza = call.args[0];
  525. if (!(stanza instanceof Element) || stanza.nodeName !== 'iq') {
  526. return;
  527. }
  528. if (sizzle('items[node="storage:bookmarks"]', stanza).length) {
  529. IQ_id = stanza.getAttribute('id');
  530. return true;
  531. }
  532. }
  533. ).pop()
  534. );
  535. expect(Strophe.serialize(call.args[0])).toBe(
  536. `<iq from="dummy@localhost/resource" id="${IQ_id}" type="get" xmlns="jabber:client">`+
  537. '<pubsub xmlns="http://jabber.org/protocol/pubsub">'+
  538. '<items node="storage:bookmarks"/>'+
  539. '</pubsub>'+
  540. '</iq>'
  541. );
  542. const stanza = $iq({'to': _converse.connection.jid, 'type':'result', 'id':IQ_id})
  543. .c('pubsub', {'xmlns': Strophe.NS.PUBSUB})
  544. .c('items', {'node': 'storage:bookmarks'})
  545. .c('item', {'id': 'current'})
  546. .c('storage', {'xmlns': 'storage:bookmarks'});
  547. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  548. _converse.bookmarks.create({
  549. 'jid': 'theplay@conference.shakespeare.lit',
  550. 'autojoin': false,
  551. 'name': 'The Play',
  552. 'nick': ''
  553. });
  554. await test_utils.waitUntil(() => $('#chatrooms .bookmarks.rooms-list .room-item:visible').length);
  555. expect($('#chatrooms .bookmarks.rooms-list').hasClass('collapsed')).toBeFalsy();
  556. expect($('#chatrooms .bookmarks.rooms-list .room-item:visible').length).toBe(1);
  557. expect(_converse.bookmarksview.list_model.get('toggle-state')).toBe(_converse.OPENED);
  558. $('#chatrooms .bookmarks-toggle')[0].click();
  559. expect($('#chatrooms .bookmarks.rooms-list').hasClass('collapsed')).toBeTruthy();
  560. expect(_converse.bookmarksview.list_model.get('toggle-state')).toBe(_converse.CLOSED);
  561. $('#chatrooms .bookmarks-toggle')[0].click();
  562. expect($('#chatrooms .bookmarks.rooms-list').hasClass('collapsed')).toBeFalsy();
  563. expect($('#chatrooms .bookmarks.rooms-list .room-item:visible').length).toBe(1);
  564. expect(_converse.bookmarksview.list_model.get('toggle-state')).toBe(_converse.OPENED);
  565. done();
  566. }));
  567. });
  568. });
  569. describe("When hide_open_bookmarks is true and a bookmarked room is opened", function () {
  570. it("can be closed", mock.initConverse(
  571. null, ['rosterGroupsFetched'],
  572. { hide_open_bookmarks: true },
  573. async function (done, _converse) {
  574. const jid = 'room@conference.example.org';
  575. await test_utils.waitUntilDiscoConfirmed(
  576. _converse, _converse.bare_jid,
  577. [{'category': 'pubsub', 'type': 'pep'}],
  578. ['http://jabber.org/protocol/pubsub#publish-options']
  579. );
  580. // XXX Create bookmarks view here, otherwise we need to mock stanza
  581. // traffic for it to get created.
  582. _converse.bookmarksview = new _converse.BookmarksView(
  583. {'model': _converse.bookmarks}
  584. );
  585. _converse.api.trigger('bookmarksInitialized');
  586. // Check that it's there
  587. _converse.bookmarks.create({
  588. 'jid': jid,
  589. 'autojoin': false,
  590. 'name': 'The Play',
  591. 'nick': ' Othello'
  592. });
  593. expect(_converse.bookmarks.length).toBe(1);
  594. const room_els = _converse.bookmarksview.el.querySelectorAll(".open-room");
  595. expect(room_els.length).toBe(1);
  596. // Check that it disappears once the room is opened
  597. const bookmark = _converse.bookmarksview.el.querySelector(".open-room");
  598. bookmark.click();
  599. await test_utils.waitUntil(() => _converse.chatboxviews.get(jid));
  600. expect(u.hasClass('hidden', _converse.bookmarksview.el.querySelector(".available-chatroom"))).toBeTruthy();
  601. // Check that it reappears once the room is closed
  602. const view = _converse.chatboxviews.get(jid);
  603. view.close();
  604. expect(u.hasClass('hidden', _converse.bookmarksview.el.querySelector(".available-chatroom"))).toBeFalsy();
  605. done();
  606. }));
  607. });
  608. }));