Rocket.Chat.ReactNative/e2e/tests/room/02-room.spec.ts

552 lines
22 KiB
TypeScript

import { device, waitFor, element, by, expect } from 'detox';
import {
navigateToLogin,
login,
tapBack,
sleep,
searchRoom,
tryTapping,
platformTypes,
TTextMatcher,
mockMessage
} from '../../helpers/app';
import { createRandomRoom, createRandomUser, ITestUser, sendMessage } from '../../helpers/data_setup';
async function navigateToRoom(roomName: string) {
await searchRoom(`${roomName}`);
await element(by.id(`rooms-list-view-item-${roomName}`)).tap();
await waitFor(element(by.id('room-view')))
.toBeVisible()
.withTimeout(5000);
}
describe('Room screen', () => {
let room: string;
let alertButtonType: string;
let textMatcher: TTextMatcher;
let user: ITestUser;
let randomMessage: string;
beforeAll(async () => {
user = await createRandomUser();
({ name: room } = await createRandomRoom(user));
await device.launchApp({ permissions: { notifications: 'YES' }, delete: true });
({ alertButtonType, textMatcher } = platformTypes[device.getPlatform()]);
await navigateToLogin();
await login(user.username, user.password);
await navigateToRoom(room);
});
describe('Render', () => {
it('should have room screen', async () => {
await expect(element(by.id('room-view'))).toExist();
await waitFor(element(by.id(`room-view-title-${room}`)))
.toExist()
.withTimeout(5000);
});
// Render - Header
describe('Header', () => {
it('should have actions button ', async () => {
await expect(element(by.id('room-header'))).toExist();
});
it('should have threads button ', async () => {
await expect(element(by.id('room-view-header-threads'))).toExist();
});
});
// Render - Messagebox
describe('Messagebox', () => {
it('should have messagebox', async () => {
await expect(element(by.id('messagebox'))).toExist();
});
it('should have open emoji button', async () => {
await expect(element(by.id('messagebox-open-emoji'))).toExist();
});
it('should have message input', async () => {
await expect(element(by.id('messagebox-input'))).toExist();
});
it('should have audio button', async () => {
await expect(element(by.id('messagebox-send-audio'))).toExist();
});
it('should have actions button', async () => {
await expect(element(by.id('messagebox-actions'))).toExist();
});
});
});
describe('Usage', () => {
describe('Messagebox', () => {
it('should send message', async () => {
randomMessage = await mockMessage('message');
});
// describe('Emoji Keyboard', () => {
// it('should open emoji keyboard, select an emoji and send it', async () => {
// await element(by.id('messagebox-open-emoji')).tap();
// await waitFor(element(by.id('messagebox-keyboard-emoji')))
// .toExist()
// .withTimeout(10000);
// await waitFor(element(by.id('emoji-picker-tab-emoji')))
// .toExist()
// .withTimeout(10000);
// await element(by.id('emoji-picker-tab-emoji')).tap();
// await expect(element(by.id('emoji-blush'))).toExist();
// await element(by.id('emoji-blush')).tap();
// await expect(element(by.id('messagebox-input'))).toHaveText('😊');
// await element(by.id('messagebox-send-message')).tap();
// await waitFor(element(by[textMatcher]('😊')))
// .toExist()
// .withTimeout(60000);
// await element(by[textMatcher]('😊')).atIndex(0).tap();
// });
// it('should open emoji keyboard, select an emoji and delete it using emoji keyboards backspace', async () => {
// await element(by.id('messagebox-open-emoji')).tap();
// await waitFor(element(by.id('messagebox-keyboard-emoji')))
// .toExist()
// .withTimeout(10000);
// await expect(element(by.id('emoji-picker-tab-emoji'))).toExist();
// await element(by.id('emoji-picker-tab-emoji')).tap();
// await expect(element(by.id('emoji-upside_down'))).toExist();
// await element(by.id('emoji-upside_down')).tap();
// await expect(element(by.id('messagebox-input'))).toHaveText('🙃');
// await waitFor(element(by.id('emoji-picker-backspace')))
// .toExist()
// .withTimeout(2000);
// await element(by.id('emoji-picker-backspace')).tap();
// await expect(element(by.id('messagebox-input'))).toHaveText('');
// await element(by.id('messagebox-close-emoji')).tap();
// await waitFor(element(by.id('messagebox-keyboard-emoji')))
// .not.toBeVisible()
// .withTimeout(10000);
// });
// it('should search emoji and send it', async () => {
// await element(by.id('messagebox-open-emoji')).tap();
// await waitFor(element(by.id('emoji-picker-search')))
// .toExist()
// .withTimeout(4000);
// await element(by.id('emoji-picker-search')).tap();
// await waitFor(element(by.id('emoji-searchbar-input')))
// .toExist()
// .withTimeout(2000);
// await element(by.id('emoji-searchbar-input')).replaceText('no_mouth');
// await waitFor(element(by.id('emoji-no_mouth')))
// .toExist()
// .withTimeout(2000);
// await element(by.id('emoji-no_mouth')).tap();
// await expect(element(by.id('messagebox-input'))).toHaveText('😶');
// await element(by.id('messagebox-send-message')).tap();
// await waitFor(element(by[textMatcher]('😶')))
// .toExist()
// .withTimeout(60000);
// await element(by[textMatcher]('😶')).atIndex(0).tap();
// });
// it('should search emojis, go back to Emoji keyboard and then close the Emoji keyboard', async () => {
// await element(by.id('messagebox-open-emoji')).tap();
// await waitFor(element(by.id('emoji-picker-search')))
// .toExist()
// .withTimeout(4000);
// await element(by.id('emoji-picker-search')).tap();
// await waitFor(element(by.id('emoji-searchbar-input')))
// .toExist()
// .withTimeout(2000);
// await element(by.id('openback-emoji-keyboard')).tap();
// await waitFor(element(by.id('emoji-searchbar-input')))
// .not.toBeVisible()
// .withTimeout(2000);
// await expect(element(by.id('messagebox-close-emoji'))).toExist();
// await element(by.id('messagebox-close-emoji')).tap();
// await waitFor(element(by.id('messagebox-keyboard-emoji')))
// .not.toBeVisible()
// .withTimeout(10000);
// });
// it('frequently used emojis should contain the recently used emojis', async () => {
// await element(by.id('messagebox-open-emoji')).tap();
// await waitFor(element(by.id('emoji-picker-tab-clock')));
// await element(by.id('emoji-picker-tab-clock')).tap();
// await waitFor(element(by.id('emoji-blush')))
// .toExist()
// .withTimeout(2000);
// await waitFor(element(by.id('emoji-upside_down')))
// .toExist()
// .withTimeout(2000);
// await waitFor(element(by.id('emoji-no_mouth')))
// .toExist()
// .withTimeout(2000);
// await expect(element(by.id('messagebox-close-emoji'))).toExist();
// await element(by.id('messagebox-close-emoji')).tap();
// await waitFor(element(by.id('messagebox-keyboard-emoji')))
// .not.toBeVisible()
// .withTimeout(10000);
// });
// });
it('should show/hide emoji autocomplete', async () => {
await element(by.id('messagebox-input')).clearText();
await element(by.id('messagebox-input')).typeText(':joy');
await sleep(300);
await waitFor(element(by.id('messagebox-container')))
.toExist()
.withTimeout(10000);
await element(by.id('messagebox-input')).clearText();
await waitFor(element(by.id('messagebox-container')))
.toBeNotVisible()
.withTimeout(10000);
});
it('should show and tap on emoji autocomplete', async () => {
await element(by.id('messagebox-input')).typeText(':joy');
await sleep(300);
await waitFor(element(by.id('messagebox-container')))
.toExist()
.withTimeout(10000);
await waitFor(element(by.id('mention-item-joy')))
.toExist()
.withTimeout(10000);
await element(by.id('mention-item-joy')).tap();
await expect(element(by.id('messagebox-input'))).toHaveText(':joy: ');
await element(by.id('messagebox-input')).clearText();
});
it('should not show emoji autocomplete on semicolon in middle of a string', async () => {
await element(by.id('messagebox-input')).typeText('name:is');
await sleep(300);
await waitFor(element(by.id('messagebox-container')))
.toNotExist()
.withTimeout(20000);
await element(by.id('messagebox-input')).clearText();
});
it('should show and tap on user autocomplete and send mention', async () => {
const { username } = user;
const messageMention = `@${username}`;
const message = 'mention';
const fullMessage = `${messageMention} ${message}`;
await element(by.id('messagebox-input')).typeText(`@${username}`);
await sleep(300);
await waitFor(element(by.id('messagebox-container')))
.toExist()
.withTimeout(4000);
await waitFor(element(by.id(`mention-item-${username}`)))
.toBeVisible()
.withTimeout(4000);
await tryTapping(element(by.id(`mention-item-${username}`)), 2000);
await expect(element(by.id('messagebox-input'))).toHaveText(`${messageMention} `);
if (device.getPlatform() === 'ios') {
await element(by.id('messagebox-input')).typeText(message);
await element(by.id('messagebox-send-message')).tap();
const fullMessageMatcher = fullMessage.substr(1); // removes `@`
await waitFor(element(by[textMatcher](fullMessageMatcher)))
.toExist()
.withTimeout(60000);
await expect(element(by[textMatcher](fullMessageMatcher))).toExist();
await element(by[textMatcher](fullMessageMatcher)).atIndex(0).tap();
} else {
await element(by.id('messagebox-input')).replaceText(fullMessage);
await element(by.id('messagebox-send-message')).tap();
}
});
it('should not show user autocomplete on @ in the middle of a string', async () => {
await element(by.id('messagebox-input')).typeText('email@gmail');
await waitFor(element(by.id('messagebox-container')))
.toNotExist()
.withTimeout(2000);
await element(by.id('messagebox-input')).clearText();
});
it('should show and tap on room autocomplete', async () => {
await element(by.id('messagebox-input')).typeText('#general');
await waitFor(element(by.id('mention-item-general')))
.toBeVisible()
.withTimeout(4000);
await tryTapping(element(by.id('mention-item-general')), 2000);
await expect(element(by.id('messagebox-input'))).toHaveText('#general ');
await element(by.id('messagebox-input')).clearText();
});
it('should not show room autocomplete on # in middle of a string', async () => {
await element(by.id('messagebox-input')).typeText('te#gen');
await waitFor(element(by.id('messagebox-container')))
.toNotExist()
.withTimeout(4000);
await element(by.id('messagebox-input')).clearText();
});
it('should draft message', async () => {
const draftMessage = 'draft';
await element(by.id('messagebox-input')).replaceText(draftMessage);
await tapBack();
await navigateToRoom(room);
await expect(element(by.id('messagebox-input'))).toHaveText(draftMessage);
await element(by.id('messagebox-input')).clearText();
await tapBack();
await navigateToRoom(room);
await expect(element(by.id('messagebox-input'))).toHaveText('');
});
});
describe('Message', () => {
it('should copy link', async () => {
await element(by[textMatcher](randomMessage)).atIndex(0).longPress();
await waitFor(element(by.id('action-sheet')))
.toExist()
.withTimeout(2000);
await expect(element(by.id('action-sheet-handle'))).toBeVisible();
await element(by.id('action-sheet-handle')).swipe('up', 'fast', 0.5);
await element(by[textMatcher]('Get link')).atIndex(0).tap();
// TODO: test clipboard
});
it('should copy message', async () => {
await element(by[textMatcher](randomMessage)).atIndex(0).longPress();
await waitFor(element(by.id('action-sheet')))
.toExist()
.withTimeout(2000);
await expect(element(by.id('action-sheet-handle'))).toBeVisible();
await element(by.id('action-sheet-handle')).swipe('up', 'fast', 0.5);
await element(by[textMatcher]('Copy')).atIndex(0).tap();
// TODO: test clipboard
});
it('should react to message', async () => {
await waitFor(element(by[textMatcher](randomMessage)))
.toExist()
.withTimeout(60000);
await element(by[textMatcher](randomMessage)).atIndex(0).tap();
await element(by[textMatcher](randomMessage)).atIndex(0).longPress();
await sleep(300); // wait for animation
await waitFor(element(by.id('action-sheet')))
.toExist()
.withTimeout(2000);
await element(by.id('action-sheet-handle')).swipe('up', 'fast', 0.5);
await element(by.id('add-reaction')).tap();
await sleep(300); // wait for animation
await waitFor(element(by.id('emoji-picker-tab-emoji')))
.toExist()
.withTimeout(2000);
await element(by.id('emoji-picker-tab-emoji')).tap();
await waitFor(element(by.id('emoji-grinning')))
.toExist()
.withTimeout(10000);
await element(by.id('emoji-grinning')).tap();
await waitFor(element(by.id('message-reaction-:grinning:')))
.toExist()
.withTimeout(60000);
});
it('should search emojis in the reaction picker and react', async () => {
await element(by[textMatcher](randomMessage)).atIndex(0).longPress();
await waitFor(element(by.id('action-sheet')))
.toExist()
.withTimeout(2000);
await expect(element(by.id('action-sheet-handle'))).toBeVisible();
await element(by.id('action-sheet-handle')).swipe('up', 'fast', 0.5);
await element(by.id('add-reaction')).tap();
await waitFor(element(by.id('emoji-searchbar-input')))
.toBeVisible()
.withTimeout(2000);
await element(by.id('emoji-searchbar-input')).replaceText('laughing');
await waitFor(element(by.id('emoji-laughing')))
.toExist()
.withTimeout(4000);
await element(by.id('emoji-laughing')).tap();
await waitFor(element(by.id('message-reaction-:laughing:')))
.toExist()
.withTimeout(60000);
});
it('should remove reaction', async () => {
await element(by.id('message-reaction-:grinning:')).tap();
await waitFor(element(by.id('message-reaction-:grinning:')))
.not.toExist()
.withTimeout(60000);
});
it('should react to message with frequently used emoji', async () => {
await element(by[textMatcher](randomMessage)).atIndex(0).longPress();
await waitFor(element(by.id('action-sheet')))
.toExist()
.withTimeout(2000);
await expect(element(by.id('action-sheet-handle'))).toBeVisible();
await element(by.id('action-sheet-handle')).swipe('up', 'fast', 0.5);
await sleep(300); // wait for animation
await waitFor(element(by.id('message-actions-emoji-grinning')))
.toBeVisible()
.withTimeout(2000);
await element(by.id('message-actions-emoji-grinning')).tap();
await waitFor(element(by.id('message-reaction-:grinning:')))
.toBeVisible()
.withTimeout(60000);
});
it('should show reaction picker on add reaction button pressed and have frequently used emoji', async () => {
await element(by.id('message-add-reaction')).tap();
await waitFor(element(by.id('action-sheet')))
.toExist()
.withTimeout(2000);
await expect(element(by.id('action-sheet-handle'))).toBeVisible();
await element(by.id('action-sheet-handle')).swipe('up', 'fast', 1);
await waitFor(element(by.id('emoji-grinning')))
.toExist()
.withTimeout(4000);
await waitFor(element(by.id('emoji-picker-tab-emoji')))
.toExist()
.withTimeout(2000);
await element(by.id('emoji-picker-tab-emoji')).tap();
await waitFor(element(by.id('emoji-wink')))
.toExist()
.withTimeout(10000);
await element(by.id('emoji-wink')).tap();
await waitFor(element(by.id('message-reaction-:wink:')))
.toExist()
.withTimeout(60000);
});
it('should open/close reactions list', async () => {
await element(by.id('message-reaction-:laughing:')).longPress();
await waitFor(element(by.id('reactionsList')))
.toExist()
.withTimeout(4000);
await expect(element(by.id('action-sheet-handle'))).toBeVisible();
await element(by.id('action-sheet-handle')).swipe('down', 'fast', 0.5);
});
it('should edit message', async () => {
const editMessage = await mockMessage('edit');
const editedMessage = `${editMessage}ed`;
await tryTapping(element(by[textMatcher](editMessage)).atIndex(0), 2000, true);
await waitFor(element(by.id('action-sheet')))
.toExist()
.withTimeout(2000);
await expect(element(by.id('action-sheet-handle'))).toBeVisible();
await element(by.id('action-sheet-handle')).swipe('up', 'fast', 0.5);
await element(by[textMatcher]('Edit')).atIndex(0).tap();
await element(by.id('messagebox-input')).replaceText(editedMessage);
await element(by.id('messagebox-send-message')).tap();
await waitFor(element(by[textMatcher](editedMessage)).atIndex(0))
.toExist()
.withTimeout(60000);
});
let quotedMessage = '';
it('should quote message', async () => {
const quoteMessage = await mockMessage('quote');
quotedMessage = `${quoteMessage}d`;
await tryTapping(element(by[textMatcher](quoteMessage)).atIndex(0), 2000, true);
await waitFor(element(by.id('action-sheet')))
.toExist()
.withTimeout(2000);
await expect(element(by.id('action-sheet-handle'))).toBeVisible();
await element(by.id('action-sheet-handle')).swipe('up', 'fast', 0.5);
await element(by[textMatcher]('Quote')).atIndex(0).tap();
await element(by.id('messagebox-input')).replaceText(quotedMessage);
await waitFor(element(by.id('messagebox-send-message')))
.toExist()
.withTimeout(2000);
await element(by.id('messagebox-send-message')).tap();
await waitFor(element(by[textMatcher](quotedMessage)).atIndex(0))
.toBeVisible()
.withTimeout(3000);
await waitFor(element(by.id(`reply-${user.name}-${quoteMessage}`).withDescendant(by[textMatcher](quoteMessage))))
.toBeVisible()
.withTimeout(3000);
});
it('should back to rooms list view and see the last message correctly and navigate again to room', async () => {
const expectedLastMessage = `You: ${quotedMessage}`;
await sleep(300);
await tapBack();
await waitFor(element(by.id(`markdown-preview-${expectedLastMessage}`)))
.toBeVisible()
.withTimeout(5000);
await element(by.id(`markdown-preview-${expectedLastMessage}`)).tap();
});
it('should delete message', async () => {
const deleteMessage = await mockMessage('delete');
await tryTapping(element(by[textMatcher](deleteMessage)).atIndex(0), 2000, true);
await waitFor(element(by.id('action-sheet')))
.toExist()
.withTimeout(2000);
await expect(element(by.id('action-sheet-handle'))).toBeVisible();
await element(by.id('action-sheet-handle')).swipe('up', 'fast', 0.5);
await sleep(300); // wait for animation
await waitFor(element(by[textMatcher]('Delete')))
.toExist()
.withTimeout(2000);
await element(by[textMatcher]('Delete')).atIndex(0).tap();
const deleteAlertMessage = 'You will not be able to recover this message!';
await waitFor(element(by[textMatcher](deleteAlertMessage)).atIndex(0))
.toExist()
.withTimeout(10000);
await element(by[textMatcher]('Delete').and(by.type(alertButtonType))).tap();
await waitFor(element(by[textMatcher](deleteMessage)).atIndex(0))
.not.toExist()
.withTimeout(2000);
await tapBack();
});
it('should reply in DM to another user', async () => {
const replyUser = await createRandomUser();
const { name: replyRoom } = await createRandomRoom(replyUser, 'c');
const originalMessage = 'Message to reply in DM';
const replyMessage = 'replied in dm';
await sendMessage(replyUser, replyRoom, originalMessage);
await waitFor(element(by.id('rooms-list-view')))
.toBeVisible()
.withTimeout(2000);
await navigateToRoom(replyRoom);
await waitFor(element(by[textMatcher](originalMessage)).atIndex(0))
.toBeVisible()
.withTimeout(10000);
await element(by.id('room-view-join-button')).tap();
await waitFor(element(by.id('room-view-join-button')))
.not.toBeVisible()
.withTimeout(10000);
await element(by[textMatcher](originalMessage)).atIndex(0).longPress();
await sleep(600); // wait for animation
await waitFor(element(by.id('action-sheet')))
.toExist()
.withTimeout(2000);
await sleep(600); // wait for animation
// Fix android flaky test. Close the action sheet, then re-open again
await element(by.id('action-sheet-handle')).swipe('down', 'fast', 0.5);
await sleep(1000); // wait for animation
await element(by[textMatcher](originalMessage)).atIndex(0).longPress();
await sleep(600); // wait for animation
await waitFor(element(by.id('action-sheet')))
.toExist()
.withTimeout(2000);
await sleep(600); // wait for animation
await waitFor(element(by[textMatcher]('Reply in direct message')).atIndex(0))
.toBeVisible()
.withTimeout(6000);
await sleep(600); // wait for animation
await element(by[textMatcher]('Reply in direct message')).atIndex(0).tap();
await sleep(600); // wait for animation
await waitFor(element(by.id(`room-view-title-${replyUser.username}`)))
.toExist()
.withTimeout(6000);
await element(by.id('messagebox-input')).replaceText(replyMessage);
await waitFor(element(by.id('messagebox-send-message')))
.toExist()
.withTimeout(2000);
await element(by.id('messagebox-send-message')).tap();
await waitFor(element(by[textMatcher](replyMessage)))
.toExist()
.withTimeout(60000);
await element(by[textMatcher](replyMessage)).atIndex(0).tap();
});
});
});
});