Merge pull request #1341 from Bilb/avatar-always-show-two-members

pull/1343/head
Audric Ackermann 5 years ago committed by GitHub
commit ffab04c7f4
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

Binary file not shown.

Before

Width:  |  Height:  |  Size: 555 B

@ -73,7 +73,6 @@
'folder-outline.svg',
'forward.svg',
'gear.svg',
'group_default.png',
'hourglass_empty.svg',
'hourglass_full.svg',
'icon_1024.png',
@ -731,7 +730,6 @@
profileName: displayName,
pubkey: ourNumber,
avatarPath,
avatarColor: conversation.getColor(),
onOk: async (newName, avatar) => {
let newAvatarPath = '';
let url = null;
@ -1065,7 +1063,6 @@
profileName: displayName,
pubkey: userPubKey,
avatarPath,
avatarColor: conversation.getColor(),
isRss: conversation.isRss(),
onStartConversation: () => {
Whisper.events.trigger('showConversation', userPubKey);

@ -32,7 +32,6 @@
UNRESTRICTED: 3,
};
const { Util } = window.Signal;
const {
Conversation,
Contact,
@ -571,7 +570,6 @@
getProps() {
const { format } = PhoneNumber;
const regionCode = storage.get('regionCode');
const color = this.getColor();
const typingKeys = Object.keys(this.contactTypingTimers || {});
const result = {
@ -579,7 +577,6 @@
isArchived: this.get('isArchived'),
activeAt: this.get('active_at'),
avatarPath: this.getAvatarPath(),
color,
type: this.isPrivate() ? 'direct' : 'group',
isMe: this.isMe(),
isPublic: this.isPublic(),
@ -2526,14 +2523,6 @@
return this.get('type') === 'private';
},
getColor() {
if (!this.isPrivate()) {
return 'signal-blue';
}
const { migrateColor } = Util;
return migrateColor(this.get('color'));
},
getAvatarPath() {
const avatar = this.get('avatar') || this.get('profileAvatar');
@ -2549,19 +2538,17 @@
},
getAvatar() {
const title = this.get('name');
const color = this.getColor();
const url = this.getAvatarPath();
if (url) {
return { url, color };
return { url };
} else if (this.isPrivate()) {
const symbol = this.isValid() ? '#' : '!';
return {
color,
content: this.getInitials(title) || symbol,
};
}
return { url: 'images/group_default.png', color };
return { url: null };
},
getNotificationIcon() {

@ -414,7 +414,6 @@
const regionCode = storage.get('regionCode');
const contactModel = this.findContact(phoneNumber);
const color = contactModel ? contactModel.getColor() : null;
let profileName;
if (phoneNumber === window.storage.get('primaryDevicePubKey')) {
profileName = i18n('you');
@ -426,7 +425,7 @@
phoneNumber: format(phoneNumber, {
ourRegionCode: regionCode,
}),
color,
color: null,
avatarPath: contactModel ? contactModel.getAvatarPath() : null,
name: contactModel ? contactModel.getName() : null,
profileName,
@ -565,7 +564,6 @@
const contact = this.findAndFormatContact(phoneNumber);
const contactModel = this.findContact(phoneNumber);
const authorColor = contactModel ? contactModel.getColor() : null;
const authorAvatarPath = contactModel
? contactModel.getAvatarPath()
: null;
@ -599,7 +597,6 @@
serverTimestamp: this.get('serverTimestamp'),
status: this.getMessagePropStatus(),
contact: this.getPropsForEmbeddedContact(),
authorColor,
authorName: contact.name,
authorProfileName: contact.profileName,
authorPhoneNumber: contact.phoneNumber,
@ -764,7 +761,6 @@
const { author, id, referencedMessageNotFound } = quote;
const contact = author && ConversationController.get(author);
const authorColor = contact ? contact.getColor() : 'grey';
const authorPhoneNumber = format(author, {
ourRegionCode: regionCode,
@ -794,7 +790,6 @@
authorPhoneNumber,
authorProfileName,
authorName,
authorColor,
onClick,
referencedMessageNotFound,
};

@ -169,14 +169,14 @@ const sendViaOnion = async (srvPubKey, url, fetchOptions, options = {}) => {
);
}
} else {
// why is
// FIXME why is
// https://chat-dev.lokinet.org/loki/v1/channel/1/deletes?count=200&since_id=
// difference in response than all the other calls....
log.info(
`loki_app_dot_net:::sendViaOnion #${
options.requestNumber
} - got object response ${url.toString()}`
);
// log.info(
// `loki_app_dot_net:::sendViaOnion #${
// options.requestNumber
// } - got object response ${url.toString()}`
// );
}
// result.status has the http response code
if (!txtResponse) {
@ -1424,7 +1424,7 @@ class LokiPublicChannelAPI {
value: {
name: 'Your Public Chat',
description: 'Your public chat room',
avatar: 'images/group_default.png',
avatar: null,
},
},
];

@ -32,7 +32,6 @@
Component: window.Signal.Components.ContactListItem,
props: {
isMe,
color: this.model.getColor(),
avatarPath: this.model.getAvatarPath(),
phoneNumber: this.model.getNumber(),
name: this.model.getName(),

@ -165,7 +165,6 @@
name: this.model.getName(),
phoneNumber: this.model.getNumber(),
profileName: this.model.getProfileName(),
color: this.model.getColor(),
avatarPath: this.model.getAvatarPath(),
isVerified: this.model.isVerified(),
isMe: this.model.isMe(),
@ -279,7 +278,6 @@
name: this.model.getName(),
phoneNumber: this.model.getNumber(),
profileName: this.model.getProfileName(),
color: this.model.getColor(),
avatarPath: this.model.getAvatarPath(),
isGroup: !this.model.isPrivate(),
isPublic: this.model.isPublic(),

@ -46,6 +46,7 @@
props: {
titleText: this.titleText,
isPublic: this.isPublic,
pubkey: this.groupId,
groupName: this.groupName,
okText: i18n('ok'),
cancelText: i18n('cancel'),

@ -8,21 +8,13 @@
Whisper.EditProfileDialogView = Whisper.View.extend({
className: 'loki-dialog modal',
initialize({
profileName,
avatarPath,
avatarColor,
pubkey,
onOk,
callback,
}) {
initialize({ profileName, avatarPath, pubkey, onOk, callback }) {
this.close = this.close.bind(this);
this.callback = callback;
this.profileName = profileName;
this.pubkey = pubkey;
this.avatarPath = avatarPath;
this.avatarColor = avatarColor;
this.onOk = onOk;
this.$el.focus();

@ -11,7 +11,6 @@
initialize({
profileName,
avatarPath,
avatarColor,
pubkey,
isRss,
onOk,
@ -23,7 +22,6 @@
this.pubkey = pubkey;
this.isRss = isRss;
this.avatarPath = avatarPath;
this.avatarColor = avatarColor;
this.onOk = onOk;
this.onStartConversation = onStartConversation;

@ -133,7 +133,6 @@ describe('Conversation', () => {
const convo = new Whisper.ConversationCollection().add(attributes);
const avatar = convo.getAvatar();
assert.property(avatar, 'content');
assert.property(avatar, 'color');
});
describe('when set to private', () => {

@ -1,23 +1,15 @@
import React from 'react';
import classNames from 'classnames';
import { getInitials } from '../util/getInitials';
import { LocalizerType } from '../types/Util';
import { AvatarPlaceHolder, ClosedGroupAvatar } from './AvatarPlaceHolder';
import { ConversationAttributes } from '../../js/models/conversations';
import { ConversationAvatar } from './session/usingClosedConversationDetails';
interface Props {
avatarPath?: string;
color?: string;
conversationType: 'group' | 'direct';
isPublic?: boolean;
noteToSelf?: boolean;
name?: string;
phoneNumber?: string;
profileName?: string;
name?: string; // display name, profileName or phoneNumber, whatever is set first
pubkey?: string;
size: number;
closedMemberConversations?: Array<ConversationAttributes>;
i18n?: LocalizerType;
memberAvatars?: Array<ConversationAvatar>; // this is added by usingClosedConversationDetails
onAvatarClick?: () => void;
}
@ -50,19 +42,15 @@ export class Avatar extends React.PureComponent<Props, State> {
}
public renderIdenticon() {
const { phoneNumber, size, name, profileName } = this.props;
const { size, name, pubkey } = this.props;
if (!phoneNumber) {
window.log.error('Empty phoneNumber for identicon');
return <></>;
}
const userName = name || '0';
const userName = profileName || name;
return (
<AvatarPlaceHolder
phoneNumber={phoneNumber}
diameter={size}
name={userName}
pubkey={pubkey}
colors={this.getAvatarColors()}
borderColor={this.getAvatarBorderColor()}
/>
@ -70,61 +58,42 @@ export class Avatar extends React.PureComponent<Props, State> {
}
public renderImage() {
const { avatarPath, name, phoneNumber, profileName } = this.props;
const { avatarPath, name } = this.props;
const { imageBroken } = this.state;
if (!avatarPath || imageBroken) {
return null;
}
const title = `${name || phoneNumber}${
!name && profileName ? ` ~${profileName}` : ''
}`;
return (
<img
onError={this.handleImageErrorBound}
alt={window.i18n('contactAvatarAlt', [title])}
alt={window.i18n('contactAvatarAlt', [name])}
src={avatarPath}
/>
);
}
public renderNoImage() {
const {
conversationType,
closedMemberConversations,
isPublic,
size,
i18n,
} = this.props;
const isGroup = conversationType === 'group';
if (isGroup && !isPublic && closedMemberConversations) {
const forcedI18n = i18n || window.i18n;
const { memberAvatars, size } = this.props;
// if no image but we have conversations set for the group, renders group members avatars
if (memberAvatars) {
return (
<ClosedGroupAvatar
size={size}
conversations={closedMemberConversations}
i18n={forcedI18n}
memberAvatars={memberAvatars}
i18n={window.i18n}
/>
);
}
console.warn(
'renderNoImage should not happen with something else than a closed group'
);
return <div className={classNames('module-avatar__icon')} />;
return this.renderIdenticon();
}
public render() {
const { avatarPath, color, size, conversationType } = this.props;
const { avatarPath, size } = this.props;
const { imageBroken } = this.state;
// If it's a direct conversation then we must have an identicon
const hasAvatar = avatarPath || conversationType === 'direct';
const hasImage = hasAvatar && !imageBroken;
const hasImage = avatarPath && !imageBroken;
if (
size !== 28 &&
@ -142,15 +111,14 @@ export class Avatar extends React.PureComponent<Props, State> {
className={classNames(
'module-avatar',
`module-avatar--${size}`,
hasImage ? 'module-avatar--with-image' : 'module-avatar--no-image',
!hasImage ? `module-avatar--${color}` : null
hasImage ? 'module-avatar--with-image' : 'module-avatar--no-image'
)}
onClick={e => {
this.onAvatarClickBound(e);
}}
role="button"
>
{hasImage ? this.renderAvatarOrIdenticon() : this.renderNoImage()}
{hasImage ? this.renderImage() : this.renderNoImage()}
</div>
);
}
@ -162,17 +130,6 @@ export class Avatar extends React.PureComponent<Props, State> {
}
}
private renderAvatarOrIdenticon() {
const { avatarPath, conversationType } = this.props;
// If it's a direct conversation then we must have an identicon
const hasAvatar = avatarPath || conversationType === 'direct';
return hasAvatar && avatarPath
? this.renderImage()
: this.renderIdenticon();
}
private getAvatarColors(): Array<string> {
// const theme = window.Events.getThemedSettings();
// defined in session-android as `profile_picture_placeholder_colors`

@ -3,10 +3,10 @@ import { getInitials } from '../../util/getInitials';
interface Props {
diameter: number;
phoneNumber: string;
name: string;
pubkey?: string;
colors: Array<string>;
borderColor: string;
name?: string;
}
interface State {
@ -23,32 +23,52 @@ export class AvatarPlaceHolder extends React.PureComponent<Props, State> {
}
public componentDidMount() {
void this.sha512(this.props.phoneNumber).then((sha512Seed: string) => {
this.setState({ sha512Seed });
});
const { pubkey } = this.props;
if (pubkey) {
void this.sha512(pubkey).then((sha512Seed: string) => {
this.setState({ sha512Seed });
});
}
}
public componentDidUpdate(prevProps: Props, prevState: State) {
if (this.props.phoneNumber === prevProps.phoneNumber) {
const { pubkey, name } = this.props;
if (pubkey === prevProps.pubkey && name === prevProps.name) {
return;
}
void this.sha512(this.props.phoneNumber).then((sha512Seed: string) => {
this.setState({ sha512Seed });
});
if (pubkey) {
void this.sha512(pubkey).then((sha512Seed: string) => {
this.setState({ sha512Seed });
});
}
}
public render() {
const { borderColor, colors, diameter, name } = this.props;
const viewBox = `0 0 ${diameter} ${diameter}`;
const r = diameter / 2;
if (!this.state.sha512Seed) {
return <></>;
// return grey circle
return (
<svg viewBox={viewBox}>
<g id="UrTavla">
<circle
cx={r}
cy={r}
r={r}
fill="#d2d2d3"
shape-rendering="geometricPrecision"
stroke={borderColor}
stroke-width="1"
/>
</g>
</svg>
);
}
const { borderColor, colors, diameter, phoneNumber, name } = this.props;
const r = diameter / 2;
const initial =
getInitials(name)?.toLocaleUpperCase() ||
getInitials(phoneNumber)?.toLocaleUpperCase() ||
'0';
const viewBox = `0 0 ${diameter} ${diameter}`;
const initial = getInitials(name)?.toLocaleUpperCase() || '0';
const fontSize = diameter * 0.5;
// Generate the seed simulate the .hashCode as Java

@ -1,94 +1,60 @@
import React from 'react';
import { Avatar } from '../Avatar';
import { LocalizerType } from '../../types/Util';
import { ConversationAttributes } from '../../../js/models/conversations';
import { ConversationAvatar } from '../session/usingClosedConversationDetails';
interface Props {
size: number;
conversations: Array<ConversationAttributes>;
memberAvatars: Array<ConversationAvatar>; // this is added by usingClosedConversationDetails
i18n: LocalizerType;
}
export class ClosedGroupAvatar extends React.PureComponent<Props> {
public getClosedGroupAvatarsSize(size: number) {
// Always use the size directly under the one requested
switch (size) {
case 36:
return 28;
case 48:
return 36;
case 64:
return 48;
case 80:
return 64;
case 300:
return 80;
default:
throw new Error(
`Invalid size request for closed group avatar: ${size}`
);
}
}
public render() {
const { conversations, size, i18n } = this.props;
const { memberAvatars, size } = this.props;
const avatarsDiameter = this.getClosedGroupAvatarsSize(size);
const conv1 = memberAvatars.length > 0 ? memberAvatars[0] : undefined;
const conv2 = memberAvatars.length > 1 ? memberAvatars[1] : undefined;
const name1 = conv1?.name || conv1?.id || undefined;
const name2 = conv2?.name || conv2?.id || undefined;
if (conversations.length === 1) {
const conv = conversations[0];
return (
// use the 2 first members as group avatars
return (
<div className="module-avatar__icon-closed">
<Avatar
avatarPath={conv.avatarPath}
noteToSelf={conv.isMe}
conversationType="direct"
i18n={i18n}
name={name}
phoneNumber={conv.id}
profileName={conv.name}
size={size}
isPublic={false}
avatarPath={conv1?.avatarPath}
name={name1}
size={avatarsDiameter}
pubkey={conv1?.id}
/>
);
} else if (conversations.length > 1) {
// in a closed group avatar, each visible avatar member size is 2/3 of the group avatar in size
// Always use the size directly under the one requested
let avatarsDiameter = 0;
switch (size) {
case 36: {
avatarsDiameter = 28;
break;
}
case 48: {
avatarsDiameter = 36;
break;
}
case 64: {
avatarsDiameter = 48;
break;
}
case 80: {
avatarsDiameter = 64;
break;
}
case 300: {
avatarsDiameter = 80;
break;
}
default:
throw new Error(
`Invalid size request for closed group avatar: ${size}`
);
}
const conv1 = conversations[0];
const conv2 = conversations[1];
// use the 2 first members as group avatars
return (
<div className="module-avatar__icon-closed">
<Avatar
avatarPath={conv1.avatarPath}
noteToSelf={conv1.isMe}
conversationType="direct"
i18n={i18n}
name={name}
phoneNumber={conv1.id}
profileName={conv1.name}
size={avatarsDiameter}
isPublic={false}
/>
<Avatar
avatarPath={conv2.avatarPath}
noteToSelf={conv2.isMe}
conversationType="direct"
i18n={i18n}
name={name}
phoneNumber={conv2.id}
profileName={conv2.name}
size={avatarsDiameter}
isPublic={false}
/>
</div>
);
} else {
return <></>;
}
<Avatar
avatarPath={conv2?.avatarPath}
name={name2}
size={avatarsDiameter}
pubkey={conv2?.id}
/>
</div>
);
}
}

@ -10,7 +10,6 @@ interface Props {
phoneNumber: string;
isMe?: boolean;
name?: string;
color: string;
verified: boolean;
profileName?: string;
avatarPath?: string;
@ -20,25 +19,16 @@ interface Props {
export class ContactListItem extends React.Component<Props> {
public renderAvatar() {
const {
avatarPath,
i18n,
color,
name,
phoneNumber,
profileName,
} = this.props;
const { avatarPath, name, phoneNumber, profileName } = this.props;
const userName = name || profileName || phoneNumber;
return (
<Avatar
avatarPath={avatarPath}
color={color}
conversationType="direct"
i18n={i18n}
name={name}
phoneNumber={phoneNumber}
profileName={profileName}
name={userName}
size={36}
pubkey={phoneNumber}
/>
);
}

@ -21,7 +21,10 @@ import {
getLeaveGroupMenuItem,
} from '../session/utils/Menu';
import { usingClosedConversationDetails } from './session/usingClosedConversationDetails';
import {
ConversationAvatar,
usingClosedConversationDetails,
} from './session/usingClosedConversationDetails';
export type PropsData = {
id: string;
@ -55,7 +58,7 @@ export type PropsData = {
isSecondary?: boolean;
isGroupInvitation?: boolean;
isKickedFromGroup?: boolean;
closedMemberConversations?: any; // this is added by usingClosedConversationDetails
memberAvatars?: Array<ConversationAvatar>; // this is added by usingClosedConversationDetails
};
type PropsHousekeeping = {
@ -82,32 +85,24 @@ class ConversationListItem extends React.PureComponent<Props> {
public renderAvatar() {
const {
avatarPath,
color,
type,
i18n,
isMe,
name,
phoneNumber,
profileName,
isPublic,
memberAvatars,
} = this.props;
const iconSize = 36;
const userName = name || profileName || phoneNumber;
return (
<div className="module-conversation-list-item__avatar-container">
<Avatar
avatarPath={avatarPath}
color={color}
noteToSelf={isMe}
conversationType={type}
i18n={i18n}
name={name}
phoneNumber={phoneNumber}
profileName={profileName}
name={userName}
size={iconSize}
isPublic={isPublic}
closedMemberConversations={this.props.closedMemberConversations}
memberAvatars={memberAvatars}
pubkey={phoneNumber}
/>
</div>
);

@ -30,7 +30,6 @@ interface Props {
i18n: any;
profileName: string;
avatarPath: string;
avatarColor: string;
pubkey: string;
onClose: any;
onOk: any;
@ -257,20 +256,12 @@ export class EditProfileDialog extends React.Component<Props, State> {
}
private renderAvatar() {
const avatarPath = this.state.avatar;
const color = this.props.avatarColor;
const { avatar, profileName } = this.state;
const { pubkey } = this.props;
const userName = name || profileName || pubkey;
return (
<Avatar
avatarPath={avatarPath}
color={color}
conversationType="direct"
i18n={this.props.i18n}
name={this.state.profileName}
phoneNumber={this.props.pubkey}
profileName={this.state.profileName}
size={80}
/>
<Avatar avatarPath={avatar} name={userName} size={80} pubkey={pubkey} />
);
}

@ -100,20 +100,15 @@ export class MessageSearchResult extends React.PureComponent<Props> {
}
public renderAvatar() {
const { from, i18n, to } = this.props;
const isNoteToSelf = from.isMe && to.isMe;
const { from } = this.props;
const userName = from.phoneNumber || from.profileName;
return (
<Avatar
avatarPath={from.avatarPath}
color={from.color}
conversationType="direct"
i18n={i18n}
name={name}
noteToSelf={isNoteToSelf}
phoneNumber={from.phoneNumber}
profileName={from.profileName}
name={userName}
size={36}
pubkey={from.phoneNumber}
/>
);
}

@ -14,7 +14,6 @@ interface Props {
isRss: boolean;
profileName: string;
avatarPath: string;
avatarColor: string;
pubkey: string;
onClose: any;
onStartConversation: any;
@ -64,21 +63,17 @@ export class UserDetailsDialog extends React.Component<Props, State> {
}
private renderAvatar() {
const avatarPath = this.props.avatarPath;
const color = this.props.avatarColor;
const { avatarPath, pubkey, profileName } = this.props;
const size = this.state.isEnlargedImageShown ? 300 : 80;
const userName = name || profileName || pubkey;
return (
<Avatar
avatarPath={avatarPath}
color={color}
conversationType="direct"
i18n={this.props.i18n}
name={this.props.profileName}
phoneNumber={this.props.pubkey}
profileName={this.props.profileName}
name={userName}
size={size}
onAvatarClick={this.handleShowEnlargedDialog}
pubkey={pubkey}
/>
);
}

@ -16,7 +16,10 @@ import {
SessionButtonType,
} from '../session/SessionButton';
import * as Menu from '../../session/utils/Menu';
import { usingClosedConversationDetails } from '../session/usingClosedConversationDetails';
import {
ConversationAvatar,
usingClosedConversationDetails,
} from '../session/usingClosedConversationDetails';
export interface TimerOption {
name: string;
@ -92,7 +95,7 @@ interface Props {
onUpdateGroupName: () => void;
i18n: LocalizerType;
closedMemberConversations?: any; // this is added by usingClosedConversationDetails
memberAvatars?: Array<ConversationAvatar>; // this is added by usingClosedConversationDetails
}
class ConversationHeader extends React.Component<Props> {
@ -198,34 +201,25 @@ class ConversationHeader extends React.Component<Props> {
public renderAvatar() {
const {
avatarPath,
closedMemberConversations,
i18n,
isGroup,
isMe,
memberAvatars,
name,
phoneNumber,
profileName,
isPublic,
} = this.props;
const conversationType = isGroup ? 'group' : 'direct';
const userName = name || profileName || phoneNumber;
return (
<span className="module-conversation-header__avatar">
<Avatar
avatarPath={avatarPath}
conversationType={conversationType}
i18n={i18n}
noteToSelf={isMe}
name={name}
phoneNumber={phoneNumber}
profileName={profileName}
name={userName}
size={36}
onAvatarClick={() => {
this.onAvatarClickBound(phoneNumber);
}}
isPublic={isPublic}
closedMemberConversations={closedMemberConversations}
memberAvatars={memberAvatars}
pubkey={phoneNumber}
/>
</span>
);

@ -50,7 +50,6 @@ export class CreateGroupDialog extends React.Component<Props, State> {
authorProfileName: name,
selected: false,
authorName: name, // different from ProfileName?
authorColor: d.getColor(),
checkmarked: false,
};
});

@ -45,7 +45,6 @@ export class InviteContactsDialog extends React.Component<Props, State> {
authorAvatarPath: d?.cachedProps?.avatarPath,
selected: false,
authorName: name,
authorColor: d.getColor(),
checkmarked: false,
existingMember,
};

@ -8,7 +8,6 @@ export interface Contact {
authorProfileName: string;
authorPhoneNumber: string;
authorName: string;
authorColor: any;
authorAvatarPath: string;
checkmarked: boolean;
existingMember: boolean;
@ -91,17 +90,20 @@ class MemberItem extends React.Component<MemberItemProps> {
}
private renderAvatar() {
const {
authorName,
authorAvatarPath,
authorPhoneNumber,
authorProfileName,
} = this.props.member;
const userName = authorName || authorProfileName || authorPhoneNumber;
return (
<Avatar
avatarPath={this.props.member.authorAvatarPath}
color={this.props.member.authorColor}
conversationType="direct"
i18n={this.props.i18n}
name={this.props.member.authorName}
phoneNumber={this.props.member.authorPhoneNumber}
profileName={this.props.member.authorProfileName}
avatarPath={authorAvatarPath}
name={userName}
size={28}
isPublic={false}
pubkey={authorPhoneNumber}
/>
);
}

@ -78,7 +78,6 @@ export interface Props {
authorProfileName?: string;
/** Note: this should be formatted for display */
authorPhoneNumber: string;
authorColor?: ColorType;
conversationType: 'group' | 'direct';
attachments?: Array<AttachmentType>;
quote?: {
@ -88,7 +87,6 @@ export interface Props {
authorPhoneNumber: string;
authorProfileName?: string;
authorName?: string;
authorColor?: ColorType;
onClick?: () => void;
referencedMessageNotFound: boolean;
};
@ -585,7 +583,6 @@ export class Message extends React.PureComponent<Props, State> {
public renderQuote() {
const {
conversationType,
authorColor,
direction,
i18n,
quote,
@ -599,8 +596,6 @@ export class Message extends React.PureComponent<Props, State> {
const withContentAbove =
conversationType === 'group' && direction === 'incoming';
const quoteColor =
direction === 'incoming' ? authorColor : quote.authorColor;
const shortenedPubkey = window.shortenPubkey(quote.authorPhoneNumber);
@ -621,7 +616,6 @@ export class Message extends React.PureComponent<Props, State> {
authorPhoneNumber={displayedPubkey}
authorProfileName={quote.authorProfileName}
authorName={quote.authorName}
authorColor={quoteColor}
referencedMessageNotFound={quote.referencedMessageNotFound}
isFromMe={quote.isFromMe}
withContentAbove={withContentAbove}
@ -668,7 +662,6 @@ export class Message extends React.PureComponent<Props, State> {
authorProfileName,
collapseMetadata,
senderIsModerator,
authorColor,
conversationType,
direction,
i18n,
@ -682,21 +675,18 @@ export class Message extends React.PureComponent<Props, State> {
) {
return;
}
const userName = authorName || authorProfileName || authorPhoneNumber;
return (
<div className="module-message__author-avatar">
<Avatar
avatarPath={authorAvatarPath}
color={authorColor}
conversationType="direct"
i18n={i18n}
name={authorName}
phoneNumber={authorPhoneNumber}
profileName={authorProfileName}
name={userName}
size={36}
onAvatarClick={() => {
onShowUserDetails(authorPhoneNumber);
}}
pubkey={authorPhoneNumber}
/>
{senderIsModerator && (
<div className="module-avatar__icon--crown-wrapper">
@ -1055,7 +1045,6 @@ export class Message extends React.PureComponent<Props, State> {
public render() {
const {
authorPhoneNumber,
authorColor,
direction,
id,
isKickedFromGroup,

@ -37,18 +37,15 @@ interface Props {
export class MessageDetail extends React.Component<Props> {
public renderAvatar(contact: Contact) {
const { i18n } = this.props;
const { avatarPath, color, phoneNumber, name, profileName } = contact;
const { avatarPath, phoneNumber, name, profileName } = contact;
const userName = name || profileName || phoneNumber;
return (
<Avatar
avatarPath={avatarPath}
color={color}
conversationType="direct"
i18n={i18n}
name={name}
phoneNumber={phoneNumber}
profileName={profileName}
name={userName}
size={36}
pubkey={phoneNumber}
/>
);
}

@ -51,7 +51,6 @@ export class AddModeratorsDialog extends React.Component<Props, State> {
authorProfileName: name,
selected: false,
authorName: name,
authorColor: d.getColor(),
checkmarked: false,
existingMember,
};
@ -97,7 +96,6 @@ export class AddModeratorsDialog extends React.Component<Props, State> {
authorAvatarPath: '',
selected: true,
authorName: this.state.inputBoxValue,
authorColor: '#000000',
checkmarked: true,
existingMember: false,
});

@ -34,7 +34,6 @@ export class RemoveModeratorsDialog extends React.Component<Props, State> {
const lokiProfile = d.getLokiProfile();
name = lokiProfile ? lokiProfile.displayName : 'Anonymous';
}
const authorColor = d.getColor ? d.getColor() : '#000000';
// TODO: should take existing members into account
const existingMember = false;
@ -44,7 +43,6 @@ export class RemoveModeratorsDialog extends React.Component<Props, State> {
authorProfileName: name,
selected: false,
authorName: name,
authorColor,
checkmarked: true,
existingMember,
};

@ -15,7 +15,6 @@ interface Props {
authorPhoneNumber: string;
authorProfileName?: string;
authorName?: string;
authorColor?: ColorType;
i18n: LocalizerType;
isFromMe: boolean;
isIncoming: boolean;
@ -365,7 +364,6 @@ export class Quote extends React.Component<Props, State> {
public render() {
const {
authorColor,
isIncoming,
onClick,
referencedMessageNotFound,

@ -20,29 +20,24 @@ export class TypingBubble extends React.Component<Props> {
public renderAvatar() {
const {
avatarPath,
color,
name,
phoneNumber,
profileName,
conversationType,
i18n,
} = this.props;
if (conversationType !== 'group') {
return;
}
const userName = name || profileName || phoneNumber;
return (
<div className="module-message__author-avatar">
<Avatar
avatarPath={avatarPath}
color={color}
conversationType="direct"
i18n={i18n}
name={name}
phoneNumber={phoneNumber}
profileName={profileName}
name={userName}
size={36}
pubkey={phoneNumber}
/>
</div>
);

@ -52,7 +52,6 @@ export class UpdateGroupMembersDialog extends React.Component<Props, State> {
authorAvatarPath: d?.cachedProps?.avatarPath,
selected: false,
authorName: name, // different from ProfileName?
authorColor: d.getColor(),
checkmarked: false,
existingMember,
};

@ -7,6 +7,7 @@ import { Avatar } from '../Avatar';
interface Props {
titleText: string;
pubkey: string;
isPublic: boolean;
groupName: string;
okText: string;
@ -176,10 +177,8 @@ export class UpdateGroupNameDialog extends React.Component<Props, State> {
<div className="avatar-center-inner">
<Avatar
avatarPath={avatarPath}
conversationType="group"
i18n={this.props.i18n}
size={80}
isPublic={isPublic}
pubkey={this.props.pubkey}
/>
<div
className="image-upload-section"

@ -35,16 +35,9 @@ export function renderAvatar({
);
}
const pubkey = contact.name?.givenName || '0';
return (
<Avatar
avatarPath={avatarPath}
color="grey"
conversationType="direct"
i18n={i18n}
name={name}
size={size}
isPublic={false}
/>
<Avatar avatarPath={avatarPath} name={name} size={size} pubkey={pubkey} />
);
}

@ -122,7 +122,6 @@ export class ActionsPanel extends React.Component<Props, State> {
onSelect?: (event: SectionType) => void;
type: SectionType;
avatarPath?: string;
avatarColor?: string;
notificationCount?: number;
}) => {
const handleClick = onSelect
@ -140,16 +139,15 @@ export class ActionsPanel extends React.Component<Props, State> {
: undefined;
if (type === SectionType.Profile) {
const pubkey = window.storage.get('primaryDevicePubKey');
const userName = window.getOurDisplayName() || pubkey;
return (
<Avatar
avatarPath={avatarPath}
conversationType="direct"
i18n={window.i18n}
// tslint:disable-next-line: no-backbone-get-set-outside-model
phoneNumber={window.storage.get('primaryDevicePubKey')}
size={28}
onAvatarClick={handleClick}
profileName={window.getOurDisplayName()}
name={userName}
pubkey={pubkey}
/>
);
}

@ -95,7 +95,6 @@ export class SessionClosableOverlay extends React.Component<Props, State> {
authorAvatarPath: d.avatarPath,
selected: false,
authorName: name,
authorColor: d.color,
checkmarked: false,
existingMember,
};

@ -10,7 +10,10 @@ import { SessionDropdown } from './SessionDropdown';
import { MediaGallery } from '../conversation/media-gallery/MediaGallery';
import _ from 'lodash';
import { TimerOption } from '../conversation/ConversationHeader';
import { usingClosedConversationDetails } from '../session/usingClosedConversationDetails';
import {
ConversationAvatar,
usingClosedConversationDetails,
} from '../session/usingClosedConversationDetails';
interface Props {
id: string;
@ -24,7 +27,7 @@ interface Props {
amMod: boolean;
isKickedFromGroup: boolean;
isBlocked: boolean;
closedMemberConversations?: any; // this is added by usingClosedConversationDetails
memberAvatars?: Array<ConversationAvatar>; // this is added by usingClosedConversationDetails
onGoBack: () => void;
onInviteContacts: () => void;
@ -311,7 +314,7 @@ class SessionGroupSettings extends React.Component<Props, any> {
private renderHeader() {
const {
closedMemberConversations,
memberAvatars,
id,
onGoBack,
onInviteContacts,
@ -325,6 +328,7 @@ class SessionGroupSettings extends React.Component<Props, any> {
const showInviteContacts =
(isPublic || isAdmin) && !isKickedFromGroup && !isBlocked;
const userName = id;
return (
<div className="group-settings-header">
<SessionIconButton
@ -335,11 +339,10 @@ class SessionGroupSettings extends React.Component<Props, any> {
/>
<Avatar
avatarPath={avatarPath}
phoneNumber={id}
conversationType="group"
name={userName}
size={80}
isPublic={isPublic}
closedMemberConversations={closedMemberConversations}
memberAvatars={memberAvatars}
pubkey={id}
/>
<div className="invite-friends-container">
{showInviteContacts && (

@ -10,7 +10,6 @@ export interface ContactType {
authorProfileName: string;
authorPhoneNumber: string;
authorName: string;
authorColor: any;
authorAvatarPath: string;
checkmarked: boolean;
existingMember: boolean;
@ -86,16 +85,19 @@ export class SessionMemberListItem extends React.Component<Props, State> {
}
private renderAvatar() {
const {
authorAvatarPath,
authorName,
authorPhoneNumber,
authorProfileName,
} = this.props.member;
const userName = authorName || authorProfileName || authorPhoneNumber;
return (
<Avatar
avatarPath={this.props.member.authorAvatarPath}
color={this.props.member.authorColor}
conversationType="direct"
i18n={window.i18n}
name={this.props.member.authorName}
phoneNumber={this.props.member.authorPhoneNumber}
profileName={this.props.member.authorProfileName}
avatarPath={authorAvatarPath}
name={userName}
size={28}
pubkey={authorPhoneNumber}
/>
);
}

@ -2,9 +2,16 @@ import { GroupUtils } from '../../session/utils';
import { UserUtil } from '../../util';
import { PubKey } from '../../session/types';
import React from 'react';
import { ConversationAttributes } from '../../../js/models/conversations';
import * as _ from 'lodash';
export type ConversationAvatar = {
avatarPath?: string;
id?: string; // member's pubkey
name?: string;
};
type State = {
closedMemberConversations?: Array<ConversationAttributes>;
memberAvatars?: Array<ConversationAvatar>; // this is added by usingClosedConversationDetails
};
export function usingClosedConversationDetails(WrappedComponent: any) {
@ -12,7 +19,7 @@ export function usingClosedConversationDetails(WrappedComponent: any) {
constructor(props: any) {
super(props);
this.state = {
closedMemberConversations: undefined,
memberAvatars: undefined,
};
}
@ -27,7 +34,7 @@ export function usingClosedConversationDetails(WrappedComponent: any) {
public render() {
return (
<WrappedComponent
closedMemberConversations={this.state.closedMemberConversations}
memberAvatars={this.state.memberAvatars}
{...this.props}
/>
);
@ -42,21 +49,40 @@ export function usingClosedConversationDetails(WrappedComponent: any) {
phoneNumber,
id,
} = this.props;
if (
!isPublic &&
(conversationType === 'group' || type === 'group' || isGroup)
) {
const groupId = id || phoneNumber;
let members = await GroupUtils.getGroupMembers(PubKey.cast(groupId));
const ourPrimary = await UserUtil.getPrimary();
let members = await GroupUtils.getGroupMembers(PubKey.cast(groupId));
const ourself = members.find(m => m.key !== ourPrimary.key);
// add ourself back at the back, so it's shown only if only 1 member and we are still a member
members = members.filter(m => m.key !== ourPrimary.key);
members.sort((a, b) => (a.key < b.key ? -1 : a.key > b.key ? 1 : 0));
const membersConvos = members.map(
m => window.ConversationController.get(m.key).cachedProps
if (ourself) {
members.push(ourPrimary);
}
// no need to forward more than 2 conversations for rendering the group avatar
members.slice(0, 2);
const memberConvos = await Promise.all(
members.map(async m =>
window.ConversationController.getOrCreateAndWait(m.key, 'private')
)
);
// no need to forward more than 2 conversation for rendering the group avatar
membersConvos.slice(0, 2);
this.setState({ closedMemberConversations: membersConvos });
const memberAvatars = memberConvos.map(m => {
return {
avatarPath: m.getAvatar()?.url || null,
id: m.id,
name: m.get('name') || m.get('profileName') || m.id,
};
});
if (!_.isEqual(memberAvatars, this.state.memberAvatars)) {
this.setState({ memberAvatars });
}
}
}
};

Loading…
Cancel
Save