2021-06-01 23:30:25 +00:00
|
|
|
// Copyright 2021 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import { mapValues, pickBy } from 'lodash';
|
2023-01-13 20:07:26 +00:00
|
|
|
import type { ReadonlyDeep } from 'type-fest';
|
2021-06-01 23:30:25 +00:00
|
|
|
import { groupBy, map, filter } from './iterables';
|
|
|
|
import { getOwn } from './getOwn';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { ConversationType } from '../state/ducks/conversations';
|
2021-06-01 23:30:25 +00:00
|
|
|
import { isConversationNameKnown } from './isConversationNameKnown';
|
2021-06-02 17:24:22 +00:00
|
|
|
import { isInSystemContacts } from './isInSystemContacts';
|
2021-06-01 23:30:25 +00:00
|
|
|
|
2022-12-23 00:13:23 +00:00
|
|
|
export type GroupNameCollisionsWithIdsByTitle = Readonly<
|
|
|
|
Record<string, Array<string>>
|
|
|
|
>;
|
2021-06-01 23:30:25 +00:00
|
|
|
export type GroupNameCollisionsWithConversationsByTitle = Record<
|
|
|
|
string,
|
|
|
|
Array<ConversationType>
|
|
|
|
>;
|
|
|
|
export type GroupNameCollisionsWithTitlesById = Record<string, string>;
|
|
|
|
|
|
|
|
export const dehydrateCollisionsWithConversations = (
|
|
|
|
withConversations: Readonly<GroupNameCollisionsWithConversationsByTitle>
|
|
|
|
): GroupNameCollisionsWithIdsByTitle =>
|
|
|
|
mapValues(withConversations, conversations => conversations.map(c => c.id));
|
|
|
|
|
|
|
|
export function getCollisionsFromMemberships(
|
|
|
|
memberships: Iterable<{ member: ConversationType }>
|
|
|
|
): GroupNameCollisionsWithConversationsByTitle {
|
|
|
|
const members = map(memberships, membership => membership.member);
|
|
|
|
const candidateMembers = filter(
|
|
|
|
members,
|
|
|
|
member => !member.isMe && isConversationNameKnown(member)
|
|
|
|
);
|
|
|
|
const groupedByTitle = groupBy(candidateMembers, member => member.title);
|
|
|
|
// This cast is here because `pickBy` returns a `Partial`, which is incompatible with
|
2023-01-01 11:41:40 +00:00
|
|
|
// `Record`. [This demonstrates the problem][0], but I don't believe it's an actual
|
2021-06-01 23:30:25 +00:00
|
|
|
// issue in the code.
|
|
|
|
//
|
|
|
|
// Alternatively, we could filter undefined keys or something like that.
|
|
|
|
//
|
|
|
|
// [0]: https://www.typescriptlang.org/play?#code/C4TwDgpgBAYg9nKBeKAFAhgJ2AS3QGwB4AlCAYzkwBNCBnYTHAOwHMAaKJgVwFsAjCJgB8QgNwAoCk3pQAZgC5YCZFADeUABY5FAVigBfCeNCQoAISwrSFanQbN2nXgOESpMvoouYVs0UA
|
2021-11-11 22:43:05 +00:00
|
|
|
return pickBy(
|
2021-06-01 23:30:25 +00:00
|
|
|
groupedByTitle,
|
2021-06-02 17:24:22 +00:00
|
|
|
group =>
|
|
|
|
group.length >= 2 && !group.every(person => isInSystemContacts(person))
|
2021-11-11 22:43:05 +00:00
|
|
|
) as unknown as GroupNameCollisionsWithConversationsByTitle;
|
2021-06-01 23:30:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns `true` if the user should see a group member name collision warning, and
|
|
|
|
* `false` otherwise. Users should see these warnings if any collisions appear that they
|
|
|
|
* haven't dismissed.
|
|
|
|
*/
|
|
|
|
export const hasUnacknowledgedCollisions = (
|
2023-01-13 20:07:26 +00:00
|
|
|
previous: ReadonlyDeep<GroupNameCollisionsWithIdsByTitle>,
|
|
|
|
current: ReadonlyDeep<GroupNameCollisionsWithIdsByTitle>
|
2021-06-01 23:30:25 +00:00
|
|
|
): boolean =>
|
|
|
|
Object.entries(current).some(([title, currentIds]) => {
|
|
|
|
const previousIds = new Set(getOwn(previous, title) || []);
|
|
|
|
return currentIds.some(currentId => !previousIds.has(currentId));
|
|
|
|
});
|
|
|
|
|
|
|
|
export const invertIdsByTitle = (
|
2023-01-13 20:07:26 +00:00
|
|
|
idsByTitle: ReadonlyDeep<GroupNameCollisionsWithIdsByTitle>
|
2021-06-01 23:30:25 +00:00
|
|
|
): GroupNameCollisionsWithTitlesById => {
|
|
|
|
const result: GroupNameCollisionsWithTitlesById = Object.create(null);
|
|
|
|
Object.entries(idsByTitle).forEach(([title, ids]) => {
|
|
|
|
ids.forEach(id => {
|
|
|
|
result[id] = title;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
return result;
|
|
|
|
};
|