2
0

chatbox.js 101 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734
  1. (function (root, factory) {
  2. define([
  3. "utils",
  4. "converse-api",
  5. "mock",
  6. "test_utils"
  7. ], factory);
  8. } (this, function (utils, converse_api, mock, test_utils) {
  9. "use strict";
  10. var _ = converse_api.env._;
  11. var $ = converse_api.env.jQuery;
  12. var $msg = converse_api.env.$msg;
  13. var Strophe = converse_api.env.Strophe;
  14. var moment = converse_api.env.moment;
  15. return describe("Chatboxes", function() {
  16. describe("A Chatbox", function () {
  17. afterEach(function () {
  18. converse_api.user.logout();
  19. converse_api.listen.not();
  20. test_utils.clearBrowserStorage();
  21. });
  22. it("is created when you click on a roster item", mock.initConverse(function (converse) {
  23. test_utils.createContacts(converse, 'current');
  24. test_utils.openControlBox();
  25. test_utils.openContactsPanel(converse);
  26. var i, $el, jid, chatboxview;
  27. // openControlBox was called earlier, so the controlbox is
  28. // visible, but no other chat boxes have been created.
  29. expect(converse.chatboxes.length).toEqual(1);
  30. spyOn(converse.chatboxviews, 'trimChats');
  31. expect($("#conversejs .chatbox").length).toBe(1); // Controlbox is open
  32. var online_contacts = converse.rosterview.$el.find('dt.roster-group').siblings('dd.current-xmpp-contact').find('a.open-chat');
  33. for (i=0; i<online_contacts.length; i++) {
  34. $el = $(online_contacts[i]);
  35. jid = $el.text().replace(/ /g,'.').toLowerCase() + '@localhost';
  36. $el.click();
  37. chatboxview = converse.chatboxviews.get(jid);
  38. expect(converse.chatboxes.length).toEqual(i+2);
  39. expect(converse.chatboxviews.trimChats).toHaveBeenCalled();
  40. // Check that new chat boxes are created to the left of the
  41. // controlbox (but to the right of all existing chat boxes)
  42. expect($("#conversejs .chatbox").length).toBe(i+2);
  43. expect($("#conversejs .chatbox")[1].id).toBe(chatboxview.model.get('box_id'));
  44. }
  45. }));
  46. it("can be trimmed to conserve space", mock.initConverse(function (converse) {
  47. test_utils.createContacts(converse, 'current');
  48. test_utils.openControlBox();
  49. test_utils.openContactsPanel(converse);
  50. var i, $el, jid, chatbox, chatboxview, trimmedview;
  51. // openControlBox was called earlier, so the controlbox is
  52. // visible, but no other chat boxes have been created.
  53. var trimmed_chatboxes = converse.minimized_chats;
  54. expect(converse.chatboxes.length).toEqual(1);
  55. spyOn(converse.chatboxviews, 'trimChats');
  56. spyOn(trimmed_chatboxes, 'addChat').andCallThrough();
  57. spyOn(trimmed_chatboxes, 'removeChat').andCallThrough();
  58. expect($("#conversejs .chatbox").length).toBe(1); // Controlbox is open
  59. // Test that they can be trimmed
  60. runs(function () {
  61. converse.rosterview.update(); // XXX: Hack to make sure $roster element is attaced.
  62. });
  63. waits(50);
  64. runs(function () {
  65. // Test that they can be maximized again
  66. var online_contacts = converse.rosterview.$el.find('dt.roster-group').siblings('dd.current-xmpp-contact').find('a.open-chat');
  67. for (i=0; i<online_contacts.length; i++) {
  68. $el = $(online_contacts[i]);
  69. jid = $el.text().replace(/ /g,'.').toLowerCase() + '@localhost';
  70. $el.click();
  71. expect(converse.chatboxviews.trimChats).toHaveBeenCalled();
  72. chatboxview = converse.chatboxviews.get(jid);
  73. spyOn(chatboxview, 'minimize').andCallThrough();
  74. chatboxview.model.set({'minimized': true});
  75. expect(trimmed_chatboxes.addChat).toHaveBeenCalled();
  76. expect(chatboxview.minimize).toHaveBeenCalled();
  77. trimmedview = trimmed_chatboxes.get(jid);
  78. }
  79. var key = converse.chatboxviews.keys()[1];
  80. trimmedview = trimmed_chatboxes.get(key);
  81. chatbox = trimmedview.model;
  82. spyOn(chatbox, 'maximize').andCallThrough();
  83. spyOn(trimmedview, 'restore').andCallThrough();
  84. trimmedview.delegateEvents();
  85. trimmedview.$("a.restore-chat").click();
  86. });
  87. waits(250);
  88. runs(function () {
  89. expect(trimmedview.restore).toHaveBeenCalled();
  90. expect(chatbox.maximize).toHaveBeenCalled();
  91. expect(converse.chatboxviews.trimChats).toHaveBeenCalled();
  92. });
  93. }));
  94. it("is focused if its already open and you click on its corresponding roster item", mock.initConverse(function (converse) {
  95. test_utils.createContacts(converse, 'current');
  96. test_utils.openControlBox();
  97. test_utils.openContactsPanel(converse);
  98. var contact_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost';
  99. var $el, jid, chatboxview, chatbox;
  100. // openControlBox was called earlier, so the controlbox is
  101. // visible, but no other chat boxes have been created.
  102. expect(converse.chatboxes.length).toEqual(1);
  103. chatbox = test_utils.openChatBoxFor(converse, contact_jid);
  104. chatboxview = converse.chatboxviews.get(contact_jid);
  105. spyOn(chatboxview, 'focus');
  106. // Test that they can be trimmed
  107. runs(function () {
  108. converse.rosterview.update(); // XXX: Hack to make sure $roster element is attaced.
  109. });
  110. waits(300); // ChatBox.show() is debounced for 250ms
  111. runs(function () {
  112. $el = converse.rosterview.$el.find('a.open-chat:contains("'+chatbox.get('fullname')+'")');
  113. jid = $el.text().replace(/ /g,'.').toLowerCase() + '@localhost';
  114. $el.click();
  115. expect(converse.chatboxes.length).toEqual(2);
  116. expect(chatboxview.focus).toHaveBeenCalled();
  117. });
  118. }));
  119. it("can be saved to, and retrieved from, browserStorage", mock.initConverse(function (converse) {
  120. test_utils.createContacts(converse, 'current');
  121. test_utils.openControlBox();
  122. test_utils.openContactsPanel(converse);
  123. spyOn(converse, 'emit');
  124. spyOn(converse.chatboxviews, 'trimChats');
  125. runs(function () {
  126. test_utils.openControlBox();
  127. });
  128. waits(250);
  129. runs(function () {
  130. test_utils.openChatBoxes(converse, 6);
  131. expect(converse.chatboxviews.trimChats).toHaveBeenCalled();
  132. // We instantiate a new ChatBoxes collection, which by default
  133. // will be empty.
  134. var newchatboxes = new converse.ChatBoxes();
  135. expect(newchatboxes.length).toEqual(0);
  136. // The chatboxes will then be fetched from browserStorage inside the
  137. // onConnected method
  138. newchatboxes.onConnected();
  139. expect(newchatboxes.length).toEqual(7);
  140. // Check that the chatboxes items retrieved from browserStorage
  141. // have the same attributes values as the original ones.
  142. var attrs = ['id', 'box_id', 'visible'];
  143. var new_attrs, old_attrs;
  144. for (var i=0; i<attrs.length; i++) {
  145. new_attrs = _.map(_.map(newchatboxes.models, 'attributes'), attrs[i]);
  146. old_attrs = _.map(_.map(converse.chatboxes.models, 'attributes'), attrs[i]);
  147. expect(_.isEqual(new_attrs, old_attrs)).toEqual(true);
  148. }
  149. converse.rosterview.render();
  150. });
  151. }));
  152. it("can be closed by clicking a DOM element with class 'close-chatbox-button'", mock.initConverse(function (converse) {
  153. test_utils.createContacts(converse, 'current');
  154. test_utils.openControlBox();
  155. test_utils.openContactsPanel(converse);
  156. var chatbox = test_utils.openChatBoxes(converse, 1)[0],
  157. controlview = converse.chatboxviews.get('controlbox'), // The controlbox is currently open
  158. chatview = converse.chatboxviews.get(chatbox.get('jid'));
  159. spyOn(chatview, 'close').andCallThrough();
  160. spyOn(controlview, 'close').andCallThrough();
  161. spyOn(converse, 'emit');
  162. // We need to rebind all events otherwise our spy won't be called
  163. controlview.delegateEvents();
  164. chatview.delegateEvents();
  165. runs(function () {
  166. controlview.$el.find('.close-chatbox-button').click();
  167. });
  168. waits(250);
  169. runs(function () {
  170. expect(controlview.close).toHaveBeenCalled();
  171. expect(converse.emit).toHaveBeenCalledWith('chatBoxClosed', jasmine.any(Object));
  172. expect(converse.emit.callCount, 1);
  173. chatview.$el.find('.close-chatbox-button').click();
  174. });
  175. waits(250);
  176. runs(function () {
  177. expect(chatview.close).toHaveBeenCalled();
  178. expect(converse.emit).toHaveBeenCalledWith('chatBoxClosed', jasmine.any(Object));
  179. expect(converse.emit.callCount, 2);
  180. });
  181. }));
  182. it("can be minimized by clicking a DOM element with class 'toggle-chatbox-button'", mock.initConverse(function (converse) {
  183. test_utils.createContacts(converse, 'current');
  184. test_utils.openControlBox();
  185. test_utils.openContactsPanel(converse);
  186. var chatbox = test_utils.openChatBoxes(converse, 1)[0],
  187. chatview = converse.chatboxviews.get(chatbox.get('jid')),
  188. trimmed_chatboxes = converse.minimized_chats,
  189. trimmedview;
  190. spyOn(chatview, 'minimize').andCallThrough();
  191. spyOn(converse, 'emit');
  192. // We need to rebind all events otherwise our spy won't be called
  193. chatview.delegateEvents();
  194. runs(function () {
  195. chatview.$el.find('.toggle-chatbox-button').click();
  196. });
  197. waits(250);
  198. runs(function () {
  199. expect(chatview.minimize).toHaveBeenCalled();
  200. expect(converse.emit).toHaveBeenCalledWith('chatBoxMinimized', jasmine.any(Object));
  201. expect(converse.emit.callCount, 2);
  202. expect(chatview.$el.is(':visible')).toBeFalsy();
  203. expect(chatview.model.get('minimized')).toBeTruthy();
  204. chatview.$el.find('.toggle-chatbox-button').click();
  205. trimmedview = trimmed_chatboxes.get(chatview.model.get('id'));
  206. spyOn(trimmedview, 'restore').andCallThrough();
  207. trimmedview.delegateEvents();
  208. trimmedview.$("a.restore-chat").click();
  209. });
  210. waits(250);
  211. runs(function () {
  212. expect(trimmedview.restore).toHaveBeenCalled();
  213. expect(converse.emit).toHaveBeenCalledWith('chatBoxMaximized', jasmine.any(Object));
  214. expect(chatview.$el.find('.chat-body').is(':visible')).toBeTruthy();
  215. expect(chatview.$el.find('.toggle-chatbox-button').hasClass('icon-minus')).toBeTruthy();
  216. expect(chatview.$el.find('.toggle-chatbox-button').hasClass('icon-plus')).toBeFalsy();
  217. expect(chatview.model.get('minimized')).toBeFalsy();
  218. });
  219. }));
  220. it("will be removed from browserStorage when closed", mock.initConverse(function (converse) {
  221. test_utils.createContacts(converse, 'current');
  222. test_utils.openControlBox();
  223. test_utils.openContactsPanel(converse);
  224. spyOn(converse, 'emit');
  225. spyOn(converse.chatboxviews, 'trimChats');
  226. converse.chatboxes.browserStorage._clear();
  227. runs(function () {
  228. test_utils.closeControlBox();
  229. });
  230. waits(50);
  231. runs(function () {
  232. expect(converse.emit).toHaveBeenCalledWith('chatBoxClosed', jasmine.any(Object));
  233. expect(converse.chatboxes.length).toEqual(1);
  234. expect(converse.chatboxes.pluck('id')).toEqual(['controlbox']);
  235. test_utils.openChatBoxes(converse, 6);
  236. expect(converse.chatboxviews.trimChats).toHaveBeenCalled();
  237. expect(converse.chatboxes.length).toEqual(7);
  238. expect(converse.emit).toHaveBeenCalledWith('chatBoxOpened', jasmine.any(Object));
  239. test_utils.closeAllChatBoxes(converse);
  240. });
  241. waits(50);
  242. runs(function () {
  243. expect(converse.chatboxes.length).toEqual(1);
  244. expect(converse.chatboxes.pluck('id')).toEqual(['controlbox']);
  245. expect(converse.emit).toHaveBeenCalledWith('chatBoxClosed', jasmine.any(Object));
  246. var newchatboxes = new converse.ChatBoxes();
  247. expect(newchatboxes.length).toEqual(0);
  248. expect(converse.chatboxes.pluck('id')).toEqual(['controlbox']);
  249. // onConnected will fetch chatboxes in browserStorage, but
  250. // because there aren't any open chatboxes, there won't be any
  251. // in browserStorage either. XXX except for the controlbox
  252. newchatboxes.onConnected();
  253. expect(newchatboxes.length).toEqual(1);
  254. expect(newchatboxes.models[0].id).toBe("controlbox");
  255. });
  256. }));
  257. describe("A chat toolbar", function () {
  258. afterEach(function () {
  259. converse_api.user.logout();
  260. converse_api.listen.not();
  261. test_utils.clearBrowserStorage();
  262. });
  263. it("can be found on each chat box", mock.initConverse(function (converse) {
  264. test_utils.createContacts(converse, 'current');
  265. test_utils.openControlBox();
  266. test_utils.openContactsPanel(converse);
  267. var contact_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost';
  268. test_utils.openChatBoxFor(converse, contact_jid);
  269. var chatbox = converse.chatboxes.get(contact_jid);
  270. var view = converse.chatboxviews.get(contact_jid);
  271. expect(chatbox).toBeDefined();
  272. expect(view).toBeDefined();
  273. var $toolbar = view.$el.find('ul.chat-toolbar');
  274. expect($toolbar.length).toBe(1);
  275. expect($toolbar.children('li').length).toBe(3);
  276. }));
  277. it("contains a button for inserting emoticons", mock.initConverse(function (converse) {
  278. test_utils.createContacts(converse, 'current');
  279. test_utils.openControlBox();
  280. test_utils.openContactsPanel(converse);
  281. var contact_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost',
  282. view, $toolbar, $textarea;
  283. waits(300); // ChatBox.show() is debounced for 250ms
  284. runs(function () {
  285. test_utils.openChatBoxFor(converse, contact_jid);
  286. view = converse.chatboxviews.get(contact_jid);
  287. $toolbar = view.$el.find('ul.chat-toolbar');
  288. $textarea = view.$el.find('textarea.chat-textarea');
  289. expect($toolbar.children('li.toggle-smiley').length).toBe(1);
  290. // Register spies
  291. spyOn(view, 'toggleEmoticonMenu').andCallThrough();
  292. spyOn(view, 'insertEmoticon').andCallThrough();
  293. view.delegateEvents(); // We need to rebind all events otherwise our spy won't be called
  294. $toolbar.children('li.toggle-smiley').click();
  295. });
  296. waits(250);
  297. runs(function () {
  298. expect(view.toggleEmoticonMenu).toHaveBeenCalled();
  299. var $menu = view.$el.find('.toggle-smiley ul');
  300. var $items = $menu.children('li');
  301. expect($menu.is(':visible')).toBeTruthy();
  302. expect($items.length).toBe(13);
  303. expect($($items[0]).children('a').data('emoticon')).toBe(':)');
  304. expect($($items[1]).children('a').data('emoticon')).toBe(';)');
  305. expect($($items[2]).children('a').data('emoticon')).toBe(':D');
  306. expect($($items[3]).children('a').data('emoticon')).toBe(':P');
  307. expect($($items[4]).children('a').data('emoticon')).toBe('8)');
  308. expect($($items[5]).children('a').data('emoticon')).toBe('>:)');
  309. expect($($items[6]).children('a').data('emoticon')).toBe(':S');
  310. expect($($items[7]).children('a').data('emoticon')).toBe(':\\');
  311. expect($($items[8]).children('a').data('emoticon')).toBe('>:(');
  312. expect($($items[9]).children('a').data('emoticon')).toBe(':(');
  313. expect($($items[10]).children('a').data('emoticon')).toBe(':O');
  314. expect($($items[11]).children('a').data('emoticon')).toBe('(^.^)b');
  315. expect($($items[12]).children('a').data('emoticon')).toBe('<3');
  316. $items.first().click();
  317. });
  318. waits(250);
  319. runs(function () {
  320. expect(view.insertEmoticon).toHaveBeenCalled();
  321. expect($textarea.val()).toBe(':) ');
  322. expect(view.$el.find('.toggle-smiley ul').is(':visible')).toBeFalsy();
  323. $toolbar.children('li.toggle-smiley').click();
  324. });
  325. waits(250);
  326. runs(function () {
  327. expect(view.toggleEmoticonMenu).toHaveBeenCalled();
  328. expect(view.$el.find('.toggle-smiley ul').is(':visible')).toBeTruthy();
  329. view.$el.find('.toggle-smiley ul').children('li').last().click();
  330. });
  331. waits(250);
  332. runs(function () {
  333. expect(view.insertEmoticon).toHaveBeenCalled();
  334. expect(view.$el.find('.toggle-smiley ul').is(':visible')).toBeFalsy();
  335. expect($textarea.val()).toBe(':) <3 ');
  336. });
  337. }));
  338. it("contains a button for starting an encrypted chat session", mock.initConverse(function (converse) {
  339. test_utils.createContacts(converse, 'current');
  340. test_utils.openControlBox();
  341. test_utils.openContactsPanel(converse);
  342. // TODO: More tests can be added here...
  343. var contact_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost';
  344. test_utils.openChatBoxFor(converse, contact_jid);
  345. var view = converse.chatboxviews.get(contact_jid);
  346. var $toolbar = view.$el.find('ul.chat-toolbar');
  347. expect($toolbar.children('li.toggle-otr').length).toBe(1);
  348. // Register spies
  349. spyOn(view, 'toggleOTRMenu').andCallThrough();
  350. view.delegateEvents(); // We need to rebind all events otherwise our spy won't be called
  351. runs(function () {
  352. $toolbar.children('li.toggle-otr').click();
  353. });
  354. waits(250);
  355. runs(function () {
  356. expect(view.toggleOTRMenu).toHaveBeenCalled();
  357. var $menu = view.$el.find('.toggle-otr ul');
  358. expect($menu.is(':visible')).toBeTruthy();
  359. expect($menu.children('li').length).toBe(2);
  360. });
  361. }));
  362. it("can contain a button for starting a call", mock.initConverse(function (converse) {
  363. test_utils.createContacts(converse, 'current');
  364. test_utils.openControlBox();
  365. test_utils.openContactsPanel(converse);
  366. var view, callButton, $toolbar;
  367. var contact_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost';
  368. spyOn(converse, 'emit');
  369. // First check that the button doesn't show if it's not enabled
  370. // via "visible_toolbar_buttons"
  371. converse.visible_toolbar_buttons.call = false;
  372. test_utils.openChatBoxFor(converse, contact_jid);
  373. view = converse.chatboxviews.get(contact_jid);
  374. $toolbar = view.$el.find('ul.chat-toolbar');
  375. callButton = $toolbar.find('.toggle-call');
  376. expect(callButton.length).toBe(0);
  377. view.close();
  378. // Now check that it's shown if enabled and that it emits
  379. // callButtonClicked
  380. converse.visible_toolbar_buttons.call = true; // enable the button
  381. test_utils.openChatBoxFor(converse, contact_jid);
  382. view = converse.chatboxviews.get(contact_jid);
  383. $toolbar = view.$el.find('ul.chat-toolbar');
  384. callButton = $toolbar.find('.toggle-call');
  385. expect(callButton.length).toBe(1);
  386. callButton.click();
  387. expect(converse.emit).toHaveBeenCalledWith('callButtonClicked', jasmine.any(Object));
  388. }));
  389. it("can contain a button for clearing messages", mock.initConverse(function (converse) {
  390. test_utils.createContacts(converse, 'current');
  391. test_utils.openControlBox();
  392. test_utils.openContactsPanel(converse);
  393. var view, clearButton, $toolbar;
  394. var contact_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost';
  395. // First check that the button doesn't show if it's not enabled
  396. // via "visible_toolbar_buttons"
  397. converse.visible_toolbar_buttons.clear = false;
  398. test_utils.openChatBoxFor(converse, contact_jid);
  399. view = converse.chatboxviews.get(contact_jid);
  400. view = converse.chatboxviews.get(contact_jid);
  401. $toolbar = view.$el.find('ul.chat-toolbar');
  402. clearButton = $toolbar.find('.toggle-clear');
  403. expect(clearButton.length).toBe(0);
  404. view.close();
  405. // Now check that it's shown if enabled and that it calls
  406. // clearMessages
  407. converse.visible_toolbar_buttons.clear = true; // enable the button
  408. test_utils.openChatBoxFor(converse, contact_jid);
  409. view = converse.chatboxviews.get(contact_jid);
  410. $toolbar = view.$el.find('ul.chat-toolbar');
  411. clearButton = $toolbar.find('.toggle-clear');
  412. expect(clearButton.length).toBe(1);
  413. spyOn(view, 'clearMessages');
  414. view.delegateEvents(); // We need to rebind all events otherwise our spy won't be called
  415. clearButton.click();
  416. expect(view.clearMessages).toHaveBeenCalled();
  417. }));
  418. });
  419. describe("A Chat Message", function () {
  420. afterEach(function () {
  421. converse_api.user.logout();
  422. converse_api.listen.not();
  423. test_utils.clearBrowserStorage();
  424. });
  425. describe("when received from someone else", function () {
  426. it("can be received which will open a chatbox and be displayed inside it", mock.initConverse(function (converse) {
  427. test_utils.createContacts(converse, 'current');
  428. test_utils.openControlBox();
  429. test_utils.openContactsPanel(converse);
  430. spyOn(converse, 'emit');
  431. var message = 'This is a received message';
  432. var sender_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  433. var msg = $msg({
  434. from: sender_jid,
  435. to: converse.connection.jid,
  436. type: 'chat',
  437. id: (new Date()).getTime()
  438. }).c('body').t(message).up()
  439. .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree();
  440. // We don't already have an open chatbox for this user
  441. expect(converse.chatboxes.get(sender_jid)).not.toBeDefined();
  442. runs(function () {
  443. // onMessage is a handler for received XMPP messages
  444. converse.chatboxes.onMessage(msg);
  445. expect(converse.emit).toHaveBeenCalledWith('message', msg);
  446. });
  447. waits(50);
  448. runs(function () {
  449. // Check that the chatbox and its view now exist
  450. var chatbox = converse.chatboxes.get(sender_jid);
  451. var chatboxview = converse.chatboxviews.get(sender_jid);
  452. expect(chatbox).toBeDefined();
  453. expect(chatboxview).toBeDefined();
  454. // Check that the message was received and check the message parameters
  455. expect(chatbox.messages.length).toEqual(1);
  456. var msg_obj = chatbox.messages.models[0];
  457. expect(msg_obj.get('message')).toEqual(message);
  458. expect(msg_obj.get('fullname')).toEqual(mock.cur_names[0]);
  459. expect(msg_obj.get('sender')).toEqual('them');
  460. expect(msg_obj.get('delayed')).toEqual(false);
  461. // Now check that the message appears inside the chatbox in the DOM
  462. var $chat_content = chatboxview.$el.find('.chat-content');
  463. var msg_txt = $chat_content.find('.chat-message').find('.chat-msg-content').text();
  464. expect(msg_txt).toEqual(message);
  465. var sender_txt = $chat_content.find('span.chat-msg-them').text();
  466. expect(sender_txt.match(/^[0-9][0-9]:[0-9][0-9] /)).toBeTruthy();
  467. });
  468. }));
  469. describe("who is not on the roster", function () {
  470. it("will open a chatbox and be displayed inside it if allow_non_roster_messaging is true", mock.initConverse(function (converse) {
  471. converse.allow_non_roster_messaging = false;
  472. spyOn(converse, 'emit');
  473. var message = 'This is a received message from someone not on the roster';
  474. var sender_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  475. var msg = $msg({
  476. from: sender_jid,
  477. to: converse.connection.jid,
  478. type: 'chat',
  479. id: (new Date()).getTime()
  480. }).c('body').t(message).up()
  481. .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree();
  482. // We don't already have an open chatbox for this user
  483. expect(converse.chatboxes.get(sender_jid)).not.toBeDefined();
  484. runs(function () {
  485. // onMessage is a handler for received XMPP messages
  486. converse.chatboxes.onMessage(msg);
  487. expect(converse.emit).toHaveBeenCalledWith('message', msg);
  488. });
  489. waits(50);
  490. runs(function () {
  491. var chatbox = converse.chatboxes.get(sender_jid);
  492. expect(chatbox).not.toBeDefined();
  493. // onMessage is a handler for received XMPP messages
  494. converse.allow_non_roster_messaging =true;
  495. converse.chatboxes.onMessage(msg);
  496. expect(converse.emit).toHaveBeenCalledWith('message', msg);
  497. });
  498. waits(50);
  499. runs(function () {
  500. // Check that the chatbox and its view now exist
  501. var chatbox = converse.chatboxes.get(sender_jid);
  502. var chatboxview = converse.chatboxviews.get(sender_jid);
  503. expect(chatbox).toBeDefined();
  504. expect(chatboxview).toBeDefined();
  505. // Check that the message was received and check the message parameters
  506. expect(chatbox.messages.length).toEqual(1);
  507. var msg_obj = chatbox.messages.models[0];
  508. expect(msg_obj.get('message')).toEqual(message);
  509. expect(msg_obj.get('fullname')).toEqual(sender_jid);
  510. expect(msg_obj.get('sender')).toEqual('them');
  511. expect(msg_obj.get('delayed')).toEqual(false);
  512. // Now check that the message appears inside the chatbox in the DOM
  513. var $chat_content = chatboxview.$el.find('.chat-content');
  514. var msg_txt = $chat_content.find('.chat-message').find('.chat-msg-content').text();
  515. expect(msg_txt).toEqual(message);
  516. var sender_txt = $chat_content.find('span.chat-msg-them').text();
  517. expect(sender_txt.match(/^[0-9][0-9]:[0-9][0-9] /)).toBeTruthy();
  518. });
  519. }));
  520. });
  521. describe("and for which then an error message is received from the server", function () {
  522. afterEach(function () {
  523. converse_api.user.logout();
  524. converse_api.listen.not();
  525. test_utils.clearBrowserStorage();
  526. });
  527. it("will have the error message displayed after itself", mock.initConverse(function (converse) {
  528. test_utils.createContacts(converse, 'current');
  529. test_utils.openControlBox();
  530. test_utils.openContactsPanel(converse);
  531. // TODO: what could still be done for error
  532. // messages... if the <error> element has type
  533. // "cancel", then we know the messages wasn't sent,
  534. // and can give the user a nicer indication of
  535. // that.
  536. /* <message from="scotty@enterprise.com/converse.js-84843526"
  537. * to="kirk@enterprise.com.com"
  538. * type="chat"
  539. * id="82bc02ce-9651-4336-baf0-fa04762ed8d2"
  540. * xmlns="jabber:client">
  541. * <body>yo</body>
  542. * <active xmlns="http://jabber.org/protocol/chatstates"/>
  543. * </message>
  544. */
  545. var sender_jid = mock.cur_names[5].replace(/ /g,'.').toLowerCase() + '@localhost';
  546. var fullname = converse.xmppstatus.get('fullname');
  547. fullname = _.isEmpty(fullname)? converse.bare_jid: fullname;
  548. converse_api.chats.open(sender_jid);
  549. var msg_text = 'This message will not be sent, due to an error';
  550. var view = converse.chatboxviews.get(sender_jid);
  551. var message = view.model.messages.create({
  552. 'msgid': '82bc02ce-9651-4336-baf0-fa04762ed8d2',
  553. 'fullname': fullname,
  554. 'sender': 'me',
  555. 'time': moment().format(),
  556. 'message': msg_text
  557. });
  558. view.sendMessage(message);
  559. var $chat_content = view.$el.find('.chat-content');
  560. var msg_txt = $chat_content.find('.chat-message:last').find('.chat-msg-content').text();
  561. expect(msg_txt).toEqual(msg_text);
  562. // We send another message, for which an error will
  563. // not be received, to test that errors appear
  564. // after the relevant message.
  565. msg_text = 'This message will be sent, and not receive an error';
  566. message = view.model.messages.create({
  567. 'msgid': '6fcdeee3-000f-4ce8-a17e-9ce28f0ae104',
  568. 'fullname': fullname,
  569. 'sender': 'me',
  570. 'time': moment().format(),
  571. 'message': msg_text
  572. });
  573. view.sendMessage(message);
  574. msg_txt = $chat_content.find('.chat-message:last').find('.chat-msg-content').text();
  575. expect(msg_txt).toEqual(msg_text);
  576. /* <message xmlns="jabber:client"
  577. * to="scotty@enterprise.com/converse.js-84843526"
  578. * type="error"
  579. * id="82bc02ce-9651-4336-baf0-fa04762ed8d2"
  580. * from="kirk@enterprise.com.com">
  581. * <error type="cancel">
  582. * <remote-server-not-found xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"/>
  583. * <text xmlns="urn:ietf:params:xml:ns:xmpp-stanzas">Server-to-server connection failed: Connecting failed: connection timeout</text>
  584. * </error>
  585. * </message>
  586. */
  587. var error_txt = 'Server-to-server connection failed: Connecting failed: connection timeout';
  588. var stanza = $msg({
  589. 'to': converse.connection.jid,
  590. 'type':'error',
  591. 'id':'82bc02ce-9651-4336-baf0-fa04762ed8d2',
  592. 'from': sender_jid
  593. })
  594. .c('error', {'type': 'cancel'})
  595. .c('remote-server-not-found', { 'xmlns': "urn:ietf:params:xml:ns:xmpp-stanzas" }).up()
  596. .c('text', { 'xmlns': "urn:ietf:params:xml:ns:xmpp-stanzas" })
  597. .t('Server-to-server connection failed: Connecting failed: connection timeout');
  598. converse.connection._dataRecv(test_utils.createRequest(stanza));
  599. expect($chat_content.find('.chat-error').text()).toEqual(error_txt);
  600. /* Incoming error messages that are not tied to a
  601. * certain show message (via the msgid attribute),
  602. * are not shown at all. The reason for this is
  603. * that we may get error messages for chat state
  604. * notifications as well.
  605. */
  606. stanza = $msg({
  607. 'to': converse.connection.jid,
  608. 'type':'error',
  609. 'id':'some-other-unused-id',
  610. 'from': sender_jid
  611. })
  612. .c('error', {'type': 'cancel'})
  613. .c('remote-server-not-found', { 'xmlns': "urn:ietf:params:xml:ns:xmpp-stanzas" }).up()
  614. .c('text', { 'xmlns': "urn:ietf:params:xml:ns:xmpp-stanzas" })
  615. .t('Server-to-server connection failed: Connecting failed: connection timeout');
  616. converse.connection._dataRecv(test_utils.createRequest(stanza));
  617. expect($chat_content.find('.chat-error').length).toEqual(1);
  618. }));
  619. });
  620. it("will cause the chat area to be scrolled down only if it was at the bottom already", mock.initConverse(function (converse) {
  621. test_utils.createContacts(converse, 'current');
  622. test_utils.openControlBox();
  623. test_utils.openContactsPanel(converse);
  624. var message = 'This message is received while the chat area is scrolled up';
  625. var sender_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  626. test_utils.openChatBoxFor(converse, sender_jid);
  627. var chatboxview = converse.chatboxviews.get(sender_jid);
  628. spyOn(chatboxview, 'scrollDown').andCallThrough();
  629. runs(function () {
  630. /* Create enough messages so that there's a
  631. * scrollbar.
  632. */
  633. for (var i=0; i<20; i++) {
  634. converse.chatboxes.onMessage($msg({
  635. from: sender_jid,
  636. to: converse.connection.jid,
  637. type: 'chat',
  638. id: (new Date()).getTime()
  639. }).c('body').t('Message: '+i).up()
  640. .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree());
  641. }
  642. });
  643. waits(50);
  644. runs(function () {
  645. chatboxview.$content.scrollTop(0);
  646. });
  647. waits(250);
  648. runs(function () {
  649. converse.chatboxes.onMessage($msg({
  650. from: sender_jid,
  651. to: converse.connection.jid,
  652. type: 'chat',
  653. id: (new Date()).getTime()
  654. }).c('body').t(message).up()
  655. .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree());
  656. });
  657. waits(150);
  658. runs(function () {
  659. // Now check that the message appears inside the chatbox in the DOM
  660. var $chat_content = chatboxview.$el.find('.chat-content');
  661. var msg_txt = $chat_content.find('.chat-message:last').find('.chat-msg-content').text();
  662. expect(msg_txt).toEqual(message);
  663. expect(chatboxview.model.get('scrolled')).toBeTruthy();
  664. expect(chatboxview.$content.scrollTop()).toBe(0);
  665. expect(chatboxview.$('.new-msgs-indicator').is(':visible')).toBeTruthy();
  666. // Scroll down again
  667. chatboxview.$content.scrollTop(chatboxview.$content[0].scrollHeight);
  668. });
  669. waits(250);
  670. runs(function () {
  671. expect(chatboxview.$('.new-msgs-indicator').is(':visible')).toBeFalsy();
  672. });
  673. }));
  674. it("is ignored if it's intended for a different resource and filter_by_resource is set to true", mock.initConverse(function (converse) {
  675. test_utils.createContacts(converse, 'current');
  676. test_utils.openControlBox();
  677. test_utils.openContactsPanel(converse);
  678. // Send a message from a different resource
  679. var message, sender_jid, msg;
  680. spyOn(converse, 'log');
  681. spyOn(converse.chatboxes, 'getChatBox').andCallThrough();
  682. runs(function () {
  683. converse.filter_by_resource = true;
  684. sender_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  685. msg = $msg({
  686. from: sender_jid,
  687. to: converse.bare_jid+"/some-other-resource",
  688. type: 'chat',
  689. id: (new Date()).getTime()
  690. }).c('body').t("This message will not be shown").up()
  691. .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree();
  692. converse.chatboxes.onMessage(msg);
  693. });
  694. waits(50);
  695. runs(function () {
  696. expect(converse.log).toHaveBeenCalledWith(
  697. "onMessage: Ignoring incoming message intended for a different resource: dummy@localhost/some-other-resource", "info");
  698. expect(converse.chatboxes.getChatBox).not.toHaveBeenCalled();
  699. converse.filter_by_resource = false;
  700. });
  701. waits(50);
  702. runs(function () {
  703. message = "This message sent to a different resource will be shown";
  704. msg = $msg({
  705. from: sender_jid,
  706. to: converse.bare_jid+"/some-other-resource",
  707. type: 'chat',
  708. id: '134234623462346'
  709. }).c('body').t(message).up()
  710. .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree();
  711. converse.chatboxes.onMessage(msg);
  712. });
  713. waits(50);
  714. runs(function () {
  715. expect(converse.chatboxes.getChatBox).toHaveBeenCalled();
  716. var chatboxview = converse.chatboxviews.get(sender_jid);
  717. var $chat_content = chatboxview.$el.find('.chat-content:last');
  718. var msg_txt = $chat_content.find('.chat-message').find('.chat-msg-content').text();
  719. expect(msg_txt).toEqual(message);
  720. });
  721. }));
  722. });
  723. it("is ignored if it's a malformed headline message", mock.initConverse(function (converse) {
  724. test_utils.createContacts(converse, 'current');
  725. test_utils.openControlBox();
  726. test_utils.openContactsPanel(converse);
  727. /* Ideally we wouldn't have to filter out headline
  728. * messages, but Prosody gives them the wrong 'type' :(
  729. */
  730. sinon.spy(converse, 'log');
  731. sinon.spy(converse.chatboxes, 'getChatBox');
  732. sinon.spy(utils, 'isHeadlineMessage');
  733. var msg = $msg({
  734. from: 'localhost',
  735. to: converse.bare_jid,
  736. type: 'chat',
  737. id: (new Date()).getTime()
  738. }).c('body').t("This headline message will not be shown").tree();
  739. converse.chatboxes.onMessage(msg);
  740. expect(converse.log.calledWith(
  741. "onMessage: Ignoring incoming headline message sent with type 'chat' from JID: localhost",
  742. "info"
  743. )).toBeTruthy();
  744. expect(utils.isHeadlineMessage.called).toBeTruthy();
  745. expect(utils.isHeadlineMessage.returned(true)).toBeTruthy();
  746. expect(converse.chatboxes.getChatBox.called).toBeFalsy();
  747. // Remove sinon spies
  748. converse.log.restore();
  749. converse.chatboxes.getChatBox.restore();
  750. utils.isHeadlineMessage.restore();
  751. }));
  752. it("can be a carbon message, as defined in XEP-0280", mock.initConverse(function (converse) {
  753. test_utils.createContacts(converse, 'current');
  754. test_utils.openControlBox();
  755. test_utils.openContactsPanel(converse);
  756. // Send a message from a different resource
  757. spyOn(converse, 'log');
  758. var msgtext = 'This is a carbon message';
  759. var sender_jid = mock.cur_names[1].replace(/ /g,'.').toLowerCase() + '@localhost';
  760. var msg = $msg({
  761. 'from': sender_jid,
  762. 'id': (new Date()).getTime(),
  763. 'to': converse.connection.jid,
  764. 'type': 'chat',
  765. 'xmlns': 'jabber:client'
  766. }).c('received', {'xmlns': 'urn:xmpp:carbons:2'})
  767. .c('forwarded', {'xmlns': 'urn:xmpp:forward:0'})
  768. .c('message', {
  769. 'xmlns': 'jabber:client',
  770. 'from': sender_jid,
  771. 'to': converse.bare_jid+'/another-resource',
  772. 'type': 'chat'
  773. }).c('body').t(msgtext).tree();
  774. converse.chatboxes.onMessage(msg);
  775. // Check that the chatbox and its view now exist
  776. var chatbox = converse.chatboxes.get(sender_jid);
  777. var chatboxview = converse.chatboxviews.get(sender_jid);
  778. expect(chatbox).toBeDefined();
  779. expect(chatboxview).toBeDefined();
  780. // Check that the message was received and check the message parameters
  781. expect(chatbox.messages.length).toEqual(1);
  782. var msg_obj = chatbox.messages.models[0];
  783. expect(msg_obj.get('message')).toEqual(msgtext);
  784. expect(msg_obj.get('fullname')).toEqual(mock.cur_names[1]);
  785. expect(msg_obj.get('sender')).toEqual('them');
  786. expect(msg_obj.get('delayed')).toEqual(false);
  787. // Now check that the message appears inside the chatbox in the DOM
  788. var $chat_content = chatboxview.$el.find('.chat-content');
  789. var msg_txt = $chat_content.find('.chat-message').find('.chat-msg-content').text();
  790. expect(msg_txt).toEqual(msgtext);
  791. var sender_txt = $chat_content.find('span.chat-msg-them').text();
  792. expect(sender_txt.match(/^[0-9][0-9]:[0-9][0-9] /)).toBeTruthy();
  793. }));
  794. it("can be a carbon message that this user sent from a different client, as defined in XEP-0280", mock.initConverse(function (converse) {
  795. test_utils.createContacts(converse, 'current');
  796. test_utils.openControlBox();
  797. test_utils.openContactsPanel(converse);
  798. // Send a message from a different resource
  799. spyOn(converse, 'log');
  800. var msgtext = 'This is a sent carbon message';
  801. var recipient_jid = mock.cur_names[5].replace(/ /g,'.').toLowerCase() + '@localhost';
  802. var msg = $msg({
  803. 'from': converse.bare_jid,
  804. 'id': (new Date()).getTime(),
  805. 'to': converse.connection.jid,
  806. 'type': 'chat',
  807. 'xmlns': 'jabber:client'
  808. }).c('sent', {'xmlns': 'urn:xmpp:carbons:2'})
  809. .c('forwarded', {'xmlns': 'urn:xmpp:forward:0'})
  810. .c('message', {
  811. 'xmlns': 'jabber:client',
  812. 'from': converse.bare_jid+'/another-resource',
  813. 'to': recipient_jid,
  814. 'type': 'chat'
  815. }).c('body').t(msgtext).tree();
  816. converse.chatboxes.onMessage(msg);
  817. // Check that the chatbox and its view now exist
  818. var chatbox = converse.chatboxes.get(recipient_jid);
  819. var chatboxview = converse.chatboxviews.get(recipient_jid);
  820. expect(chatbox).toBeDefined();
  821. expect(chatboxview).toBeDefined();
  822. // Check that the message was received and check the message parameters
  823. expect(chatbox.messages.length).toEqual(1);
  824. var msg_obj = chatbox.messages.models[0];
  825. expect(msg_obj.get('message')).toEqual(msgtext);
  826. expect(msg_obj.get('fullname')).toEqual(mock.cur_names[5]);
  827. expect(msg_obj.get('sender')).toEqual('me');
  828. expect(msg_obj.get('delayed')).toEqual(false);
  829. // Now check that the message appears inside the chatbox in the DOM
  830. var $chat_content = chatboxview.$el.find('.chat-content');
  831. var msg_txt = $chat_content.find('.chat-message').find('.chat-msg-content').text();
  832. expect(msg_txt).toEqual(msgtext);
  833. }));
  834. it("will be discarded if it's a malicious message meant to look like a carbon copy", mock.initConverse(function (converse) {
  835. test_utils.createContacts(converse, 'current');
  836. test_utils.openControlBox();
  837. test_utils.openContactsPanel(converse);
  838. /* <message from="mallory@evil.example" to="b@xmpp.example">
  839. * <received xmlns='urn:xmpp:carbons:2'>
  840. * <forwarded xmlns='urn:xmpp:forward:0'>
  841. * <message from="alice@xmpp.example" to="bob@xmpp.example/client1">
  842. * <body>Please come to Creepy Valley tonight, alone!</body>
  843. * </message>
  844. * </forwarded>
  845. * </received>
  846. * </message>
  847. */
  848. spyOn(converse, 'log');
  849. var msgtext = 'Please come to Creepy Valley tonight, alone!';
  850. var sender_jid = mock.cur_names[1].replace(/ /g,'.').toLowerCase() + '@localhost';
  851. var impersonated_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost';
  852. var msg = $msg({
  853. 'from': sender_jid,
  854. 'id': (new Date()).getTime(),
  855. 'to': converse.connection.jid,
  856. 'type': 'chat',
  857. 'xmlns': 'jabber:client'
  858. }).c('received', {'xmlns': 'urn:xmpp:carbons:2'})
  859. .c('forwarded', {'xmlns': 'urn:xmpp:forward:0'})
  860. .c('message', {
  861. 'xmlns': 'jabber:client',
  862. 'from': impersonated_jid,
  863. 'to': converse.connection.jid,
  864. 'type': 'chat'
  865. }).c('body').t(msgtext).tree();
  866. converse.chatboxes.onMessage(msg);
  867. // Check that chatbox for impersonated user is not created.
  868. var chatbox = converse.chatboxes.get(impersonated_jid);
  869. expect(chatbox).not.toBeDefined();
  870. // Check that the chatbox for the malicous user is not created
  871. chatbox = converse.chatboxes.get(sender_jid);
  872. expect(chatbox).not.toBeDefined();
  873. }));
  874. it("received for a minimized chat box will increment a counter on its header", mock.initConverse(function (converse) {
  875. test_utils.createContacts(converse, 'current');
  876. test_utils.openControlBox();
  877. test_utils.openContactsPanel(converse);
  878. var contact_name = mock.cur_names[0];
  879. var contact_jid = contact_name.replace(/ /g,'.').toLowerCase() + '@localhost';
  880. runs(function () {
  881. spyOn(converse, 'emit').andCallThrough();
  882. test_utils.openChatBoxFor(converse, contact_jid);
  883. var chatview = converse.chatboxviews.get(contact_jid);
  884. expect(chatview.$el.is(':visible')).toBeTruthy();
  885. expect(chatview.model.get('minimized')).toBeFalsy();
  886. chatview.$el.find('.toggle-chatbox-button').click();
  887. expect(chatview.model.get('minimized')).toBeTruthy();
  888. var message = 'This message is sent to a minimized chatbox';
  889. var sender_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  890. var msg = $msg({
  891. from: sender_jid,
  892. to: converse.connection.jid,
  893. type: 'chat',
  894. id: (new Date()).getTime()
  895. }).c('body').t(message).up()
  896. .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree();
  897. converse.chatboxes.onMessage(msg);
  898. expect(converse.emit).toHaveBeenCalledWith('message', msg);
  899. var trimmed_chatboxes = converse.minimized_chats;
  900. var trimmedview = trimmed_chatboxes.get(contact_jid);
  901. var $count = trimmedview.$el.find('.chat-head-message-count');
  902. expect(chatview.$el.is(':visible')).toBeFalsy();
  903. expect(trimmedview.model.get('minimized')).toBeTruthy();
  904. expect($count.is(':visible')).toBeTruthy();
  905. expect($count.html()).toBe('1');
  906. converse.chatboxes.onMessage(
  907. $msg({
  908. from: mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost',
  909. to: converse.connection.jid,
  910. type: 'chat',
  911. id: (new Date()).getTime()
  912. }).c('body').t('This message is also sent to a minimized chatbox').up()
  913. .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree()
  914. );
  915. expect(chatview.$el.is(':visible')).toBeFalsy();
  916. expect(trimmedview.model.get('minimized')).toBeTruthy();
  917. $count = trimmedview.$el.find('.chat-head-message-count');
  918. expect($count.is(':visible')).toBeTruthy();
  919. expect($count.html()).toBe('2');
  920. trimmedview.$el.find('.restore-chat').click();
  921. expect(trimmed_chatboxes.keys().length).toBe(0);
  922. });
  923. }));
  924. it("will indicate when it has a time difference of more than a day between it and its predecessor", mock.initConverse(function (converse) {
  925. test_utils.createContacts(converse, 'current');
  926. test_utils.openControlBox();
  927. test_utils.openContactsPanel(converse);
  928. spyOn(converse, 'emit');
  929. var contact_name = mock.cur_names[1];
  930. var contact_jid = contact_name.replace(/ /g,'.').toLowerCase() + '@localhost';
  931. test_utils.openChatBoxFor(converse, contact_jid);
  932. test_utils.clearChatBoxMessages(converse, contact_jid);
  933. var one_day_ago = moment();
  934. one_day_ago.subtract('days', 1);
  935. var message = 'This is a day old message';
  936. var chatbox = converse.chatboxes.get(contact_jid);
  937. var chatboxview = converse.chatboxviews.get(contact_jid);
  938. var $chat_content = chatboxview.$el.find('.chat-content');
  939. var msg_obj;
  940. var msg_txt;
  941. var sender_txt;
  942. var msg = $msg({
  943. from: contact_jid,
  944. to: converse.connection.jid,
  945. type: 'chat',
  946. id: one_day_ago.unix()
  947. }).c('body').t(message).up()
  948. .c('delay', { xmlns:'urn:xmpp:delay', from: 'localhost', stamp: one_day_ago.format() })
  949. .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree();
  950. converse.chatboxes.onMessage(msg);
  951. expect(converse.emit).toHaveBeenCalledWith('message', msg);
  952. expect(chatbox.messages.length).toEqual(1);
  953. msg_obj = chatbox.messages.models[0];
  954. expect(msg_obj.get('message')).toEqual(message);
  955. expect(msg_obj.get('fullname')).toEqual(contact_name);
  956. expect(msg_obj.get('sender')).toEqual('them');
  957. expect(msg_obj.get('delayed')).toEqual(true);
  958. msg_txt = $chat_content.find('.chat-message').find('.chat-msg-content').text();
  959. expect(msg_txt).toEqual(message);
  960. sender_txt = $chat_content.find('span.chat-msg-them').text();
  961. expect(sender_txt.match(/^[0-9][0-9]:[0-9][0-9] /)).toBeTruthy();
  962. var $time = $chat_content.find('time');
  963. expect($time.length).toEqual(1);
  964. expect($time.attr('class')).toEqual('chat-info chat-date');
  965. expect($time.data('isodate')).toEqual(moment(one_day_ago.startOf('day')).format());
  966. expect($time.text()).toEqual(moment(one_day_ago.startOf('day')).format("dddd MMM Do YYYY"));
  967. message = 'This is a current message';
  968. msg = $msg({
  969. from: contact_jid,
  970. to: converse.connection.jid,
  971. type: 'chat',
  972. id: new Date().getTime()
  973. }).c('body').t(message).up()
  974. .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree();
  975. converse.chatboxes.onMessage(msg);
  976. expect(converse.emit).toHaveBeenCalledWith('message', msg);
  977. // Check that there is a <time> element, with the required
  978. // props.
  979. $time = $chat_content.find('time');
  980. expect($time.length).toEqual(2); // There are now two time elements
  981. $time = $chat_content.find('time:last'); // We check the last one
  982. var message_date = new Date();
  983. expect($time.attr('class')).toEqual('chat-info chat-date');
  984. expect($time.data('isodate')).toEqual(moment(message_date).startOf('day').format());
  985. expect($time.text()).toEqual(moment(message_date).startOf('day').format("dddd MMM Do YYYY"));
  986. // Normal checks for the 2nd message
  987. expect(chatbox.messages.length).toEqual(2);
  988. msg_obj = chatbox.messages.models[1];
  989. expect(msg_obj.get('message')).toEqual(message);
  990. expect(msg_obj.get('fullname')).toEqual(contact_name);
  991. expect(msg_obj.get('sender')).toEqual('them');
  992. expect(msg_obj.get('delayed')).toEqual(false);
  993. msg_txt = $chat_content.find('.chat-message').last().find('.chat-msg-content').text();
  994. expect(msg_txt).toEqual(message);
  995. sender_txt = $chat_content.find('span.chat-msg-them').last().text();
  996. expect(sender_txt.match(/^[0-9][0-9]:[0-9][0-9] /)).toBeTruthy();
  997. }));
  998. it("can be sent from a chatbox, and will appear inside it", mock.initConverse(function (converse) {
  999. test_utils.createContacts(converse, 'current');
  1000. test_utils.openControlBox();
  1001. test_utils.openContactsPanel(converse);
  1002. spyOn(converse, 'emit');
  1003. var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  1004. test_utils.openChatBoxFor(converse, contact_jid);
  1005. expect(converse.emit).toHaveBeenCalledWith('chatBoxFocused', jasmine.any(Object));
  1006. var view = converse.chatboxviews.get(contact_jid);
  1007. var message = 'This message is sent from this chatbox';
  1008. spyOn(view, 'sendMessage').andCallThrough();
  1009. test_utils.sendMessage(view, message);
  1010. expect(view.sendMessage).toHaveBeenCalled();
  1011. expect(view.model.messages.length, 2);
  1012. expect(converse.emit.mostRecentCall.args, ['messageSend', message]);
  1013. expect(view.$el.find('.chat-content').find('.chat-message').last().find('.chat-msg-content').text()).toEqual(message);
  1014. }));
  1015. it("is sanitized to prevent Javascript injection attacks", mock.initConverse(function (converse) {
  1016. test_utils.createContacts(converse, 'current');
  1017. test_utils.openControlBox();
  1018. test_utils.openContactsPanel(converse);
  1019. var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  1020. test_utils.openChatBoxFor(converse, contact_jid);
  1021. var view = converse.chatboxviews.get(contact_jid);
  1022. var message = '<p>This message contains <em>some</em> <b>markup</b></p>';
  1023. spyOn(view, 'sendMessage').andCallThrough();
  1024. test_utils.sendMessage(view, message);
  1025. expect(view.sendMessage).toHaveBeenCalled();
  1026. var msg = view.$el.find('.chat-content').find('.chat-message').last().find('.chat-msg-content');
  1027. expect(msg.text()).toEqual(message);
  1028. expect(msg.html()).toEqual('&lt;p&gt;This message contains &lt;em&gt;some&lt;/em&gt; &lt;b&gt;markup&lt;/b&gt;&lt;/p&gt;');
  1029. }));
  1030. it("should display emoticons correctly", mock.initConverse(function (converse) {
  1031. test_utils.createContacts(converse, 'current');
  1032. test_utils.openControlBox();
  1033. test_utils.openContactsPanel(converse);
  1034. var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  1035. test_utils.openChatBoxFor(converse, contact_jid);
  1036. var view = converse.chatboxviews.get(contact_jid);
  1037. var messages = [':)', ';)', ':D', ':P', '8)', '>:)', ':S', ':\\', '>:(', ':(', ':O', '(^.^)b', '<3'];
  1038. var emoticons = [
  1039. '<span class="emoticon icon-smiley"></span>', '<span class="emoticon icon-wink"></span>',
  1040. '<span class="emoticon icon-grin"></span>', '<span class="emoticon icon-tongue"></span>',
  1041. '<span class="emoticon icon-cool"></span>', '<span class="emoticon icon-evil"></span>',
  1042. '<span class="emoticon icon-confused"></span>', '<span class="emoticon icon-wondering"></span>',
  1043. '<span class="emoticon icon-angry"></span>', '<span class="emoticon icon-sad"></span>',
  1044. '<span class="emoticon icon-shocked"></span>', '<span class="emoticon icon-thumbs-up"></span>',
  1045. '<span class="emoticon icon-heart"></span>'
  1046. ];
  1047. spyOn(view, 'sendMessage').andCallThrough();
  1048. for (var i = 0; i < messages.length; i++) {
  1049. var message = messages[i];
  1050. test_utils.sendMessage(view, message);
  1051. expect(view.sendMessage).toHaveBeenCalled();
  1052. var msg = view.$el.find('.chat-content').find('.chat-message').last().find('.chat-msg-content');
  1053. expect(msg.html()).toEqual(emoticons[i]);
  1054. }
  1055. }));
  1056. it("can contain hyperlinks, which will be clickable", mock.initConverse(function (converse) {
  1057. test_utils.createContacts(converse, 'current');
  1058. test_utils.openControlBox();
  1059. test_utils.openContactsPanel(converse);
  1060. var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  1061. test_utils.openChatBoxFor(converse, contact_jid);
  1062. var view = converse.chatboxviews.get(contact_jid);
  1063. var message = 'This message contains a hyperlink: www.opkode.com';
  1064. spyOn(view, 'sendMessage').andCallThrough();
  1065. runs(function () {
  1066. test_utils.sendMessage(view, message);
  1067. });
  1068. waits(500);
  1069. runs(function () {
  1070. expect(view.sendMessage).toHaveBeenCalled();
  1071. var msg = view.$el.find('.chat-content').find('.chat-message').last().find('.chat-msg-content');
  1072. expect(msg.text()).toEqual(message);
  1073. expect(msg.html()).toEqual('This message contains a hyperlink: <a target="_blank" rel="noopener" href="http://www.opkode.com">www.opkode.com</a>');
  1074. });
  1075. }));
  1076. it("will have properly escaped URLs", mock.initConverse(function (converse) {
  1077. test_utils.createContacts(converse, 'current');
  1078. test_utils.openControlBox();
  1079. test_utils.openContactsPanel(converse);
  1080. var message, msg;
  1081. var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  1082. test_utils.openChatBoxFor(converse, contact_jid);
  1083. var view = converse.chatboxviews.get(contact_jid);
  1084. spyOn(view, 'sendMessage').andCallThrough();
  1085. runs(function () {
  1086. message = "http://www.opkode.com/'onmouseover='alert(1)'whatever";
  1087. test_utils.sendMessage(view, message);
  1088. });
  1089. waits(50);
  1090. runs(function () {
  1091. expect(view.sendMessage).toHaveBeenCalled();
  1092. msg = view.$el.find('.chat-content').find('.chat-message').last().find('.chat-msg-content');
  1093. expect(msg.text()).toEqual(message);
  1094. expect(msg.html()).toEqual('<a target="_blank" rel="noopener" href="http://www.opkode.com/%27onmouseover=%27alert%281%29%27whatever">http://www.opkode.com/\'onmouseover=\'alert(1)\'whatever</a>');
  1095. message = 'http://www.opkode.com/"onmouseover="alert(1)"whatever';
  1096. test_utils.sendMessage(view, message);
  1097. });
  1098. waits(50);
  1099. runs(function () {
  1100. expect(view.sendMessage).toHaveBeenCalled();
  1101. msg = view.$el.find('.chat-content').find('.chat-message').last().find('.chat-msg-content');
  1102. expect(msg.text()).toEqual(message);
  1103. expect(msg.html()).toEqual('<a target="_blank" rel="noopener" href="http://www.opkode.com/%22onmouseover=%22alert%281%29%22whatever">http://www.opkode.com/"onmouseover="alert(1)"whatever</a>');
  1104. message = "https://en.wikipedia.org/wiki/Ender's_Game";
  1105. test_utils.sendMessage(view, message);
  1106. });
  1107. waits(50);
  1108. runs(function () {
  1109. expect(view.sendMessage).toHaveBeenCalled();
  1110. msg = view.$el.find('.chat-content').find('.chat-message').last().find('.chat-msg-content');
  1111. expect(msg.text()).toEqual(message);
  1112. expect(msg.html()).toEqual('<a target="_blank" rel="noopener" href="https://en.wikipedia.org/wiki/Ender%27s_Game">https://en.wikipedia.org/wiki/Ender\'s_Game</a>');
  1113. message = "https://en.wikipedia.org/wiki/Ender%27s_Game";
  1114. test_utils.sendMessage(view, message);
  1115. });
  1116. waits(50);
  1117. runs(function () {
  1118. expect(view.sendMessage).toHaveBeenCalled();
  1119. msg = view.$el.find('.chat-content').find('.chat-message').last().find('.chat-msg-content');
  1120. expect(msg.text()).toEqual(message);
  1121. expect(msg.html()).toEqual('<a target="_blank" rel="noopener" href="https://en.wikipedia.org/wiki/Ender%27s_Game">https://en.wikipedia.org/wiki/Ender%27s_Game</a>');
  1122. });
  1123. }));
  1124. it("will render images from their URLs", mock.initConverse(function (converse) {
  1125. test_utils.createContacts(converse, 'current');
  1126. test_utils.openControlBox();
  1127. test_utils.openContactsPanel(converse);
  1128. if (/PhantomJS/.test(window.navigator.userAgent)) {
  1129. // Doesn't work when running tests in PhantomJS, since
  1130. // the page is loaded via file:///
  1131. return;
  1132. }
  1133. var message = document.URL.split(window.location.pathname)[0] + "/logo/conversejs.svg";
  1134. var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  1135. test_utils.openChatBoxFor(converse, contact_jid);
  1136. var view = converse.chatboxviews.get(contact_jid);
  1137. spyOn(view, 'sendMessage').andCallThrough();
  1138. runs(function () {
  1139. test_utils.sendMessage(view, message);
  1140. });
  1141. waits(500);
  1142. runs(function () {
  1143. expect(view.sendMessage).toHaveBeenCalled();
  1144. var msg = view.$el.find('.chat-content').find('.chat-message').last().find('.chat-msg-content');
  1145. expect(msg.html()).toEqual('<img src="'+message+'" class="chat-image">');
  1146. });
  1147. }));
  1148. });
  1149. describe("A Chat Status Notification", function () {
  1150. afterEach(function () {
  1151. converse_api.user.logout();
  1152. converse_api.listen.not();
  1153. test_utils.clearBrowserStorage();
  1154. });
  1155. it("does not open automatically if a chat state notification is received", mock.initConverse(function (converse) {
  1156. test_utils.createContacts(converse, 'current');
  1157. test_utils.openControlBox();
  1158. test_utils.openContactsPanel(converse);
  1159. spyOn(converse, 'emit');
  1160. var sender_jid = mock.cur_names[1].replace(/ /g,'.').toLowerCase() + '@localhost';
  1161. // <composing> state
  1162. var msg = $msg({
  1163. from: sender_jid,
  1164. to: converse.connection.jid,
  1165. type: 'chat',
  1166. id: (new Date()).getTime()
  1167. }).c('composing', {'xmlns': Strophe.NS.CHATSTATES}).tree();
  1168. converse.chatboxes.onMessage(msg);
  1169. expect(converse.emit).toHaveBeenCalledWith('message', msg);
  1170. }));
  1171. describe("An active notification", function () {
  1172. afterEach(function () {
  1173. converse_api.user.logout();
  1174. converse_api.listen.not();
  1175. test_utils.clearBrowserStorage();
  1176. });
  1177. it("is sent when the user opens a chat box", mock.initConverse(function (converse) {
  1178. test_utils.createContacts(converse, 'current');
  1179. test_utils.openControlBox();
  1180. test_utils.openContactsPanel(converse);
  1181. waits(300); // ChatBox.show() is debounced for 250ms
  1182. runs(function () {
  1183. spyOn(converse.connection, 'send');
  1184. var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  1185. test_utils.openChatBoxFor(converse, contact_jid);
  1186. var view = converse.chatboxviews.get(contact_jid);
  1187. expect(view.model.get('chat_state')).toBe('active');
  1188. expect(converse.connection.send).toHaveBeenCalled();
  1189. var $stanza = $(converse.connection.send.argsForCall[0][0].tree());
  1190. expect($stanza.attr('to')).toBe(contact_jid);
  1191. expect($stanza.children().length).toBe(3);
  1192. expect($stanza.children().get(0).tagName).toBe('active');
  1193. expect($stanza.children().get(1).tagName).toBe('no-store');
  1194. expect($stanza.children().get(2).tagName).toBe('no-permanent-store');
  1195. });
  1196. }));
  1197. it("is sent when the user maximizes a minimized a chat box", mock.initConverse(function (converse) {
  1198. test_utils.createContacts(converse, 'current');
  1199. test_utils.openControlBox();
  1200. test_utils.openContactsPanel(converse);
  1201. var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  1202. runs(function () {
  1203. test_utils.openChatBoxFor(converse, contact_jid);
  1204. });
  1205. waits(300); // ChatBox.show() is debounced for 250ms
  1206. runs(function () {
  1207. var view = converse.chatboxviews.get(contact_jid);
  1208. view.model.minimize();
  1209. expect(view.model.get('chat_state')).toBe('inactive');
  1210. spyOn(converse.connection, 'send');
  1211. view.model.maximize();
  1212. expect(view.model.get('chat_state')).toBe('active');
  1213. expect(converse.connection.send).toHaveBeenCalled();
  1214. var $stanza = $(converse.connection.send.argsForCall[0][0].tree());
  1215. expect($stanza.attr('to')).toBe(contact_jid);
  1216. expect($stanza.children().length).toBe(3);
  1217. expect($stanza.children().get(0).tagName).toBe('active');
  1218. expect($stanza.children().get(1).tagName).toBe('no-store');
  1219. expect($stanza.children().get(2).tagName).toBe('no-permanent-store');
  1220. });
  1221. }));
  1222. });
  1223. describe("A composing notification", function () {
  1224. afterEach(function () {
  1225. converse_api.user.logout();
  1226. converse_api.listen.not();
  1227. test_utils.clearBrowserStorage();
  1228. });
  1229. it("is sent as soon as the user starts typing a message which is not a command", mock.initConverse(function (converse) {
  1230. test_utils.createContacts(converse, 'current');
  1231. test_utils.openControlBox();
  1232. test_utils.openContactsPanel(converse);
  1233. waits(300); // ChatBox.show() is debounced for 250ms
  1234. runs(function () {
  1235. var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  1236. test_utils.openChatBoxFor(converse, contact_jid);
  1237. var view = converse.chatboxviews.get(contact_jid);
  1238. expect(view.model.get('chat_state')).toBe('active');
  1239. spyOn(converse.connection, 'send');
  1240. view.keyPressed({
  1241. target: view.$el.find('textarea.chat-textarea'),
  1242. keyCode: 1
  1243. });
  1244. expect(view.model.get('chat_state')).toBe('composing');
  1245. expect(converse.connection.send).toHaveBeenCalled();
  1246. var $stanza = $(converse.connection.send.argsForCall[0][0].tree());
  1247. expect($stanza.attr('to')).toBe(contact_jid);
  1248. expect($stanza.children().get(0).tagName).toBe('composing');
  1249. expect($stanza.children().get(1).tagName).toBe('no-store');
  1250. expect($stanza.children().get(2).tagName).toBe('no-permanent-store');
  1251. // The notification is not sent again
  1252. view.keyPressed({
  1253. target: view.$el.find('textarea.chat-textarea'),
  1254. keyCode: 1
  1255. });
  1256. expect(view.model.get('chat_state')).toBe('composing');
  1257. expect(converse.emit.callCount, 1);
  1258. });
  1259. }));
  1260. it("will be shown if received", mock.initConverse(function (converse) {
  1261. test_utils.createContacts(converse, 'current');
  1262. test_utils.openControlBox();
  1263. test_utils.openContactsPanel(converse);
  1264. // See XEP-0085 http://xmpp.org/extensions/xep-0085.html#definitions
  1265. spyOn(converse, 'emit');
  1266. var sender_jid = mock.cur_names[1].replace(/ /g,'.').toLowerCase() + '@localhost';
  1267. // <composing> state
  1268. var msg = $msg({
  1269. from: sender_jid,
  1270. to: converse.connection.jid,
  1271. type: 'chat',
  1272. id: (new Date()).getTime()
  1273. }).c('body').c('composing', {'xmlns': Strophe.NS.CHATSTATES}).tree();
  1274. converse.chatboxes.onMessage(msg);
  1275. expect(converse.emit).toHaveBeenCalledWith('message', msg);
  1276. var chatboxview = converse.chatboxviews.get(sender_jid);
  1277. expect(chatboxview).toBeDefined();
  1278. // Check that the notification appears inside the chatbox in the DOM
  1279. var $events = chatboxview.$el.find('.chat-event');
  1280. expect($events.text()).toEqual(mock.cur_names[1] + ' is typing');
  1281. }));
  1282. });
  1283. describe("A paused notification", function () {
  1284. afterEach(function () {
  1285. converse_api.user.logout();
  1286. converse_api.listen.not();
  1287. test_utils.clearBrowserStorage();
  1288. });
  1289. it("is sent if the user has stopped typing since 30 seconds", mock.initConverse(function (converse) {
  1290. test_utils.createContacts(converse, 'current');
  1291. test_utils.openControlBox();
  1292. test_utils.openContactsPanel(converse);
  1293. var view, contact_jid;
  1294. converse.TIMEOUTS.PAUSED = 200; // Make the timeout shorter so that we can test
  1295. waits(300); // ChatBox.show() is debounced for 250ms
  1296. runs(function () {
  1297. contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  1298. test_utils.openChatBoxFor(converse, contact_jid);
  1299. view = converse.chatboxviews.get(contact_jid);
  1300. spyOn(converse.connection, 'send');
  1301. spyOn(view, 'setChatState').andCallThrough();
  1302. expect(view.model.get('chat_state')).toBe('active');
  1303. view.keyPressed({
  1304. target: view.$el.find('textarea.chat-textarea'),
  1305. keyCode: 1
  1306. });
  1307. expect(view.model.get('chat_state')).toBe('composing');
  1308. expect(converse.connection.send).toHaveBeenCalled();
  1309. var $stanza = $(converse.connection.send.argsForCall[0][0].tree());
  1310. expect($stanza.children().get(0).tagName).toBe('composing');
  1311. });
  1312. waits(250);
  1313. runs(function () {
  1314. expect(view.model.get('chat_state')).toBe('paused');
  1315. expect(converse.connection.send).toHaveBeenCalled();
  1316. var $stanza = $(converse.connection.send.argsForCall[1][0].tree());
  1317. expect($stanza.attr('to')).toBe(contact_jid);
  1318. expect($stanza.children().length).toBe(3);
  1319. expect($stanza.children().get(0).tagName).toBe('paused');
  1320. expect($stanza.children().get(1).tagName).toBe('no-store');
  1321. expect($stanza.children().get(2).tagName).toBe('no-permanent-store');
  1322. // Test #359. A paused notification should not be sent
  1323. // out if the user simply types longer than the
  1324. // timeout.
  1325. view.keyPressed({
  1326. target: view.$el.find('textarea.chat-textarea'),
  1327. keyCode: 1
  1328. });
  1329. expect(view.setChatState).toHaveBeenCalled();
  1330. expect(view.model.get('chat_state')).toBe('composing');
  1331. });
  1332. waits(100);
  1333. runs(function () {
  1334. view.keyPressed({
  1335. target: view.$el.find('textarea.chat-textarea'),
  1336. keyCode: 1
  1337. });
  1338. expect(view.model.get('chat_state')).toBe('composing');
  1339. });
  1340. waits(150);
  1341. runs(function () {
  1342. expect(view.model.get('chat_state')).toBe('composing');
  1343. });
  1344. }));
  1345. it("will be shown if received", mock.initConverse(function (converse) {
  1346. test_utils.createContacts(converse, 'current');
  1347. test_utils.openControlBox();
  1348. test_utils.openContactsPanel(converse);
  1349. // TODO: only show paused state if the previous state was composing
  1350. // See XEP-0085 http://xmpp.org/extensions/xep-0085.html#definitions
  1351. spyOn(converse, 'emit');
  1352. var sender_jid = mock.cur_names[1].replace(/ /g,'.').toLowerCase() + '@localhost';
  1353. // <paused> state
  1354. var msg = $msg({
  1355. from: sender_jid,
  1356. to: converse.connection.jid,
  1357. type: 'chat',
  1358. id: (new Date()).getTime()
  1359. }).c('body').c('paused', {'xmlns': Strophe.NS.CHATSTATES}).tree();
  1360. converse.chatboxes.onMessage(msg);
  1361. expect(converse.emit).toHaveBeenCalledWith('message', msg);
  1362. var chatboxview = converse.chatboxviews.get(sender_jid);
  1363. var $events = chatboxview.$el.find('.chat-event');
  1364. expect($events.text()).toEqual(mock.cur_names[1] + ' has stopped typing');
  1365. }));
  1366. });
  1367. describe("An inactive notifciation", function () {
  1368. afterEach(function () {
  1369. converse_api.user.logout();
  1370. converse_api.listen.not();
  1371. test_utils.clearBrowserStorage();
  1372. });
  1373. it("is sent if the user has stopped typing since 2 minutes", mock.initConverse(function (converse) {
  1374. test_utils.createContacts(converse, 'current');
  1375. test_utils.openControlBox();
  1376. test_utils.openContactsPanel(converse);
  1377. // Make the timeouts shorter so that we can test
  1378. converse.TIMEOUTS.PAUSED = 200;
  1379. converse.TIMEOUTS.INACTIVE = 200;
  1380. var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  1381. test_utils.openChatBoxFor(converse, contact_jid);
  1382. var view = converse.chatboxviews.get(contact_jid);
  1383. runs(function () {
  1384. expect(view.model.get('chat_state')).toBe('active');
  1385. view.keyPressed({
  1386. target: view.$el.find('textarea.chat-textarea'),
  1387. keyCode: 1
  1388. });
  1389. expect(view.model.get('chat_state')).toBe('composing');
  1390. });
  1391. waits(250);
  1392. runs(function () {
  1393. expect(view.model.get('chat_state')).toBe('paused');
  1394. spyOn(converse.connection, 'send');
  1395. });
  1396. waits(250);
  1397. runs(function () {
  1398. expect(view.model.get('chat_state')).toBe('inactive');
  1399. expect(converse.connection.send).toHaveBeenCalled();
  1400. var $stanza = $(converse.connection.send.argsForCall[0][0].tree());
  1401. expect($stanza.attr('to')).toBe(contact_jid);
  1402. expect($stanza.children().length).toBe(3);
  1403. expect($stanza.children().get(0).tagName).toBe('inactive');
  1404. expect($stanza.children().get(1).tagName).toBe('no-store');
  1405. expect($stanza.children().get(2).tagName).toBe('no-permanent-store');
  1406. });
  1407. }));
  1408. it("is sent when the user a minimizes a chat box", mock.initConverse(function (converse) {
  1409. test_utils.createContacts(converse, 'current');
  1410. test_utils.openControlBox();
  1411. test_utils.openContactsPanel(converse);
  1412. var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  1413. test_utils.openChatBoxFor(converse, contact_jid);
  1414. var view = converse.chatboxviews.get(contact_jid);
  1415. spyOn(converse.connection, 'send');
  1416. view.minimize();
  1417. expect(view.model.get('chat_state')).toBe('inactive');
  1418. expect(converse.connection.send).toHaveBeenCalled();
  1419. var $stanza = $(converse.connection.send.argsForCall[0][0].tree());
  1420. expect($stanza.attr('to')).toBe(contact_jid);
  1421. expect($stanza.children().get(0).tagName).toBe('inactive');
  1422. }));
  1423. it("is sent if the user closes a chat box", mock.initConverse(function (converse) {
  1424. test_utils.createContacts(converse, 'current');
  1425. test_utils.openControlBox();
  1426. test_utils.openContactsPanel(converse);
  1427. waits(300); // ChatBox.show() is debounced for 250ms
  1428. runs(function () {
  1429. var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  1430. test_utils.openChatBoxFor(converse, contact_jid);
  1431. var view = converse.chatboxviews.get(contact_jid);
  1432. expect(view.model.get('chat_state')).toBe('active');
  1433. spyOn(converse.connection, 'send');
  1434. view.close();
  1435. expect(view.model.get('chat_state')).toBe('inactive');
  1436. expect(converse.connection.send).toHaveBeenCalled();
  1437. var $stanza = $(converse.connection.send.argsForCall[0][0].tree());
  1438. expect($stanza.attr('to')).toBe(contact_jid);
  1439. expect($stanza.children().length).toBe(3);
  1440. expect($stanza.children().get(0).tagName).toBe('inactive');
  1441. expect($stanza.children().get(1).tagName).toBe('no-store');
  1442. expect($stanza.children().get(2).tagName).toBe('no-permanent-store');
  1443. });
  1444. }));
  1445. it("will clear any other chat status notifications if its received", mock.initConverse(function (converse) {
  1446. test_utils.createContacts(converse, 'current');
  1447. test_utils.openControlBox();
  1448. test_utils.openContactsPanel(converse);
  1449. // See XEP-0085 http://xmpp.org/extensions/xep-0085.html#definitions
  1450. spyOn(converse, 'emit');
  1451. var sender_jid = mock.cur_names[1].replace(/ /g,'.').toLowerCase() + '@localhost';
  1452. test_utils.openChatBoxFor(converse, sender_jid);
  1453. var view = converse.chatboxviews.get(sender_jid);
  1454. expect(view.$el.find('.chat-event').length).toBe(0);
  1455. view.showStatusNotification(sender_jid+' is typing');
  1456. expect(view.$el.find('.chat-event').length).toBe(1);
  1457. var msg = $msg({
  1458. from: sender_jid,
  1459. to: converse.connection.jid,
  1460. type: 'chat',
  1461. id: (new Date()).getTime()
  1462. }).c('body').c('inactive', {'xmlns': Strophe.NS.CHATSTATES}).tree();
  1463. converse.chatboxes.onMessage(msg);
  1464. expect(converse.emit).toHaveBeenCalledWith('message', msg);
  1465. expect(view.$el.find('.chat-event').length).toBe(0);
  1466. }));
  1467. });
  1468. describe("A gone notifciation", function () {
  1469. afterEach(function () {
  1470. converse_api.user.logout();
  1471. converse_api.listen.not();
  1472. test_utils.clearBrowserStorage();
  1473. });
  1474. it("will be shown if received", mock.initConverse(function (converse) {
  1475. test_utils.createContacts(converse, 'current');
  1476. test_utils.openControlBox();
  1477. test_utils.openContactsPanel(converse);
  1478. spyOn(converse, 'emit');
  1479. var sender_jid = mock.cur_names[1].replace(/ /g,'.').toLowerCase() + '@localhost';
  1480. // <paused> state
  1481. var msg = $msg({
  1482. from: sender_jid,
  1483. to: converse.connection.jid,
  1484. type: 'chat',
  1485. id: (new Date()).getTime()
  1486. }).c('body').c('gone', {'xmlns': Strophe.NS.CHATSTATES}).tree();
  1487. converse.chatboxes.onMessage(msg);
  1488. expect(converse.emit).toHaveBeenCalledWith('message', msg);
  1489. var chatboxview = converse.chatboxviews.get(sender_jid);
  1490. var $events = chatboxview.$el.find('.chat-event');
  1491. expect($events.text()).toEqual(mock.cur_names[1] + ' has gone away');
  1492. }));
  1493. });
  1494. });
  1495. });
  1496. describe("Special Messages", function () {
  1497. afterEach(function () {
  1498. converse_api.user.logout();
  1499. converse_api.listen.not();
  1500. test_utils.clearBrowserStorage();
  1501. });
  1502. it("'/clear' can be used to clear messages in a conversation", mock.initConverse(function (converse) {
  1503. test_utils.createContacts(converse, 'current');
  1504. test_utils.openControlBox();
  1505. test_utils.openContactsPanel(converse);
  1506. spyOn(converse, 'emit');
  1507. var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
  1508. test_utils.openChatBoxFor(converse, contact_jid);
  1509. var view = converse.chatboxviews.get(contact_jid);
  1510. var message = 'This message is another sent from this chatbox';
  1511. // Lets make sure there is at least one message already
  1512. // (e.g for when this test is run on its own).
  1513. test_utils.sendMessage(view, message);
  1514. expect(view.model.messages.length > 0).toBeTruthy();
  1515. expect(view.model.messages.browserStorage.records.length > 0).toBeTruthy();
  1516. expect(converse.emit).toHaveBeenCalledWith('messageSend', message);
  1517. message = '/clear';
  1518. spyOn(view, 'onMessageSubmitted').andCallThrough();
  1519. spyOn(view, 'clearMessages').andCallThrough();
  1520. spyOn(window, 'confirm').andCallFake(function () {
  1521. return true;
  1522. });
  1523. test_utils.sendMessage(view, message);
  1524. expect(view.onMessageSubmitted).toHaveBeenCalled();
  1525. expect(view.clearMessages).toHaveBeenCalled();
  1526. expect(window.confirm).toHaveBeenCalled();
  1527. expect(view.model.messages.length, 0); // The messages must be removed from the chatbox
  1528. expect(view.model.messages.browserStorage.records.length, 0); // And also from browserStorage
  1529. expect(converse.emit.callCount, 1);
  1530. expect(converse.emit.mostRecentCall.args, ['messageSend', message]);
  1531. }));
  1532. });
  1533. describe("A Message Counter", function () {
  1534. afterEach(function () {
  1535. converse_api.user.logout();
  1536. converse_api.listen.not();
  1537. test_utils.clearBrowserStorage();
  1538. });
  1539. it("is incremented when the message is received and the window is not focused", mock.initConverse(function (converse) {
  1540. test_utils.createContacts(converse, 'current');
  1541. test_utils.openControlBox();
  1542. test_utils.openContactsPanel(converse);
  1543. spyOn(converse, 'emit');
  1544. expect(converse.msg_counter).toBe(0);
  1545. spyOn(converse, 'incrementMsgCounter').andCallThrough();
  1546. var previous_state = converse.windowState;
  1547. var message = 'This message will increment the message counter';
  1548. var sender_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost',
  1549. msg = $msg({
  1550. from: sender_jid,
  1551. to: converse.connection.jid,
  1552. type: 'chat',
  1553. id: (new Date()).getTime()
  1554. }).c('body').t(message).up()
  1555. .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree();
  1556. converse.windowState = 'hidden';
  1557. converse.chatboxes.onMessage(msg);
  1558. expect(converse.incrementMsgCounter).toHaveBeenCalled();
  1559. expect(converse.msg_counter).toBe(1);
  1560. expect(converse.emit).toHaveBeenCalledWith('message', msg);
  1561. converse.windowSate = previous_state;
  1562. }));
  1563. it("is cleared when the window is focused", mock.initConverse(function (converse) {
  1564. test_utils.createContacts(converse, 'current');
  1565. test_utils.openControlBox();
  1566. test_utils.openContactsPanel(converse);
  1567. converse.windowState = 'hidden';
  1568. spyOn(converse, 'clearMsgCounter').andCallThrough();
  1569. runs(function () {
  1570. converse.saveWindowState(null, 'focus');
  1571. converse.saveWindowState(null, 'blur');
  1572. });
  1573. waits(50);
  1574. runs(function () {
  1575. expect(converse.clearMsgCounter).toHaveBeenCalled();
  1576. });
  1577. }));
  1578. it("is not incremented when the message is received and the window is focused", mock.initConverse(function (converse) {
  1579. test_utils.createContacts(converse, 'current');
  1580. test_utils.openControlBox();
  1581. test_utils.openContactsPanel(converse);
  1582. expect(converse.msg_counter).toBe(0);
  1583. spyOn(converse, 'incrementMsgCounter').andCallThrough();
  1584. converse.saveWindowState(null, 'focus');
  1585. var message = 'This message will not increment the message counter';
  1586. var sender_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost',
  1587. msg = $msg({
  1588. from: sender_jid,
  1589. to: converse.connection.jid,
  1590. type: 'chat',
  1591. id: (new Date()).getTime()
  1592. }).c('body').t(message).up()
  1593. .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree();
  1594. converse.chatboxes.onMessage(msg);
  1595. expect(converse.incrementMsgCounter).not.toHaveBeenCalled();
  1596. expect(converse.msg_counter).toBe(0);
  1597. }));
  1598. });
  1599. });
  1600. }));