roomslist.js 16 KB

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