123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419 |
- /*global mock, converse */
- const { u, Promise, $msg } = converse.env;
- describe("An incoming chat Message", function () {
- it("can have styling disabled via an \"unstyled\" element",
- mock.initConverse(['rosterContactsFetched', 'chatBoxesFetched'], {},
- async function (done, _converse) {
- const include_nick = false;
- await mock.waitForRoster(_converse, 'current', 2, include_nick);
- await mock.openControlBox(_converse);
- const msg_text = '> _ >';
- const sender_jid = mock.cur_names[1].replace(/ /g,'.').toLowerCase() + '@montague.lit';
- const msg = $msg({
- 'from': sender_jid,
- 'id': u.getUniqueId(),
- 'to': _converse.connection.jid,
- 'type': 'chat',
- 'xmlns': 'jabber:client'
- }).c('body').t(msg_text).up()
- .c('unstyled', {'xmlns': 'urn:xmpp:styling:0'}).tree();
- await _converse.handleMessageStanza(msg);
- const view = _converse.api.chatviews.get(sender_jid);
- await u.waitUntil(() => view.model.messages.length);
- expect(view.model.messages.models[0].get('is_unstyled')).toBe(true);
- setTimeout(() => {
- const msg_el = view.querySelector('converse-chat-message-body');
- expect(msg_el.innerText).toBe(msg_text);
- done();
- }, 500);
- }));
- it("can have styling disabled via the allow_message_styling setting",
- mock.initConverse(['rosterContactsFetched', 'chatBoxesFetched'], {'allow_message_styling': false},
- async function (done, _converse) {
- const include_nick = false;
- await mock.waitForRoster(_converse, 'current', 2, include_nick);
- await mock.openControlBox(_converse);
- const msg_text = '> _ >';
- const sender_jid = mock.cur_names[1].replace(/ /g,'.').toLowerCase() + '@montague.lit';
- const msg = $msg({
- 'from': sender_jid,
- 'id': u.getUniqueId(),
- 'to': _converse.connection.jid,
- 'type': 'chat',
- 'xmlns': 'jabber:client'
- }).c('body').t(msg_text).tree();
- await _converse.handleMessageStanza(msg);
- const view = _converse.api.chatviews.get(sender_jid);
- await u.waitUntil(() => view.model.messages.length);
- expect(view.model.messages.models[0].get('is_unstyled')).toBe(false);
- setTimeout(() => {
- const msg_el = view.querySelector('converse-chat-message-body');
- expect(msg_el.innerText).toBe(msg_text);
- done();
- }, 500);
- }));
- it("can be styled with span XEP-0393 message styling hints",
- mock.initConverse(['rosterContactsFetched', 'chatBoxesFetched'], {},
- async function (done, _converse) {
- let msg_text, msg, msg_el;
- await mock.waitForRoster(_converse, 'current', 1);
- const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
- await mock.openChatBoxFor(_converse, contact_jid);
- const view = _converse.api.chatviews.get(contact_jid);
- msg_text = "This *message _contains_* styling hints! \`Here's *some* code\`";
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = view.querySelector('converse-chat-message-body');
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- 'This <span class="styling-directive">*</span>'+
- '<b>message <span class="styling-directive">_</span><i>contains</i><span class="styling-directive">_</span></b>'+
- '<span class="styling-directive">*</span>'+
- ' styling hints! '+
- '<span class="styling-directive">`</span><code>Here\'s *some* code</code><span class="styling-directive">`</span>'
- );
- msg_text = "Here's a ~strikethrough section~";
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- 'Here\'s a <span class="styling-directive">~</span><del>strikethrough section</del><span class="styling-directive">~</span>');
- // Span directives containing hyperlinks
- msg_text = "~Check out this site: https://conversejs.org~"
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- '<span class="styling-directive">~</span>'+
- '<del>Check out this site: <a target="_blank" rel="noopener" href="https://conversejs.org/">https://conversejs.org</a></del>'+
- '<span class="styling-directive">~</span>');
- // Images inside directives aren't shown inline
- const base_url = 'https://conversejs.org';
- msg_text = `*${base_url}/logo/conversejs-filled.svg*`;
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- '<span class="styling-directive">*</span>'+
- '<b><a target="_blank" rel="noopener" href="https://conversejs.org/logo/conversejs-filled.svg">https://conversejs.org/logo/conversejs-filled.svg</a></b>'+
- '<span class="styling-directive">*</span>');
- // Emojis inside directives
- msg_text = `~ Hello! :poop: ~`;
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- '<span class="styling-directive">~</span><del> Hello! <span title=":poop:">💩</span> </del><span class="styling-directive">~</span>');
- // Span directives don't cross lines
- msg_text = "This *is not a styling hint \n * _But this is_!";
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- 'This *is not a styling hint \n'+
- ' * <span class="styling-directive">_</span><i>But this is</i><span class="styling-directive">_</span>!');
- msg_text = `(There are three blocks in this body marked by parens,)\n (but there is no *formatting)\n (as spans* may not escape blocks.)\n ~strikethrough~`;
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- '(There are three blocks in this body marked by parens,)\n'+
- ' (but there is no *formatting)\n'+
- ' (as spans* may not escape blocks.)\n'+
- ' <span class="styling-directive">~</span><del>strikethrough</del><span class="styling-directive">~</span>');
- // Some edge-case (unspecified) spans
- msg_text = `__ hello world _`;
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- '_<span class="styling-directive">_</span><i> hello world </i><span class="styling-directive">_</span>');
- // Directives which are parts of words aren't matched
- msg_text = `Go to ~https://conversejs.org~now _please_`;
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- 'Go to ~https://conversejs.org~now <span class="styling-directive">_</span><i>please</i><span class="styling-directive">_</span>');
- msg_text = `Go to _https://converse_js.org_ _please_`;
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- 'Go to <span class="styling-directive">_</span>'+
- '<i><a target="_blank" rel="noopener" href="https://converse_js.org/">https://converse_js.org</a></i>'+
- '<span class="styling-directive">_</span> <span class="styling-directive">_</span><i>please</i><span class="styling-directive">_</span>');
- done();
- }));
- it("can be styled with block XEP-0393 message styling hints",
- mock.initConverse(['rosterContactsFetched', 'chatBoxesFetched'], {},
- async function (done, _converse) {
- let msg_text, msg, msg_el;
- await mock.waitForRoster(_converse, 'current', 1);
- const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
- await mock.openChatBoxFor(_converse, contact_jid);
- const view = _converse.api.chatviews.get(contact_jid);
- msg_text = `Here's a code block: \n\`\`\`\nInside the code-block, <code>hello</code> we don't enable *styling hints* like ~these~\n\`\`\``;
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- 'Here\'s a code block: \n'+
- '<div class="styling-directive">```</div><code class="block">Inside the code-block, <code>hello</code> we don\'t enable *styling hints* like ~these~\n'+
- '</code><div class="styling-directive">```</div>'
- );
- msg_text = "```\nignored\n(println \"Hello, world!\")\n```\nThis should show up as monospace, preformatted text ^";
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- '<div class="styling-directive">```</div>'+
- '<code class="block">ignored\n(println "Hello, world!")\n</code>'+
- '<div class="styling-directive">```</div>\n'+
- 'This should show up as monospace, preformatted text ^');
- msg_text = "```ignored\n (println \"Hello, world!\")\n ```\n\n This should not show up as monospace, *preformatted* text ^";
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- '```ignored\n (println "Hello, world!")\n ```\n\n'+
- ' This should not show up as monospace, '+
- '<span class="styling-directive">*</span><b>preformatted</b><span class="styling-directive">*</span> text ^');
- done();
- }));
- it("can be styled with quote XEP-0393 message styling hints",
- mock.initConverse(['rosterContactsFetched', 'chatBoxesFetched'], {},
- async function (done, _converse) {
- let msg_text, msg, msg_el;
- await mock.waitForRoster(_converse, 'current', 1);
- const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
- await mock.openChatBoxFor(_converse, contact_jid);
- const view = _converse.api.chatviews.get(contact_jid);
- msg_text = `> This is quoted text\n>This is also quoted\nThis is not quoted`;
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- '<blockquote>This is quoted text\nThis is also quoted</blockquote>\nThis is not quoted');
- msg_text = `> This is *quoted* text\n>This is \`also _quoted_\`\nThis is not quoted`;
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- '<blockquote>This is <span class="styling-directive">*</span><b>quoted</b><span class="styling-directive">*</span> text\n'+
- 'This is <span class="styling-directive">`</span><code>also _quoted_</code><span class="styling-directive">`</span></blockquote>\n'+
- 'This is not quoted');
- msg_text = `> > This is doubly quoted text`;
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') === "<blockquote><blockquote>This is doubly quoted text</blockquote></blockquote>");
- msg_text = `>> This is doubly quoted text`;
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') === "<blockquote><blockquote>This is doubly quoted text</blockquote></blockquote>");
- msg_text = ">```\n>ignored\n> <span></span> (println \"Hello, world!\")\n>```\n> This should show up as monospace, preformatted text ^";
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- '<blockquote>'+
- '<div class="styling-directive">```</div>'+
- '<code class="block">ignored\n <span></span> (println "Hello, world!")\n'+
- '</code><div class="styling-directive">```</div>\n'+
- ' This should show up as monospace, preformatted text ^'+
- '</blockquote>');
- msg_text = '> ```\n> (println "Hello, world!")\n\nThe entire blockquote is a preformatted text block, but this line is plaintext!';
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- '<blockquote>```\n (println "Hello, world!")</blockquote>\n\n'+
- 'The entire blockquote is a preformatted text block, but this line is plaintext!');
- msg_text = '> Also, icons.js is loaded from /dist, instead of dist.\nhttps://conversejs.org/docs/html/configuration.html#assets-path'
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- '<blockquote>Also, icons.js is loaded from /dist, instead of dist.</blockquote>\n'+
- '<a target="_blank" rel="noopener" href="https://conversejs.org/docs/html/configuration.html#assets-path">https://conversejs.org/docs/html/configuration.html#assets-path</a>');
- msg_text = '> Where is it located?\ngeo:37.786971,-122.399677';
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- '<blockquote>Where is it located?</blockquote>\n'+
- '<a target="_blank" rel="noopener" '+
- 'href="https://www.openstreetmap.org/?mlat=37.786971&mlon=-122.399677#map=18/37.786971/-122.399677">https://www.openstreetmap.org/?mlat=37.786971&mlon=-122.399677#map=18/37.786971/-122.399677</a>');
- msg_text = '> What do you think of it?\n :poop:';
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- '<blockquote>What do you think of it?</blockquote>\n <span title=":poop:">💩</span>');
- msg_text = '> What do you think of it?\n~hello~';
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- '<blockquote>What do you think of it?</blockquote>\n<span class="styling-directive">~</span><del>hello</del><span class="styling-directive">~</span>');
- msg_text = 'hello world > this is not a quote';
- msg = mock.createChatMessage(_converse, contact_jid, msg_text)
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') === 'hello world > this is not a quote');
- msg_text = '> What do you think of it romeo?\n Did you see this romeo?';
- msg = $msg({
- from: contact_jid,
- to: _converse.connection.jid,
- type: 'chat',
- id: (new Date()).getTime()
- }).c('body').t(msg_text).up()
- .c('reference', {
- 'xmlns':'urn:xmpp:reference:0',
- 'begin':'26',
- 'end':'31',
- 'type':'mention',
- 'uri': _converse.bare_jid
- })
- .c('reference', {
- 'xmlns':'urn:xmpp:reference:0',
- 'begin':'51',
- 'end':'56',
- 'type':'mention',
- 'uri': _converse.bare_jid
- }).nodeTree;
- await _converse.handleMessageStanza(msg);
- await new Promise(resolve => view.model.messages.once('rendered', resolve));
- msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- `<blockquote>What do you think of it <span class="mention">romeo</span>?</blockquote>\n Did you see this <span class="mention">romeo</span>?`);
- done();
- }));
- });
- describe("A outgoing groupchat Message", function () {
- it("can be styled with span XEP-0393 message styling hints that contain mentions",
- mock.initConverse(['rosterContactsFetched', 'chatBoxesFetched'], {},
- async function (done, _converse) {
- const muc_jid = 'lounge@montague.lit';
- await mock.openAndEnterChatRoom(_converse, muc_jid, 'romeo');
- const view = _converse.api.chatviews.get(muc_jid);
- const msg_text = "This *message mentions romeo*";
- const msg = $msg({
- from: 'lounge@montague.lit/gibson',
- id: u.getUniqueId(),
- to: 'romeo@montague.lit',
- type: 'groupchat'
- }).c('body').t(msg_text).up()
- .c('reference', {'xmlns':'urn:xmpp:reference:0', 'begin':'23', 'end':'29', 'type':'mention', 'uri':'xmpp:romeo@montague.lit'}).nodeTree;
- await view.model.handleMessageStanza(msg);
- const message = await u.waitUntil(() => view.querySelector('.chat-msg__text'));
- expect(message.classList.length).toEqual(1);
- const msg_el = Array.from(view.querySelectorAll('converse-chat-message-body')).pop();
- expect(msg_el.innerText).toBe(msg_text);
- await u.waitUntil(() => msg_el.innerHTML.replace(/<!---->/g, '') ===
- 'This <span class="styling-directive">*</span><b>message mentions <span class="mention mention--self badge badge-info">romeo</span></b><span class="styling-directive">*</span>');
- done();
- }));
- });
|