/* global mock, converse */ const { Strophe, u, sizzle, $iq } = converse.env; describe("A chat room", function () { it("can be bookmarked", mock.initConverse( ['chatBoxesFetched'], {}, async function (done, _converse) { await mock.waitForRoster(_converse, 'current', 0); await mock.waitUntilDiscoConfirmed( _converse, _converse.bare_jid, [{'category': 'pubsub', 'type': 'pep'}], ['http://jabber.org/protocol/pubsub#publish-options'] ); const { u, $iq } = converse.env; spyOn(_converse.connection, 'getUniqueId').and.callThrough(); const nick = 'JC'; const muc_jid = 'theplay@conference.shakespeare.lit'; await mock.openChatRoom(_converse, 'theplay', 'conference.shakespeare.lit', 'JC'); await mock.getRoomFeatures(_converse, muc_jid, []); await mock.waitForReservedNick(_converse, muc_jid, nick); await mock.receiveOwnMUCPresence(_converse, muc_jid, nick); const view = _converse.chatboxviews.get(muc_jid); await u.waitUntil(() => (view.model.session.get('connection_status') === converse.ROOMSTATUS.ENTERED)); await mock.returnMemberLists(_converse, muc_jid, [], ['member', 'admin', 'owner']); await u.waitUntil(() => view.querySelector('.toggle-bookmark') !== null); const toggle = view.querySelector('.toggle-bookmark'); expect(toggle.title).toBe('Bookmark this groupchat'); toggle.click(); const cancel_button = await u.waitUntil(() => view.querySelector('.button-cancel')); expect(view.model.session.get('view')).toBe('bookmark-form'); cancel_button.click(); await u.waitUntil(() => view.model.session.get('view') === null); expect(u.hasClass('on-button', toggle), false); expect(toggle.title).toBe('Bookmark this groupchat'); toggle.click(); /* Client uploads data: * -------------------- * * * * * * * JC * * * * * * * * http://jabber.org/protocol/pubsub#publish-options * * * true * * * whitelist * * * * * */ expect(view.model.get('bookmarked')).toBeFalsy(); const form = await u.waitUntil(() => view.querySelector('.chatroom-form')); form.querySelector('input[name="name"]').value = 'Play's the Thing'; form.querySelector('input[name="autojoin"]').checked = 'checked'; form.querySelector('input[name="nick"]').value = 'JC'; const IQ_stanzas = _converse.connection.IQ_stanzas; view.querySelector('converse-muc-bookmark-form .btn-primary').click(); const sent_stanza = await u.waitUntil( () => IQ_stanzas.filter(s => sizzle('iq publish[node="storage:bookmarks"]', s).length).pop()); expect(Strophe.serialize(sent_stanza)).toBe( ``+ ``+ ``+ ``+ ``+ ``+ `JC`+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ `http://jabber.org/protocol/pubsub#publish-options`+ ``+ ``+ `true`+ ``+ ``+ `whitelist`+ ``+ ``+ ``+ ``+ `` ); /* Server acknowledges successful storage * * */ const stanza = $iq({ 'to':_converse.connection.jid, 'type':'result', 'id': sent_stanza.getAttribute('id') }); _converse.connection._dataRecv(mock.createRequest(stanza)); await u.waitUntil(() => view.model.get('bookmarked')); expect(view.model.get('bookmarked')).toBeTruthy(); await u.waitUntil(() => view.querySelector('.toggle-bookmark')?.title === 'Unbookmark this groupchat'); expect(u.hasClass('on-button', view.querySelector('.toggle-bookmark')), true); // We ignore this IQ stanza... (unless it's an error stanza), so // nothing to test for here. done(); })); it("will be automatically opened if 'autojoin' is set on the bookmark", mock.initConverse( [], {}, async function (done, _converse) { const { u } = converse.env; await mock.waitForRoster(_converse, 'current', 0); await mock.waitUntilDiscoConfirmed( _converse, _converse.bare_jid, [{'category': 'pubsub', 'type': 'pep'}], ['http://jabber.org/protocol/pubsub#publish-options'] ); await u.waitUntil(() => _converse.bookmarks); let jid = 'lounge@montague.lit'; _converse.bookmarks.create({ 'jid': jid, 'autojoin': false, 'name': 'The Lounge', 'nick': ' Othello' }); expect(_converse.chatboxviews.get(jid) === undefined).toBeTruthy(); jid = 'theplay@conference.shakespeare.lit'; _converse.bookmarks.create({ 'jid': jid, 'autojoin': true, 'name': 'The Play', 'nick': ' Othello' }); await new Promise(resolve => _converse.api.listen.once('chatRoomViewInitialized', resolve)); expect(!!_converse.chatboxviews.get(jid)).toBe(true); // Check that we don't auto-join if muc_respect_autojoin is false _converse.muc_respect_autojoin = false; jid = 'balcony@conference.shakespeare.lit'; _converse.bookmarks.create({ 'jid': jid, 'autojoin': true, 'name': 'Balcony', 'nick': ' Othello' }); expect(_converse.chatboxviews.get(jid) === undefined).toBe(true); done(); })); describe("when bookmarked", function () { it("will use the nickname from the bookmark", mock.initConverse([], {}, async function (done, _converse) { const { u } = converse.env; await mock.waitForRoster(_converse, 'current', 0); await mock.waitUntilBookmarksReturned(_converse); const muc_jid = 'coven@chat.shakespeare.lit'; _converse.bookmarks.create({ 'jid': muc_jid, 'autojoin': false, 'name': 'The Play', 'nick': 'Othello' }); spyOn(_converse.ChatRoom.prototype, 'getAndPersistNickname').and.callThrough(); const room_creation_promise = _converse.api.rooms.open(muc_jid); await mock.getRoomFeatures(_converse, muc_jid); const room = await room_creation_promise; await u.waitUntil(() => room.getAndPersistNickname.calls.count()); expect(room.get('nick')).toBe('Othello'); done(); })); it("displays that it's bookmarked through its bookmark icon", mock.initConverse([], {}, async function (done, _converse) { const { u } = converse.env; await mock.waitForRoster(_converse, 'current', 0); mock.waitUntilDiscoConfirmed( _converse, _converse.bare_jid, [{'category': 'pubsub', 'type': 'pep'}], ['http://jabber.org/protocol/pubsub#publish-options'] ); await _converse.api.rooms.open(`lounge@montague.lit`); const view = _converse.chatboxviews.get('lounge@montague.lit'); expect(view.querySelector('.chatbox-title__text .fa-bookmark')).toBe(null); _converse.bookmarks.create({ 'jid': view.model.get('jid'), 'autojoin': false, 'name': 'The lounge', 'nick': ' some1' }); view.model.set('bookmarked', true); await u.waitUntil(() => view.querySelector('.chatbox-title__text .fa-bookmark') !== null); view.model.set('bookmarked', false); await u.waitUntil(() => view.querySelector('.chatbox-title__text .fa-bookmark') === null); done(); })); it("can be unbookmarked", mock.initConverse([], {}, async function (done, _converse) { const { u, Strophe } = converse.env; await mock.waitForRoster(_converse, 'current', 0); await mock.waitUntilBookmarksReturned(_converse); const muc_jid = 'theplay@conference.shakespeare.lit'; await _converse.api.rooms.open(muc_jid); const view = _converse.chatboxviews.get(muc_jid); await u.waitUntil(() => view.querySelector('.toggle-bookmark')); spyOn(view, 'toggleBookmark').and.callThrough(); spyOn(_converse.bookmarks, 'sendBookmarkStanza').and.callThrough(); _converse.bookmarks.create({ 'jid': view.model.get('jid'), 'autojoin': false, 'name': 'The Play', 'nick': ' Othello' }); expect(_converse.bookmarks.length).toBe(1); await u.waitUntil(() => _converse.chatboxes.length >= 1); expect(view.model.get('bookmarked')).toBeTruthy(); await u.waitUntil(() => view.querySelector('.chatbox-title__text .fa-bookmark') !== null); spyOn(_converse.connection, 'getUniqueId').and.callThrough(); const bookmark_icon = view.querySelector('.toggle-bookmark'); bookmark_icon.click(); expect(view.toggleBookmark).toHaveBeenCalled(); await u.waitUntil(() => view.querySelector('.chatbox-title__text .fa-bookmark') === null); expect(_converse.bookmarks.length).toBe(0); // Check that an IQ stanza is sent out, containing no // conferences to bookmark (since we removed the one and // only bookmark). const sent_stanza = _converse.connection.IQ_stanzas.pop(); expect(Strophe.serialize(sent_stanza)).toBe( ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ `http://jabber.org/protocol/pubsub#publish-options`+ ``+ ``+ `true`+ ``+ ``+ `whitelist`+ ``+ ``+ ``+ ``+ `` ); done(); })); }); describe("and when autojoin is set", function () { it("will be be opened and joined automatically upon login", mock.initConverse( [], {}, async function (done, _converse) { await mock.waitForRoster(_converse, 'current', 0); await mock.waitUntilBookmarksReturned(_converse); spyOn(_converse.api.rooms, 'create').and.callThrough(); const jid = 'theplay@conference.shakespeare.lit'; const model = _converse.bookmarks.create({ 'jid': jid, 'autojoin': false, 'name': 'The Play', 'nick': '' }); expect(_converse.api.rooms.create).not.toHaveBeenCalled(); _converse.bookmarks.remove(model); _converse.bookmarks.create({ 'jid': jid, 'autojoin': true, 'name': 'Hamlet', 'nick': '' }); expect(_converse.api.rooms.create).toHaveBeenCalled(); done(); })); }); }); describe("Bookmarks", function () { it("can be pushed from the XMPP server", mock.initConverse( ['connected', 'chatBoxesFetched'], {}, async function (done, _converse) { const { $msg, u } = converse.env; await mock.waitForRoster(_converse, 'current', 0); await mock.waitUntilBookmarksReturned(_converse); /* The stored data is automatically pushed to all of the user's * connected resources. * * Publisher receives event notification * ------------------------------------- * * * * * * * JC * * * * * * */ let stanza = $msg({ 'from': 'romeo@montague.lit', 'to': _converse.jid, 'type': 'headline', 'id': u.getUniqueId() }).c('event', {'xmlns': 'http://jabber.org/protocol/pubsub#event'}) .c('items', {'node': 'storage:bookmarks'}) .c('item', {'id': 'current'}) .c('storage', {'xmlns': 'storage:bookmarks'}) .c('conference', { 'name': 'The Play's the Thing', 'autojoin': 'true', 'jid':'theplay@conference.shakespeare.lit' }).c('nick').t('JC').up().up() .c('conference', { 'name': 'Another bookmark', 'autojoin': 'false', 'jid':'another@conference.shakespeare.lit' }).c('nick').t('JC'); _converse.connection._dataRecv(mock.createRequest(stanza)); await u.waitUntil(() => _converse.bookmarks.length); expect(_converse.bookmarks.length).toBe(2); expect(_converse.bookmarks.map(b => b.get('name'))).toEqual(['Another bookmark', 'The Play's the Thing']); expect(_converse.chatboxviews.get('theplay@conference.shakespeare.lit')).not.toBeUndefined(); stanza = $msg({ 'from': 'romeo@montague.lit', 'to': _converse.jid, 'type': 'headline', 'id': u.getUniqueId() }).c('event', {'xmlns': 'http://jabber.org/protocol/pubsub#event'}) .c('items', {'node': 'storage:bookmarks'}) .c('item', {'id': 'current'}) .c('storage', {'xmlns': 'storage:bookmarks'}) .c('conference', { 'name': 'The Play's the Thing', 'autojoin': 'true', 'jid':'theplay@conference.shakespeare.lit' }).c('nick').t('JC').up().up() .c('conference', { 'name': 'Second bookmark', 'autojoin': 'false', 'jid':'another@conference.shakespeare.lit' }).c('nick').t('JC').up().up() .c('conference', { 'name': 'Yet another bookmark', 'autojoin': 'false', 'jid':'yab@conference.shakespeare.lit' }).c('nick').t('JC'); _converse.connection._dataRecv(mock.createRequest(stanza)); await u.waitUntil(() => _converse.bookmarks.length === 3); expect(_converse.bookmarks.map(b => b.get('name'))).toEqual(['Second bookmark', 'The Play's the Thing', 'Yet another bookmark']); expect(_converse.chatboxviews.get('theplay@conference.shakespeare.lit')).not.toBeUndefined(); expect(Object.keys(_converse.chatboxviews.getAll()).length).toBe(2); done(); })); it("can be retrieved from the XMPP server", mock.initConverse( ['chatBoxesFetched'], {}, async function (done, _converse) { const { Strophe, sizzle, u, $iq } = converse.env; await mock.waitForRoster(_converse, 'current', 0); await mock.waitUntilDiscoConfirmed( _converse, _converse.bare_jid, [{'category': 'pubsub', 'type': 'pep'}], ['http://jabber.org/protocol/pubsub#publish-options'] ); /* Client requests all items * ------------------------- * * * * * * */ const IQ_stanzas = _converse.connection.IQ_stanzas; const sent_stanza = await u.waitUntil( () => IQ_stanzas.filter(s => sizzle('items[node="storage:bookmarks"]', s).length).pop()); expect(Strophe.serialize(sent_stanza)).toBe( ``+ ''+ ''+ ''+ ''); /* * Server returns all items * ------------------------ * * * * * * * JC * * * * * * */ expect(_converse.bookmarks.models.length).toBe(0); spyOn(_converse.bookmarks, 'onBookmarksReceived').and.callThrough(); var stanza = $iq({'to': _converse.connection.jid, 'type':'result', 'id':sent_stanza.getAttribute('id')}) .c('pubsub', {'xmlns': Strophe.NS.PUBSUB}) .c('items', {'node': 'storage:bookmarks'}) .c('item', {'id': 'current'}) .c('storage', {'xmlns': 'storage:bookmarks'}) .c('conference', { 'name': 'The Play's the Thing', 'autojoin': 'true', 'jid': 'theplay@conference.shakespeare.lit' }).c('nick').t('JC').up().up() .c('conference', { 'name': 'Another room', 'autojoin': 'false', 'jid': 'another@conference.shakespeare.lit' }); // Purposefully exclude the element to test #1043 _converse.connection._dataRecv(mock.createRequest(stanza)); await u.waitUntil(() => _converse.bookmarks.onBookmarksReceived.calls.count()); await _converse.api.waitUntil('bookmarksInitialized'); expect(_converse.bookmarks.models.length).toBe(2); expect(_converse.bookmarks.findWhere({'jid': 'theplay@conference.shakespeare.lit'}).get('autojoin')).toBe(true); expect(_converse.bookmarks.findWhere({'jid': 'another@conference.shakespeare.lit'}).get('autojoin')).toBe(false); done(); })); describe("The bookmarks list", function () { it("shows a list of bookmarks", mock.initConverse( [], {}, async function (done, _converse) { await mock.waitUntilDiscoConfirmed( _converse, _converse.bare_jid, [{'category': 'pubsub', 'type': 'pep'}], ['http://jabber.org/protocol/pubsub#publish-options'] ); await mock.waitForRoster(_converse, 'current', 0); mock.openControlBox(_converse); const IQ_stanzas = _converse.connection.IQ_stanzas; const sent_stanza = await u.waitUntil( () => IQ_stanzas.filter(s => sizzle('items[node="storage:bookmarks"]', s).length).pop()); expect(Strophe.serialize(sent_stanza)).toBe( ``+ ''+ ''+ ''+ '' ); const stanza = $iq({'to': _converse.connection.jid, 'type':'result', 'id':sent_stanza.getAttribute('id')}) .c('pubsub', {'xmlns': Strophe.NS.PUBSUB}) .c('items', {'node': 'storage:bookmarks'}) .c('item', {'id': 'current'}) .c('storage', {'xmlns': 'storage:bookmarks'}) .c('conference', { 'name': 'The Play's the Thing', 'autojoin': 'false', 'jid': 'theplay@conference.shakespeare.lit' }).c('nick').t('JC').up().up() .c('conference', { 'name': '1st Bookmark', 'autojoin': 'false', 'jid': 'first@conference.shakespeare.lit' }).c('nick').t('JC').up().up() .c('conference', { 'autojoin': 'false', 'jid': 'noname@conference.shakespeare.lit' }).c('nick').t('JC').up().up() .c('conference', { 'name': 'Bookmark with a very very long name that will be shortened', 'autojoin': 'false', 'jid': 'longname@conference.shakespeare.lit' }).c('nick').t('JC').up().up() .c('conference', { 'name': 'Another room', 'autojoin': 'false', 'jid': 'another@conference.shakespeare.lit' }).c('nick').t('JC').up().up(); _converse.connection._dataRecv(mock.createRequest(stanza)); await u.waitUntil(() => document.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item').length); expect(document.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item').length).toBe(5); let els = document.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item a.list-item-link'); expect(els[0].textContent).toBe("1st Bookmark"); expect(els[1].textContent).toBe("Another room"); expect(els[2].textContent).toBe("Bookmark with a very very long name that will be shortened"); expect(els[3].textContent).toBe("noname@conference.shakespeare.lit"); expect(els[4].textContent).toBe("The Play's the Thing"); spyOn(window, 'confirm').and.returnValue(true); document.querySelector('#chatrooms .bookmarks.rooms-list .room-item:nth-child(2) a:nth-child(2)').click(); expect(window.confirm).toHaveBeenCalled(); await u.waitUntil(() => document.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item').length === 4) els = document.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item a.list-item-link'); expect(els[0].textContent).toBe("1st Bookmark"); expect(els[1].textContent).toBe("Bookmark with a very very long name that will be shortened"); expect(els[2].textContent).toBe("noname@conference.shakespeare.lit"); expect(els[3].textContent).toBe("The Play's the Thing"); done(); })); it("can be used to open a MUC from a bookmark", mock.initConverse( [], {'view_mode': 'fullscreen'}, async function (done, _converse) { const api = _converse.api; await mock.waitUntilDiscoConfirmed( _converse, _converse.bare_jid, [{'category': 'pubsub', 'type': 'pep'}], ['http://jabber.org/protocol/pubsub#publish-options'] ); await mock.waitForRoster(_converse, 'current', 0); await mock.openControlBox(_converse); const view = await _converse.chatboxviews.get('controlbox'); const IQ_stanzas = _converse.connection.IQ_stanzas; const sent_stanza = await u.waitUntil( () => IQ_stanzas.filter(s => sizzle('items[node="storage:bookmarks"]', s).length).pop()); const stanza = $iq({'to': _converse.connection.jid, 'type':'result', 'id':sent_stanza.getAttribute('id')}) .c('pubsub', {'xmlns': Strophe.NS.PUBSUB}) .c('items', {'node': 'storage:bookmarks'}) .c('item', {'id': 'current'}) .c('storage', {'xmlns': 'storage:bookmarks'}) .c('conference', { 'name': 'The Play's the Thing', 'autojoin': 'false', 'jid': 'theplay@conference.shakespeare.lit' }).c('nick').t('JC').up().up() .c('conference', { 'name': '1st Bookmark', 'autojoin': 'false', 'jid': 'first@conference.shakespeare.lit' }).c('nick').t('JC'); _converse.connection._dataRecv(mock.createRequest(stanza)); await u.waitUntil(() => view.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item').length); expect(view.querySelectorAll('#chatrooms div.bookmarks.rooms-list .room-item').length).toBe(2); view.querySelector('.bookmarks.rooms-list .open-room').click(); await u.waitUntil(() => _converse.chatboxes.length === 2); expect((await api.rooms.get('first@conference.shakespeare.lit')).get('hidden')).toBe(false); await u.waitUntil(() => view.querySelectorAll('.list-container--bookmarks .available-chatroom:not(.hidden)').length === 1); view.querySelector('.list-container--bookmarks .available-chatroom:not(.hidden) .open-room').click(); await u.waitUntil(() => _converse.chatboxes.length === 3); expect((await api.rooms.get('first@conference.shakespeare.lit')).get('hidden')).toBe(true); expect((await api.rooms.get('theplay@conference.shakespeare.lit')).get('hidden')).toBe(false); view.querySelector('.list-container--openrooms .open-room:first-child').click(); await u.waitUntil(() => view.querySelector('.list-item.open').getAttribute('data-room-jid') === 'first@conference.shakespeare.lit'); expect((await api.rooms.get('first@conference.shakespeare.lit')).get('hidden')).toBe(false); expect((await api.rooms.get('theplay@conference.shakespeare.lit')).get('hidden')).toBe(true); done(); })); it("remembers the toggle state of the bookmarks list", mock.initConverse( [], {}, async function (done, _converse) { await mock.waitForRoster(_converse, 'current', 0); await mock.openControlBox(_converse); await mock.waitUntilDiscoConfirmed( _converse, _converse.bare_jid, [{'category': 'pubsub', 'type': 'pep'}], ['http://jabber.org/protocol/pubsub#publish-options'] ); const { Strophe, u, sizzle, $iq } = converse.env; const IQ_stanzas = _converse.connection.IQ_stanzas; const sent_stanza = await u.waitUntil( () => IQ_stanzas.filter(s => sizzle('iq items[node="storage:bookmarks"]', s).length).pop()); expect(Strophe.serialize(sent_stanza)).toBe( ``+ ''+ ''+ ''+ '' ); const stanza = $iq({'to': _converse.connection.jid, 'type':'result', 'id': sent_stanza.getAttribute('id')}) .c('pubsub', {'xmlns': Strophe.NS.PUBSUB}) .c('items', {'node': 'storage:bookmarks'}) .c('item', {'id': 'current'}) .c('storage', {'xmlns': 'storage:bookmarks'}); _converse.connection._dataRecv(mock.createRequest(stanza)); await _converse.api.waitUntil('bookmarksInitialized'); _converse.bookmarks.create({ 'jid': 'theplay@conference.shakespeare.lit', 'autojoin': false, 'name': 'The Play', 'nick': '' }); const chats_el = document.querySelector('converse-chats'); const selector = '#chatrooms .bookmarks.rooms-list .room-item'; await u.waitUntil(() => sizzle(selector, chats_el).filter(u.isVisible).length); expect(u.hasClass('collapsed', sizzle('#chatrooms .bookmarks.rooms-list', chats_el).pop())).toBeFalsy(); expect(sizzle(selector, chats_el).filter(u.isVisible).length).toBe(1); const bookmarks_el = chats_el.querySelector('converse-bookmarks'); expect(bookmarks_el.model.get('toggle-state')).toBe(_converse.OPENED); sizzle('#chatrooms .bookmarks-toggle', chats_el).pop().click(); expect(u.hasClass('collapsed', sizzle('#chatrooms .bookmarks.rooms-list', chats_el).pop())).toBeTruthy(); expect(bookmarks_el.model.get('toggle-state')).toBe(_converse.CLOSED); sizzle('#chatrooms .bookmarks-toggle', chats_el).pop().click(); expect(u.hasClass('collapsed', sizzle('#chatrooms .bookmarks.rooms-list', chats_el).pop())).toBeFalsy(); expect(sizzle(selector, chats_el).filter(u.isVisible).length).toBe(1); expect(bookmarks_el.model.get('toggle-state')).toBe(_converse.OPENED); done(); })); }); }); describe("When hide_open_bookmarks is true and a bookmarked room is opened", function () { it("can be closed", mock.initConverse( [], { hide_open_bookmarks: true }, async function (done, _converse) { await mock.waitForRoster(_converse, 'current', 0); await mock.openControlBox(_converse); await mock.waitUntilBookmarksReturned(_converse); // Check that it's there const jid = 'room@conference.example.org'; _converse.bookmarks.create({ 'jid': jid, 'autojoin': false, 'name': 'The Play', 'nick': ' Othello' }); expect(_converse.bookmarks.length).toBe(1); const u = converse.env.utils; const bookmarks_el = document.querySelector('converse-bookmarks'); await u.waitUntil(() => bookmarks_el.querySelectorAll(".open-room").length, 500); const room_els = bookmarks_el.querySelectorAll(".open-room"); expect(room_els.length).toBe(1); const bookmark = bookmarks_el.querySelector(".open-room"); bookmark.click(); await u.waitUntil(() => _converse.chatboxviews.get(jid)); expect(u.hasClass('hidden', bookmarks_el.querySelector(".available-chatroom"))).toBeTruthy(); // Check that it reappears once the room is closed const view = _converse.chatboxviews.get(jid); view.close(); await u.waitUntil(() => !u.hasClass('hidden', bookmarks_el.querySelector(".available-chatroom"))); done(); })); });