654 lines
21 KiB
TypeScript
654 lines
21 KiB
TypeScript
// Copyright 2021 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import { assert } from 'chai';
|
|
import * as sinon from 'sinon';
|
|
import { v4 as uuid } from 'uuid';
|
|
import { RowType } from '../../../components/ConversationList';
|
|
import { FindDirection } from '../../../components/leftPane/LeftPaneHelper';
|
|
|
|
import { LeftPaneInboxHelper } from '../../../components/leftPane/LeftPaneInboxHelper';
|
|
|
|
describe('LeftPaneInboxHelper', () => {
|
|
const fakeConversation = () => ({
|
|
id: uuid(),
|
|
title: uuid(),
|
|
type: 'direct' as const,
|
|
});
|
|
|
|
describe('getBackAction', () => {
|
|
it("returns undefined; you can't go back from the main inbox", () => {
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [],
|
|
pinnedConversations: [],
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.isUndefined(
|
|
helper.getBackAction({
|
|
showChooseGroupMembers: sinon.fake(),
|
|
showInbox: sinon.fake(),
|
|
startComposing: sinon.fake(),
|
|
})
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('getRowCount', () => {
|
|
it('returns 0 if there are no conversations', () => {
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [],
|
|
pinnedConversations: [],
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.strictEqual(helper.getRowCount(), 0);
|
|
});
|
|
|
|
it('returns 1 if there are only archived conversations', () => {
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [],
|
|
pinnedConversations: [],
|
|
archivedConversations: [fakeConversation()],
|
|
});
|
|
|
|
assert.strictEqual(helper.getRowCount(), 1);
|
|
});
|
|
|
|
it("returns the number of non-pinned conversations if that's all there is", () => {
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
],
|
|
pinnedConversations: [],
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.strictEqual(helper.getRowCount(), 3);
|
|
});
|
|
|
|
it("returns the number of pinned conversations if that's all there is", () => {
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [],
|
|
pinnedConversations: [
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
],
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.strictEqual(helper.getRowCount(), 3);
|
|
});
|
|
|
|
it('adds 2 rows for each header if there are pinned and non-pinned conversations,', () => {
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
],
|
|
pinnedConversations: [fakeConversation()],
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.strictEqual(helper.getRowCount(), 6);
|
|
});
|
|
|
|
it('adds 1 row for the archive button if there are any archived conversations', () => {
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
],
|
|
pinnedConversations: [],
|
|
archivedConversations: [fakeConversation()],
|
|
});
|
|
|
|
assert.strictEqual(helper.getRowCount(), 4);
|
|
});
|
|
});
|
|
|
|
describe('getRowIndexToScrollTo', () => {
|
|
it('returns undefined if no conversation is selected', () => {
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [fakeConversation(), fakeConversation()],
|
|
pinnedConversations: [fakeConversation()],
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.isUndefined(helper.getRowIndexToScrollTo(undefined));
|
|
});
|
|
|
|
it('returns undefined if the selected conversation is not pinned or non-pinned', () => {
|
|
const archivedConversations = [fakeConversation()];
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [fakeConversation(), fakeConversation()],
|
|
pinnedConversations: [fakeConversation()],
|
|
archivedConversations,
|
|
});
|
|
|
|
assert.isUndefined(
|
|
helper.getRowIndexToScrollTo(archivedConversations[0].id)
|
|
);
|
|
});
|
|
|
|
it("returns the pinned conversation's index if there are only pinned conversations", () => {
|
|
const pinnedConversations = [fakeConversation(), fakeConversation()];
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [],
|
|
pinnedConversations,
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.strictEqual(
|
|
helper.getRowIndexToScrollTo(pinnedConversations[0].id),
|
|
0
|
|
);
|
|
assert.strictEqual(
|
|
helper.getRowIndexToScrollTo(pinnedConversations[1].id),
|
|
1
|
|
);
|
|
});
|
|
|
|
it("returns the conversation's index if there are only non-pinned conversations", () => {
|
|
const conversations = [fakeConversation(), fakeConversation()];
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations,
|
|
pinnedConversations: [],
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.strictEqual(helper.getRowIndexToScrollTo(conversations[0].id), 0);
|
|
assert.strictEqual(helper.getRowIndexToScrollTo(conversations[1].id), 1);
|
|
});
|
|
|
|
it("returns the pinned conversation's index + 1 (for the header) if there are both pinned and non-pinned conversations", () => {
|
|
const pinnedConversations = [fakeConversation(), fakeConversation()];
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [fakeConversation()],
|
|
pinnedConversations,
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.strictEqual(
|
|
helper.getRowIndexToScrollTo(pinnedConversations[0].id),
|
|
1
|
|
);
|
|
assert.strictEqual(
|
|
helper.getRowIndexToScrollTo(pinnedConversations[1].id),
|
|
2
|
|
);
|
|
});
|
|
|
|
it("returns the non-pinned conversation's index + pinnedConversations.length + 2 (for the headers) if there are both pinned and non-pinned conversations", () => {
|
|
const conversations = [fakeConversation(), fakeConversation()];
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations,
|
|
pinnedConversations: [
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
],
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.strictEqual(helper.getRowIndexToScrollTo(conversations[0].id), 5);
|
|
assert.strictEqual(helper.getRowIndexToScrollTo(conversations[1].id), 6);
|
|
});
|
|
});
|
|
|
|
describe('getRow', () => {
|
|
it('returns the archive button if there are only archived conversations', () => {
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [],
|
|
pinnedConversations: [],
|
|
archivedConversations: [fakeConversation(), fakeConversation()],
|
|
});
|
|
|
|
assert.deepEqual(helper.getRow(0), {
|
|
type: RowType.ArchiveButton,
|
|
archivedConversationsCount: 2,
|
|
});
|
|
assert.isUndefined(helper.getRow(1));
|
|
});
|
|
|
|
it("returns pinned conversations if that's all there are", () => {
|
|
const pinnedConversations = [fakeConversation(), fakeConversation()];
|
|
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [],
|
|
pinnedConversations,
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.deepEqual(helper.getRow(0), {
|
|
type: RowType.Conversation,
|
|
conversation: pinnedConversations[0],
|
|
});
|
|
assert.deepEqual(helper.getRow(1), {
|
|
type: RowType.Conversation,
|
|
conversation: pinnedConversations[1],
|
|
});
|
|
assert.isUndefined(helper.getRow(2));
|
|
});
|
|
|
|
it('returns pinned conversations and an archive button if there are no non-pinned conversations', () => {
|
|
const pinnedConversations = [fakeConversation(), fakeConversation()];
|
|
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [],
|
|
pinnedConversations,
|
|
archivedConversations: [fakeConversation()],
|
|
});
|
|
|
|
assert.deepEqual(helper.getRow(0), {
|
|
type: RowType.Conversation,
|
|
conversation: pinnedConversations[0],
|
|
});
|
|
assert.deepEqual(helper.getRow(1), {
|
|
type: RowType.Conversation,
|
|
conversation: pinnedConversations[1],
|
|
});
|
|
assert.deepEqual(helper.getRow(2), {
|
|
type: RowType.ArchiveButton,
|
|
archivedConversationsCount: 1,
|
|
});
|
|
assert.isUndefined(helper.getRow(3));
|
|
});
|
|
|
|
it("returns non-pinned conversations if that's all there are", () => {
|
|
const conversations = [fakeConversation(), fakeConversation()];
|
|
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations,
|
|
pinnedConversations: [],
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.deepEqual(helper.getRow(0), {
|
|
type: RowType.Conversation,
|
|
conversation: conversations[0],
|
|
});
|
|
assert.deepEqual(helper.getRow(1), {
|
|
type: RowType.Conversation,
|
|
conversation: conversations[1],
|
|
});
|
|
assert.isUndefined(helper.getRow(2));
|
|
});
|
|
|
|
it('returns non-pinned conversations and an archive button if there are no pinned conversations', () => {
|
|
const conversations = [fakeConversation(), fakeConversation()];
|
|
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations,
|
|
pinnedConversations: [],
|
|
archivedConversations: [fakeConversation()],
|
|
});
|
|
|
|
assert.deepEqual(helper.getRow(0), {
|
|
type: RowType.Conversation,
|
|
conversation: conversations[0],
|
|
});
|
|
assert.deepEqual(helper.getRow(1), {
|
|
type: RowType.Conversation,
|
|
conversation: conversations[1],
|
|
});
|
|
assert.deepEqual(helper.getRow(2), {
|
|
type: RowType.ArchiveButton,
|
|
archivedConversationsCount: 1,
|
|
});
|
|
assert.isUndefined(helper.getRow(3));
|
|
});
|
|
|
|
it('returns headers if there are both pinned and non-pinned conversations', () => {
|
|
const conversations = [
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
];
|
|
const pinnedConversations = [fakeConversation(), fakeConversation()];
|
|
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations,
|
|
pinnedConversations,
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.deepEqual(helper.getRow(0), {
|
|
type: RowType.Header,
|
|
i18nKey: 'LeftPane--pinned',
|
|
});
|
|
assert.deepEqual(helper.getRow(1), {
|
|
type: RowType.Conversation,
|
|
conversation: pinnedConversations[0],
|
|
});
|
|
assert.deepEqual(helper.getRow(2), {
|
|
type: RowType.Conversation,
|
|
conversation: pinnedConversations[1],
|
|
});
|
|
assert.deepEqual(helper.getRow(3), {
|
|
type: RowType.Header,
|
|
i18nKey: 'LeftPane--chats',
|
|
});
|
|
assert.deepEqual(helper.getRow(4), {
|
|
type: RowType.Conversation,
|
|
conversation: conversations[0],
|
|
});
|
|
assert.deepEqual(helper.getRow(5), {
|
|
type: RowType.Conversation,
|
|
conversation: conversations[1],
|
|
});
|
|
assert.deepEqual(helper.getRow(6), {
|
|
type: RowType.Conversation,
|
|
conversation: conversations[2],
|
|
});
|
|
assert.isUndefined(helper.getRow(7));
|
|
});
|
|
|
|
it('returns headers if there are both pinned and non-pinned conversations, and an archive button', () => {
|
|
const conversations = [
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
];
|
|
const pinnedConversations = [fakeConversation(), fakeConversation()];
|
|
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations,
|
|
pinnedConversations,
|
|
archivedConversations: [fakeConversation()],
|
|
});
|
|
|
|
assert.deepEqual(helper.getRow(0), {
|
|
type: RowType.Header,
|
|
i18nKey: 'LeftPane--pinned',
|
|
});
|
|
assert.deepEqual(helper.getRow(1), {
|
|
type: RowType.Conversation,
|
|
conversation: pinnedConversations[0],
|
|
});
|
|
assert.deepEqual(helper.getRow(2), {
|
|
type: RowType.Conversation,
|
|
conversation: pinnedConversations[1],
|
|
});
|
|
assert.deepEqual(helper.getRow(3), {
|
|
type: RowType.Header,
|
|
i18nKey: 'LeftPane--chats',
|
|
});
|
|
assert.deepEqual(helper.getRow(4), {
|
|
type: RowType.Conversation,
|
|
conversation: conversations[0],
|
|
});
|
|
assert.deepEqual(helper.getRow(5), {
|
|
type: RowType.Conversation,
|
|
conversation: conversations[1],
|
|
});
|
|
assert.deepEqual(helper.getRow(6), {
|
|
type: RowType.Conversation,
|
|
conversation: conversations[2],
|
|
});
|
|
assert.deepEqual(helper.getRow(7), {
|
|
type: RowType.ArchiveButton,
|
|
archivedConversationsCount: 1,
|
|
});
|
|
assert.isUndefined(helper.getRow(8));
|
|
});
|
|
});
|
|
|
|
describe('getConversationAndMessageAtIndex', () => {
|
|
it('returns pinned converastions, then non-pinned conversations', () => {
|
|
const conversations = [
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
];
|
|
const pinnedConversations = [fakeConversation(), fakeConversation()];
|
|
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations,
|
|
pinnedConversations,
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.strictEqual(
|
|
helper.getConversationAndMessageAtIndex(0)?.conversationId,
|
|
pinnedConversations[0].id
|
|
);
|
|
assert.strictEqual(
|
|
helper.getConversationAndMessageAtIndex(1)?.conversationId,
|
|
pinnedConversations[1].id
|
|
);
|
|
assert.strictEqual(
|
|
helper.getConversationAndMessageAtIndex(2)?.conversationId,
|
|
conversations[0].id
|
|
);
|
|
assert.strictEqual(
|
|
helper.getConversationAndMessageAtIndex(3)?.conversationId,
|
|
conversations[1].id
|
|
);
|
|
assert.strictEqual(
|
|
helper.getConversationAndMessageAtIndex(4)?.conversationId,
|
|
conversations[2].id
|
|
);
|
|
});
|
|
|
|
it("when requesting an index out of bounds, returns the last pinned conversation when that's all there is", () => {
|
|
const pinnedConversations = [fakeConversation(), fakeConversation()];
|
|
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [],
|
|
pinnedConversations,
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.strictEqual(
|
|
helper.getConversationAndMessageAtIndex(2)?.conversationId,
|
|
pinnedConversations[1].id
|
|
);
|
|
assert.strictEqual(
|
|
helper.getConversationAndMessageAtIndex(99)?.conversationId,
|
|
pinnedConversations[1].id
|
|
);
|
|
// This is mostly a resilience measure in case we're ever called with an invalid
|
|
// index.
|
|
assert.strictEqual(
|
|
helper.getConversationAndMessageAtIndex(-1)?.conversationId,
|
|
pinnedConversations[1].id
|
|
);
|
|
});
|
|
|
|
it("when requesting an index out of bounds, returns the last non-pinned conversation when that's all there is", () => {
|
|
const conversations = [fakeConversation(), fakeConversation()];
|
|
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations,
|
|
pinnedConversations: [],
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.strictEqual(
|
|
helper.getConversationAndMessageAtIndex(2)?.conversationId,
|
|
conversations[1].id
|
|
);
|
|
assert.strictEqual(
|
|
helper.getConversationAndMessageAtIndex(99)?.conversationId,
|
|
conversations[1].id
|
|
);
|
|
// This is mostly a resilience measure in case we're ever called with an invalid
|
|
// index.
|
|
assert.strictEqual(
|
|
helper.getConversationAndMessageAtIndex(-1)?.conversationId,
|
|
conversations[1].id
|
|
);
|
|
});
|
|
|
|
it('when requesting an index out of bounds, returns the last non-pinned conversation when there are both pinned and non-pinned conversations', () => {
|
|
const conversations = [fakeConversation(), fakeConversation()];
|
|
const pinnedConversations = [fakeConversation(), fakeConversation()];
|
|
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations,
|
|
pinnedConversations,
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.strictEqual(
|
|
helper.getConversationAndMessageAtIndex(4)?.conversationId,
|
|
conversations[1].id
|
|
);
|
|
assert.strictEqual(
|
|
helper.getConversationAndMessageAtIndex(99)?.conversationId,
|
|
conversations[1].id
|
|
);
|
|
// This is mostly a resilience measure in case we're ever called with an invalid
|
|
// index.
|
|
assert.strictEqual(
|
|
helper.getConversationAndMessageAtIndex(-1)?.conversationId,
|
|
conversations[1].id
|
|
);
|
|
});
|
|
|
|
it('returns undefined if there are no conversations', () => {
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [],
|
|
pinnedConversations: [],
|
|
archivedConversations: [fakeConversation()],
|
|
});
|
|
|
|
assert.isUndefined(helper.getConversationAndMessageAtIndex(0));
|
|
assert.isUndefined(helper.getConversationAndMessageAtIndex(1));
|
|
assert.isUndefined(helper.getConversationAndMessageAtIndex(-1));
|
|
});
|
|
});
|
|
|
|
describe('getConversationAndMessageInDirection', () => {
|
|
it('returns the next conversation when searching downward', () => {
|
|
const pinnedConversations = [fakeConversation(), fakeConversation()];
|
|
const conversations = [fakeConversation()];
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations,
|
|
pinnedConversations,
|
|
archivedConversations: [],
|
|
});
|
|
|
|
assert.deepEqual(
|
|
helper.getConversationAndMessageInDirection(
|
|
{ direction: FindDirection.Down, unreadOnly: false },
|
|
pinnedConversations[1].id,
|
|
undefined
|
|
),
|
|
{ conversationId: conversations[0].id }
|
|
);
|
|
});
|
|
|
|
// Additional tests are found with `getConversationInDirection`.
|
|
});
|
|
|
|
describe('shouldRecomputeRowHeights', () => {
|
|
it("returns false if the number of conversations in each section doesn't change", () => {
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
],
|
|
pinnedConversations: [fakeConversation(), fakeConversation()],
|
|
archivedConversations: [fakeConversation()],
|
|
});
|
|
|
|
assert.isFalse(
|
|
helper.shouldRecomputeRowHeights({
|
|
conversations: [
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
],
|
|
pinnedConversations: [fakeConversation(), fakeConversation()],
|
|
archivedConversations: [fakeConversation(), fakeConversation()],
|
|
})
|
|
);
|
|
});
|
|
|
|
it('returns false if the only thing changed is whether conversations are archived', () => {
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
],
|
|
pinnedConversations: [fakeConversation(), fakeConversation()],
|
|
archivedConversations: [fakeConversation()],
|
|
});
|
|
|
|
assert.isFalse(
|
|
helper.shouldRecomputeRowHeights({
|
|
conversations: [
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
],
|
|
pinnedConversations: [fakeConversation(), fakeConversation()],
|
|
archivedConversations: [],
|
|
})
|
|
);
|
|
});
|
|
|
|
it('returns false if the only thing changed is the number of non-pinned conversations', () => {
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
],
|
|
pinnedConversations: [fakeConversation(), fakeConversation()],
|
|
archivedConversations: [fakeConversation()],
|
|
});
|
|
|
|
assert.isFalse(
|
|
helper.shouldRecomputeRowHeights({
|
|
conversations: [fakeConversation()],
|
|
pinnedConversations: [fakeConversation(), fakeConversation()],
|
|
archivedConversations: [fakeConversation(), fakeConversation()],
|
|
})
|
|
);
|
|
});
|
|
|
|
it('returns true if the number of pinned conversations changes', () => {
|
|
const helper = new LeftPaneInboxHelper({
|
|
conversations: [fakeConversation()],
|
|
pinnedConversations: [fakeConversation(), fakeConversation()],
|
|
archivedConversations: [fakeConversation()],
|
|
});
|
|
|
|
assert.isTrue(
|
|
helper.shouldRecomputeRowHeights({
|
|
conversations: [fakeConversation()],
|
|
pinnedConversations: [
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
fakeConversation(),
|
|
],
|
|
archivedConversations: [fakeConversation()],
|
|
})
|
|
);
|
|
assert.isTrue(
|
|
helper.shouldRecomputeRowHeights({
|
|
conversations: [fakeConversation()],
|
|
pinnedConversations: [fakeConversation()],
|
|
archivedConversations: [fakeConversation()],
|
|
})
|
|
);
|
|
assert.isTrue(
|
|
helper.shouldRecomputeRowHeights({
|
|
conversations: [fakeConversation()],
|
|
pinnedConversations: [],
|
|
archivedConversations: [fakeConversation()],
|
|
})
|
|
);
|
|
});
|
|
});
|
|
});
|