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

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

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

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

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

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

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

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

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

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

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

@ -3,10 +3,10 @@ import { getInitials } from '../../util/getInitials';
interface Props { interface Props {
diameter: number; diameter: number;
phoneNumber: string; name: string;
pubkey?: string;
colors: Array<string>; colors: Array<string>;
borderColor: string; borderColor: string;
name?: string;
} }
interface State { interface State {
@ -23,32 +23,52 @@ export class AvatarPlaceHolder extends React.PureComponent<Props, State> {
} }
public componentDidMount() { public componentDidMount() {
void this.sha512(this.props.phoneNumber).then((sha512Seed: string) => { const { pubkey } = this.props;
this.setState({ sha512Seed }); if (pubkey) {
}); void this.sha512(pubkey).then((sha512Seed: string) => {
this.setState({ sha512Seed });
});
}
} }
public componentDidUpdate(prevProps: Props, prevState: State) { 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; 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() { public render() {
const { borderColor, colors, diameter, name } = this.props;
const viewBox = `0 0 ${diameter} ${diameter}`;
const r = diameter / 2;
if (!this.state.sha512Seed) { 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 initial = getInitials(name)?.toLocaleUpperCase() || '0';
const r = diameter / 2;
const initial =
getInitials(name)?.toLocaleUpperCase() ||
getInitials(phoneNumber)?.toLocaleUpperCase() ||
'0';
const viewBox = `0 0 ${diameter} ${diameter}`;
const fontSize = diameter * 0.5; const fontSize = diameter * 0.5;
// Generate the seed simulate the .hashCode as Java // Generate the seed simulate the .hashCode as Java

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

@ -2,9 +2,16 @@ import { GroupUtils } from '../../session/utils';
import { UserUtil } from '../../util'; import { UserUtil } from '../../util';
import { PubKey } from '../../session/types'; import { PubKey } from '../../session/types';
import React from 'react'; 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 = { type State = {
closedMemberConversations?: Array<ConversationAttributes>; memberAvatars?: Array<ConversationAvatar>; // this is added by usingClosedConversationDetails
}; };
export function usingClosedConversationDetails(WrappedComponent: any) { export function usingClosedConversationDetails(WrappedComponent: any) {
@ -12,7 +19,7 @@ export function usingClosedConversationDetails(WrappedComponent: any) {
constructor(props: any) { constructor(props: any) {
super(props); super(props);
this.state = { this.state = {
closedMemberConversations: undefined, memberAvatars: undefined,
}; };
} }
@ -27,7 +34,7 @@ export function usingClosedConversationDetails(WrappedComponent: any) {
public render() { public render() {
return ( return (
<WrappedComponent <WrappedComponent
closedMemberConversations={this.state.closedMemberConversations} memberAvatars={this.state.memberAvatars}
{...this.props} {...this.props}
/> />
); );
@ -42,21 +49,40 @@ export function usingClosedConversationDetails(WrappedComponent: any) {
phoneNumber, phoneNumber,
id, id,
} = this.props; } = this.props;
if ( if (
!isPublic && !isPublic &&
(conversationType === 'group' || type === 'group' || isGroup) (conversationType === 'group' || type === 'group' || isGroup)
) { ) {
const groupId = id || phoneNumber; const groupId = id || phoneNumber;
let members = await GroupUtils.getGroupMembers(PubKey.cast(groupId));
const ourPrimary = await UserUtil.getPrimary(); 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 = members.filter(m => m.key !== ourPrimary.key);
members.sort((a, b) => (a.key < b.key ? -1 : a.key > b.key ? 1 : 0)); members.sort((a, b) => (a.key < b.key ? -1 : a.key > b.key ? 1 : 0));
const membersConvos = members.map( if (ourself) {
m => window.ConversationController.get(m.key).cachedProps 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 const memberAvatars = memberConvos.map(m => {
membersConvos.slice(0, 2); return {
this.setState({ closedMemberConversations: membersConvos }); 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