roomslist.js 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345
  1. (function (root, factory) {
  2. define(["jasmine", "mock", "test-utils"], factory);
  3. } (this, function (jasmine, mock, test_utils) {
  4. const { Backbone, Promise, Strophe, $iq, $msg, $pres, b64_sha1, sizzle, _ } = converse.env;
  5. const u = converse.env.utils;
  6. describe("A list of open groupchats", function () {
  7. it("is shown in controlbox", mock.initConverse(
  8. null, ['rosterGroupsFetched', 'chatBoxesFetched', 'emojisInitialized'],
  9. { allow_bookmarks: false // Makes testing easier, otherwise we
  10. // have to mock stanza traffic.
  11. }, async function (done, _converse) {
  12. test_utils.openControlBox();
  13. const controlbox = _converse.chatboxviews.get('controlbox');
  14. let list = controlbox.el.querySelector('div.rooms-list-container');
  15. expect(_.includes(list.classList, 'hidden')).toBeTruthy();
  16. await test_utils.openChatRoom(_converse, 'room', 'conference.shakespeare.lit', 'JC');
  17. expect(_converse.rooms_list_view === undefined).toBeFalsy();
  18. const lview = _converse.rooms_list_view
  19. await u.waitUntil(() => lview.el.querySelectorAll(".open-room").length);
  20. let room_els = lview.el.querySelectorAll(".open-room");
  21. expect(room_els.length).toBe(1);
  22. expect(room_els[0].innerText).toBe('room@conference.shakespeare.lit');
  23. await test_utils.openChatRoom(_converse, 'lounge', 'montague.lit', 'romeo');
  24. await u.waitUntil(() => lview.el.querySelectorAll(".open-room").length > 1);
  25. room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
  26. expect(room_els.length).toBe(2);
  27. let view = _converse.chatboxviews.get('room@conference.shakespeare.lit');
  28. view.close();
  29. room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
  30. expect(room_els.length).toBe(1);
  31. expect(room_els[0].innerText).toBe('lounge@montague.lit');
  32. list = controlbox.el.querySelector('div.rooms-list-container');
  33. u.waitUntil(() => _.includes(list.classList, 'hidden'));
  34. view = _converse.chatboxviews.get('lounge@montague.lit');
  35. view.close();
  36. room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
  37. expect(room_els.length).toBe(0);
  38. list = controlbox.el.querySelector('div.rooms-list-container');
  39. expect(_.includes(list.classList, 'hidden')).toBeTruthy();
  40. done();
  41. }
  42. ));
  43. it("uses bookmarks to determine groupchat names",
  44. mock.initConverse(
  45. {'connection': ['send']},
  46. ['rosterGroupsFetched', 'chatBoxesFetched', 'emojisInitialized'],
  47. {'view_mode': 'fullscreen'},
  48. async function (done, _converse) {
  49. await test_utils.openAndEnterChatRoom(_converse, 'lounge@montague.lit', 'romeo');
  50. const view = _converse.chatboxviews.get('lounge@montague.lit');
  51. const contact_jid = 'newguy@montague.lit';
  52. let stanza = $pres({
  53. to: 'romeo@montague.lit/orchard',
  54. from: 'lounge@montague.lit/newguy'
  55. })
  56. .c('x', {xmlns: Strophe.NS.MUC_USER})
  57. .c('item', {
  58. 'affiliation': 'none',
  59. 'jid': 'newguy@montague.lit/_converse.js-290929789',
  60. 'role': 'participant'
  61. }).tree();
  62. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  63. spyOn(_converse.Bookmarks.prototype, 'fetchBookmarks').and.callThrough();
  64. await test_utils.waitUntilDiscoConfirmed(
  65. _converse, _converse.bare_jid,
  66. [{'category': 'pubsub', 'type':'pep'}],
  67. [`${Strophe.NS.PUBSUB}#publish-options`]
  68. );
  69. const call = await u.waitUntil(() =>
  70. _.filter(
  71. _converse.connection.send.calls.all(),
  72. c => sizzle('items[node="storage:bookmarks"]', c.args[0]).length
  73. ).pop()
  74. );
  75. expect(Strophe.serialize(call.args[0])).toBe(
  76. `<iq from="romeo@montague.lit/orchard" id="${call.args[0].getAttribute('id')}" type="get" xmlns="jabber:client">`+
  77. '<pubsub xmlns="http://jabber.org/protocol/pubsub">'+
  78. '<items node="storage:bookmarks"/>'+
  79. '</pubsub>'+
  80. '</iq>');
  81. stanza = $iq({'to': _converse.connection.jid, 'type':'result', 'id':call.args[0].getAttribute('id')})
  82. .c('pubsub', {'xmlns': Strophe.NS.PUBSUB})
  83. .c('items', {'node': 'storage:bookmarks'})
  84. .c('item', {'id': 'current'})
  85. .c('storage', {'xmlns': 'storage:bookmarks'})
  86. .c('conference', {
  87. 'name': 'Bookmarked Lounge',
  88. 'jid': 'lounge@montague.lit'
  89. });
  90. _converse.connection._dataRecv(test_utils.createRequest(stanza));
  91. await _converse.api.waitUntil('roomsListInitialized');
  92. const controlbox = _converse.chatboxviews.get('controlbox');
  93. const list = controlbox.el.querySelector('div.rooms-list-container');
  94. expect(_.includes(list.classList, 'hidden')).toBeFalsy();
  95. const items = list.querySelectorAll('.list-item');
  96. expect(items.length).toBe(1);
  97. expect(items[0].textContent.trim()).toBe('Bookmarked Lounge');
  98. expect(_converse.bookmarks.fetchBookmarks).toHaveBeenCalled();
  99. done();
  100. }));
  101. });
  102. describe("A groupchat shown in the groupchats list", function () {
  103. it("is highlighted if its currently open", mock.initConverse(
  104. null, ['rosterGroupsFetched', 'chatBoxesFetched', 'emojisInitialized'],
  105. { view_mode: 'fullscreen',
  106. allow_bookmarks: false // Makes testing easier, otherwise we have to mock stanza traffic.
  107. }, async function (done, _converse) {
  108. await _converse.api.rooms.open('coven@chat.shakespeare.lit', {'nick': 'some1'});
  109. const lview = _converse.rooms_list_view
  110. await u.waitUntil(() => lview.el.querySelectorAll(".open-room").length);
  111. let room_els = lview.el.querySelectorAll(".available-chatroom");
  112. expect(room_els.length).toBe(1);
  113. let item = room_els[0];
  114. expect(u.hasClass('open', item)).toBe(true);
  115. expect(item.textContent.trim()).toBe('coven@chat.shakespeare.lit');
  116. await _converse.api.rooms.open('balcony@chat.shakespeare.lit', {'nick': 'some1'});
  117. await u.waitUntil(() => lview.el.querySelectorAll(".open-room").length > 1);
  118. room_els = lview.el.querySelectorAll(".open-room");
  119. expect(room_els.length).toBe(2);
  120. room_els = lview.el.querySelectorAll(".available-chatroom.open");
  121. expect(room_els.length).toBe(1);
  122. item = room_els[0];
  123. expect(item.textContent.trim()).toBe('balcony@chat.shakespeare.lit');
  124. const conv_el = document.querySelector('#conversejs');
  125. conv_el.parentElement.removeChild(conv_el);
  126. done();
  127. }));
  128. it("has an info icon which opens a details modal when clicked", mock.initConverse(
  129. null, ['rosterGroupsFetched', 'chatBoxesFetched', 'emojisInitialized'],
  130. { whitelisted_plugins: ['converse-roomslist'],
  131. allow_bookmarks: false // Makes testing easier, otherwise we
  132. // have to mock stanza traffic.
  133. }, async function (done, _converse) {
  134. const IQ_stanzas = _converse.connection.IQ_stanzas;
  135. const room_jid = 'coven@chat.shakespeare.lit';
  136. test_utils.openControlBox();
  137. await _converse.api.rooms.open(room_jid, {'nick': 'some1'});
  138. const view = _converse.chatboxviews.get(room_jid);
  139. const selector = `iq[to="${room_jid}"] query[xmlns="http://jabber.org/protocol/disco#info"]`;
  140. const features_query = await u.waitUntil(() => IQ_stanzas.filter(iq => iq.querySelector(selector)).pop());
  141. const features_stanza = $iq({
  142. 'from': 'coven@chat.shakespeare.lit',
  143. 'id': features_query.getAttribute('id'),
  144. 'to': 'romeo@montague.lit/desktop',
  145. 'type': 'result'
  146. })
  147. .c('query', { 'xmlns': 'http://jabber.org/protocol/disco#info'})
  148. .c('identity', {
  149. 'category': 'conference',
  150. 'name': 'A Dark Cave',
  151. 'type': 'text'
  152. }).up()
  153. .c('feature', {'var': 'http://jabber.org/protocol/muc'}).up()
  154. .c('feature', {'var': 'muc_passwordprotected'}).up()
  155. .c('feature', {'var': 'muc_hidden'}).up()
  156. .c('feature', {'var': 'muc_temporary'}).up()
  157. .c('feature', {'var': 'muc_open'}).up()
  158. .c('feature', {'var': 'muc_unmoderated'}).up()
  159. .c('feature', {'var': 'muc_nonanonymous'}).up()
  160. .c('feature', {'var': 'urn:xmpp:mam:0'}).up()
  161. .c('x', { 'xmlns':'jabber:x:data', 'type':'result'})
  162. .c('field', {'var':'FORM_TYPE', 'type':'hidden'})
  163. .c('value').t('http://jabber.org/protocol/muc#roominfo').up().up()
  164. .c('field', {'type':'text-single', 'var':'muc#roominfo_description', 'label':'Description'})
  165. .c('value').t('This is the description').up().up()
  166. .c('field', {'type':'text-single', 'var':'muc#roominfo_occupants', 'label':'Number of occupants'})
  167. .c('value').t(0);
  168. _converse.connection._dataRecv(test_utils.createRequest(features_stanza));
  169. await u.waitUntil(() => view.model.get('connection_status') === converse.ROOMSTATUS.CONNECTING)
  170. let presence = $pres({
  171. to: _converse.connection.jid,
  172. from: 'coven@chat.shakespeare.lit/some1',
  173. id: 'DC352437-C019-40EC-B590-AF29E879AF97'
  174. }).c('x').attrs({xmlns:'http://jabber.org/protocol/muc#user'})
  175. .c('item').attrs({
  176. affiliation: 'member',
  177. jid: _converse.bare_jid,
  178. role: 'participant'
  179. }).up()
  180. .c('status').attrs({code:'110'});
  181. _converse.connection._dataRecv(test_utils.createRequest(presence));
  182. await u.waitUntil(() => _converse.rooms_list_view.el.querySelectorAll(".open-room").length, 500);
  183. const room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
  184. expect(room_els.length).toBe(1);
  185. const info_el = _converse.rooms_list_view.el.querySelector(".room-info");
  186. info_el.click();
  187. const modal = view.model.room_details_modal;
  188. await u.waitUntil(() => u.isVisible(modal.el), 1000);
  189. let els = modal.el.querySelectorAll('p.room-info');
  190. expect(els[0].textContent).toBe("Name: A Dark Cave")
  191. expect(els[1].textContent).toBe("Groupchat address (JID): coven@chat.shakespeare.lit")
  192. expect(els[2].textContent).toBe("Description: This is the description")
  193. expect(els[3].textContent).toBe("Online users: 1")
  194. const features_list = modal.el.querySelector('.features-list');
  195. expect(features_list.textContent.replace(/(\n|\s{2,})/g, '')).toBe(
  196. 'Password protected - This groupchat requires a password before entry'+
  197. 'Hidden - This groupchat is not publicly searchable'+
  198. 'Open - Anyone can join this groupchat'+
  199. 'Temporary - This groupchat will disappear once the last person leaves'+
  200. 'Not anonymous - All other groupchat participants can see your XMPP address'+
  201. 'Not moderated - Participants entering this groupchat can write right away'
  202. );
  203. presence = $pres({
  204. to: 'romeo@montague.lit/_converse.js-29092160',
  205. from: 'coven@chat.shakespeare.lit/newguy'
  206. })
  207. .c('x', {xmlns: Strophe.NS.MUC_USER})
  208. .c('item', {
  209. 'affiliation': 'none',
  210. 'jid': 'newguy@montague.lit/_converse.js-290929789',
  211. 'role': 'participant'
  212. });
  213. _converse.connection._dataRecv(test_utils.createRequest(presence));
  214. els = modal.el.querySelectorAll('p.room-info');
  215. expect(els[3].textContent).toBe("Online users: 2")
  216. view.model.set({'subject': {'author': 'someone', 'text': 'Hatching dark plots'}});
  217. els = modal.el.querySelectorAll('p.room-info');
  218. expect(els[0].textContent).toBe("Name: A Dark Cave")
  219. expect(els[1].textContent).toBe("Groupchat address (JID): coven@chat.shakespeare.lit")
  220. expect(els[2].textContent).toBe("Description: This is the description")
  221. expect(els[3].textContent).toBe("Topic: Hatching dark plots")
  222. expect(els[4].textContent).toBe("Topic author: someone")
  223. expect(els[5].textContent).toBe("Online users: 2")
  224. done();
  225. }));
  226. it("can be closed", mock.initConverse(
  227. null, ['rosterGroupsFetched', 'emojisInitialized'],
  228. { whitelisted_plugins: ['converse-roomslist'],
  229. allow_bookmarks: false // Makes testing easier, otherwise we have to mock stanza traffic.
  230. },
  231. async function (done, _converse) {
  232. spyOn(window, 'confirm').and.callFake(() => true);
  233. expect(_converse.chatboxes.length).toBe(1);
  234. await test_utils.openChatRoom(_converse, 'lounge', 'conference.shakespeare.lit', 'JC');
  235. expect(_converse.chatboxes.length).toBe(2);
  236. const lview = _converse.rooms_list_view
  237. await u.waitUntil(() => lview.el.querySelectorAll(".open-room").length);
  238. let room_els = lview.el.querySelectorAll(".open-room");
  239. expect(room_els.length).toBe(1);
  240. const close_el = _converse.rooms_list_view.el.querySelector(".close-room");
  241. close_el.click();
  242. expect(window.confirm).toHaveBeenCalledWith(
  243. 'Are you sure you want to leave the groupchat lounge@conference.shakespeare.lit?');
  244. room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
  245. expect(room_els.length).toBe(0);
  246. expect(_converse.chatboxes.length).toBe(1);
  247. done();
  248. }));
  249. it("shows unread messages directed at the user", mock.initConverse(
  250. null, null,
  251. { whitelisted_plugins: ['converse-roomslist'],
  252. allow_bookmarks: false // Makes testing easier, otherwise we have to mock stanza traffic.
  253. }, async (done, _converse) => {
  254. test_utils.openControlBox();
  255. const room_jid = 'kitchen@conference.shakespeare.lit';
  256. await u.waitUntil(() => _converse.rooms_list_view !== undefined, 500);
  257. await test_utils.openAndEnterChatRoom(_converse, 'kitchen@conference.shakespeare.lit', 'romeo');
  258. const view = _converse.chatboxviews.get(room_jid);
  259. view.model.set({'minimized': true});
  260. const contact_jid = mock.cur_names[5].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  261. const nick = mock.chatroom_names[0];
  262. await view.model.onMessage(
  263. $msg({
  264. from: room_jid+'/'+nick,
  265. id: (new Date()).getTime(),
  266. to: 'romeo@montague.lit',
  267. type: 'groupchat'
  268. }).c('body').t('foo').tree());
  269. const lview = _converse.rooms_list_view
  270. await u.waitUntil(() => lview.el.querySelectorAll(".available-chatroom").length, 500);
  271. // If the user isn't mentioned, the counter doesn't get incremented, but the text of the groupchat is bold
  272. let room_el = lview.el.querySelector(".available-chatroom");
  273. expect(_.includes(room_el.classList, 'unread-msgs')).toBeTruthy();
  274. // If the user is mentioned, the counter also gets updated
  275. await view.model.onMessage(
  276. $msg({
  277. from: room_jid+'/'+nick,
  278. id: (new Date()).getTime(),
  279. to: 'romeo@montague.lit',
  280. type: 'groupchat'
  281. }).c('body').t('romeo: Your attention is required').tree()
  282. );
  283. await u.waitUntil(() => _converse.rooms_list_view.el.querySelectorAll(".msgs-indicator").length);
  284. spyOn(view.model, 'incrementUnreadMsgCounter').and.callThrough();
  285. let indicator_el = _converse.rooms_list_view.el.querySelector(".msgs-indicator");
  286. expect(indicator_el.textContent).toBe('1');
  287. await view.model.onMessage(
  288. $msg({
  289. from: room_jid+'/'+nick,
  290. id: (new Date()).getTime(),
  291. to: 'romeo@montague.lit',
  292. type: 'groupchat'
  293. }).c('body').t('romeo: and another thing...').tree()
  294. );
  295. await u.waitUntil(() => view.model.incrementUnreadMsgCounter.calls.count());
  296. indicator_el = _converse.rooms_list_view.el.querySelector(".msgs-indicator");
  297. expect(indicator_el.textContent).toBe('2');
  298. // When the chat gets maximized again, the unread indicators are removed
  299. view.model.set({'minimized': false});
  300. indicator_el = _converse.rooms_list_view.el.querySelector(".msgs-indicator");
  301. expect(indicator_el === null);
  302. room_el = _converse.rooms_list_view.el.querySelector(".available-chatroom");
  303. expect(_.includes(room_el.classList, 'unread-msgs')).toBeFalsy();
  304. done();
  305. }));
  306. });
  307. }));