Merge pull request #3113 from oxen-io/unstable

Unstable to clearnet before merge of onboarding to unstable
pull/3118/head
Audric Ackermann 10 months ago committed by GitHub
commit 7ad91ca79b
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

@ -5,6 +5,8 @@
<!-- Mac distribution --> <!-- Mac distribution -->
<key>com.apple.security.cs.allow-unsigned-executable-memory</key> <key>com.apple.security.cs.allow-unsigned-executable-memory</key>
<true/> <true/>
<key>com.apple.security.cs.allow-jit</key>
<true/>
<key>com.apple.security.cs.disable-library-validation</key> <key>com.apple.security.cs.disable-library-validation</key>
<true/> <true/>
<key>com.apple.security.device.audio-input</key> <key>com.apple.security.device.audio-input</key>

@ -5,6 +5,10 @@
<!-- Mac app store --> <!-- Mac app store -->
<key>com.apple.security.app-sandbox</key> <key>com.apple.security.app-sandbox</key>
<true/> <true/>
<key>com.apple.security.cs.allow-unsigned-executable-memory</key>
<true/>
<key>com.apple.security.cs.allow-jit</key>
<true/>
<key>com.apple.security.network.client</key> <key>com.apple.security.network.client</key>
<true/> <true/>
<key>com.apple.security.files.user-selected.read-only</key> <key>com.apple.security.files.user-selected.read-only</key>

@ -29,7 +29,7 @@ exports.default = async function notarizing(context) {
} }
const options = { const options = {
appBundleId: 'org.getsession.desktop', appBundleId: 'com.loki-project.messenger-desktop',
appPath: `${appOutDir}/${appName}.app`, appPath: `${appOutDir}/${appName}.app`,
appleId: SIGNING_APPLE_ID, appleId: SIGNING_APPLE_ID,
appleIdPassword: SIGNING_APP_PASSWORD, appleIdPassword: SIGNING_APP_PASSWORD,

@ -20,11 +20,6 @@
.module-contact-name__profile-number.italic { .module-contact-name__profile-number.italic {
font-style: italic; font-style: italic;
} }
.module-contact-name.compact {
display: block;
}
// Module: Message // Module: Message
.module-message__error-container { .module-message__error-container {

@ -13,12 +13,11 @@ type Props = {
profileName?: string | null; profileName?: string | null;
module?: string; module?: string;
boldProfileName?: boolean; boldProfileName?: boolean;
compact?: boolean;
shouldShowPubkey: boolean; shouldShowPubkey: boolean;
}; };
export const ContactName = (props: Props) => { export const ContactName = (props: Props) => {
const { pubkey, name, profileName, module, boldProfileName, compact, shouldShowPubkey } = props; const { pubkey, name, profileName, module, boldProfileName, shouldShowPubkey } = props;
const prefix = module || 'module-contact-name'; const prefix = module || 'module-contact-name';
const convoName = useNicknameOrProfileNameOrShortenedPubkey(pubkey); const convoName = useNicknameOrProfileNameOrShortenedPubkey(pubkey);
@ -43,7 +42,7 @@ export const ContactName = (props: Props) => {
return ( return (
<span <span
className={classNames(prefix, compact && 'compact')} className={classNames(prefix)}
dir="auto" dir="auto"
data-testid={`${prefix}__profile-name`} data-testid={`${prefix}__profile-name`}
style={{ style={{

@ -1,6 +1,6 @@
import classNames from 'classnames'; import classNames from 'classnames';
import React, { useState } from 'react';
import { noop } from 'lodash'; import { noop } from 'lodash';
import React, { useState } from 'react';
import * as MIME from '../../../../types/MIME'; import * as MIME from '../../../../types/MIME';
import * as GoogleChrome from '../../../../util/GoogleChrome'; import * as GoogleChrome from '../../../../util/GoogleChrome';
@ -291,7 +291,6 @@ const QuoteAuthor = (props: QuoteAuthorProps) => {
pubkey={PubKey.shorten(author)} pubkey={PubKey.shorten(author)}
name={authorName} name={authorName}
profileName={authorProfileName} profileName={authorProfileName}
compact={true}
shouldShowPubkey={Boolean(props.showPubkeyForAuthor)} shouldShowPubkey={Boolean(props.showPubkeyForAuthor)}
/> />
)} )}

@ -41,7 +41,6 @@ export const QuoteAuthor = (props: QuoteAuthorProps) => {
<ContactName <ContactName
pubkey={PubKey.shorten(author)} pubkey={PubKey.shorten(author)}
name={authorName} name={authorName}
compact={true}
shouldShowPubkey={Boolean(authorName && !isMe && isPublic)} shouldShowPubkey={Boolean(authorName && !isMe && isPublic)}
/> />
</StyledQuoteAuthor> </StyledQuoteAuthor>

@ -14,6 +14,7 @@ import {
} from '../../state/ducks/modalDialog'; } from '../../state/ducks/modalDialog';
import { import {
useSelectedIsPublic, useSelectedIsPublic,
useSelectedWeAreAdmin,
useSelectedWeAreModerator, useSelectedWeAreModerator,
} from '../../state/selectors/selectedConversation'; } from '../../state/selectors/selectedConversation';
import { SortedReactionList } from '../../types/Reaction'; import { SortedReactionList } from '../../types/Reaction';
@ -50,6 +51,11 @@ const StyledSendersContainer = styled(Flex)`
padding: 0 16px 16px; padding: 0 16px 16px;
`; `;
const StyledContactContainer = styled.span`
text-overflow: ellipsis;
overflow: hidden;
`;
const StyledReactionBar = styled(Flex)` const StyledReactionBar = styled(Flex)`
width: 100%; width: 100%;
margin: 12px 0 20px 4px; margin: 12px 0 20px 4px;
@ -132,7 +138,7 @@ const ReactionSenders = (props: ReactionSendersProps) => {
justifyContent={'space-between'} justifyContent={'space-between'}
alignItems={'center'} alignItems={'center'}
> >
<Flex container={true} alignItems={'center'}> <Flex container={true} alignItems={'center'} style={{ overflow: 'hidden' }}>
<Avatar <Avatar
size={AvatarSize.XS} size={AvatarSize.XS}
pubkey={sender} pubkey={sender}
@ -143,11 +149,13 @@ const ReactionSenders = (props: ReactionSendersProps) => {
{sender === me ? ( {sender === me ? (
window.i18n('you') window.i18n('you')
) : ( ) : (
<ContactName <StyledContactContainer>
pubkey={sender} <ContactName
module="module-conversation__user" pubkey={sender}
shouldShowPubkey={false} module="module-conversation__user"
/> shouldShowPubkey={false}
/>
</StyledContactContainer>
)} )}
</Flex> </Flex>
{sender === me && ( {sender === me && (
@ -231,6 +239,7 @@ export const ReactListModal = (props: Props) => {
const msgProps = useMessageReactsPropsById(messageId); const msgProps = useMessageReactsPropsById(messageId);
const isPublic = useSelectedIsPublic(); const isPublic = useSelectedIsPublic();
const weAreAdmin = useSelectedWeAreAdmin();
const weAreModerator = useSelectedWeAreModerator(); const weAreModerator = useSelectedWeAreModerator();
const me = UserUtils.getOurPubKeyStrFromCache(); const me = UserUtils.getOurPubKeyStrFromCache();
@ -362,7 +371,7 @@ export const ReactListModal = (props: Props) => {
</> </>
)} )}
</p> </p>
{isPublic && weAreModerator && ( {isPublic && (weAreAdmin || weAreModerator) && (
<SessionButton <SessionButton
text={window.i18n('clearAll')} text={window.i18n('clearAll')}
buttonColor={SessionButtonColor.Danger} buttonColor={SessionButtonColor.Danger}

@ -88,7 +88,19 @@ export type SessionIconType =
export type SessionIconSize = 'tiny' | 'small' | 'medium' | 'large' | 'huge' | 'huge2' | 'max'; export type SessionIconSize = 'tiny' | 'small' | 'medium' | 'large' | 'huge' | 'huge2' | 'max';
export const icons: Record<string, { path: string; viewBox: string; ratio: number }> = { export type ClipRule = 'nonzero' | 'evenodd' | 'inherit';
export type FillRule = 'nonzero' | 'evenodd';
type IconProps = {
path: string;
viewBox: string;
ratio: number;
fill?: string;
clipRule?: ClipRule;
fillRule?: FillRule;
};
export const icons: Record<SessionIconType, IconProps> = {
addUser: { addUser: {
path: 'M8.85,2.17c-1.73,0-3.12,1.4-3.12,3.12s1.4,3.12,3.12,3.12c1.73,0,3.13-1.4,3.13-3.12S10.58,2.17,8.85,2.17z M8.85,0.08c2.88,0,5.21,2.33,5.21,5.21s-2.33,5.21-5.21,5.21s-5.2-2.33-5.2-5.21C3.65,2.42,5.98,0.08,8.85,0.08z M20.83,5.29 c0.54,0,0.98,0.41,1.04,0.93l0.01,0.11v2.08h2.08c0.54,0,0.98,0.41,1.04,0.93v0.12c0,0.54-0.41,0.98-0.93,1.04l-0.11,0.01h-2.08 v2.08c0,0.58-0.47,1.04-1.04,1.04c-0.54,0-0.98-0.41-1.04-0.93l-0.01-0.11v-2.08h-2.08c-0.54,0-0.98-0.41-1.04-0.93l-0.01-0.11 c0-0.54,0.41-0.98,0.93-1.04l0.11-0.01h2.08V6.34C19.79,5.76,20.26,5.29,20.83,5.29z M12.5,12.58c2.8,0,5.09,2.21,5.2,4.99v0.22 v2.08c0,0.58-0.47,1.04-1.04,1.04c-0.54,0-0.98-0.41-1.04-0.93l-0.01-0.11v-2.08c0-1.67-1.3-3.03-2.95-3.12h-0.18H5.21 c-1.67,0-3.03,1.3-3.12,2.95v0.18v2.08c0,0.58-0.47,1.04-1.04,1.04c-0.54,0-0.98-0.41-1.04-0.93L0,19.88V17.8 c0-2.8,2.21-5.09,4.99-5.2h0.22h7.29V12.58z', path: 'M8.85,2.17c-1.73,0-3.12,1.4-3.12,3.12s1.4,3.12,3.12,3.12c1.73,0,3.13-1.4,3.13-3.12S10.58,2.17,8.85,2.17z M8.85,0.08c2.88,0,5.21,2.33,5.21,5.21s-2.33,5.21-5.21,5.21s-5.2-2.33-5.2-5.21C3.65,2.42,5.98,0.08,8.85,0.08z M20.83,5.29 c0.54,0,0.98,0.41,1.04,0.93l0.01,0.11v2.08h2.08c0.54,0,0.98,0.41,1.04,0.93v0.12c0,0.54-0.41,0.98-0.93,1.04l-0.11,0.01h-2.08 v2.08c0,0.58-0.47,1.04-1.04,1.04c-0.54,0-0.98-0.41-1.04-0.93l-0.01-0.11v-2.08h-2.08c-0.54,0-0.98-0.41-1.04-0.93l-0.01-0.11 c0-0.54,0.41-0.98,0.93-1.04l0.11-0.01h2.08V6.34C19.79,5.76,20.26,5.29,20.83,5.29z M12.5,12.58c2.8,0,5.09,2.21,5.2,4.99v0.22 v2.08c0,0.58-0.47,1.04-1.04,1.04c-0.54,0-0.98-0.41-1.04-0.93l-0.01-0.11v-2.08c0-1.67-1.3-3.03-2.95-3.12h-0.18H5.21 c-1.67,0-3.03,1.3-3.12,2.95v0.18v2.08c0,0.58-0.47,1.04-1.04,1.04c-0.54,0-0.98-0.41-1.04-0.93L0,19.88V17.8 c0-2.8,2.21-5.09,4.99-5.2h0.22h7.29V12.58z',
viewBox: '0 0 25 21', viewBox: '0 0 25 21',

@ -1,7 +1,8 @@
import React from 'react'; import React, { memo } from 'react';
import styled, { css, keyframes } from 'styled-components'; import styled, { css, CSSProperties, keyframes } from 'styled-components';
import { icons, SessionIconSize, SessionIconType } from '.'; import { icons, SessionIconSize, SessionIconType } from '.';
import { ClipRule, FillRule } from './Icons';
export type SessionIconProps = { export type SessionIconProps = {
iconType: SessionIconType; iconType: SessionIconType;
@ -15,6 +16,7 @@ export type SessionIconProps = {
glowStartDelay?: number; glowStartDelay?: number;
noScale?: boolean; noScale?: boolean;
backgroundColor?: string; backgroundColor?: string;
style?: CSSProperties;
dataTestId?: string; dataTestId?: string;
unreadCount?: number; unreadCount?: number;
}; };
@ -55,6 +57,9 @@ type StyledSvgProps = {
noScale?: boolean; noScale?: boolean;
iconColor?: string; iconColor?: string;
backgroundColor?: string; backgroundColor?: string;
fill?: string;
clipRule?: ClipRule;
filleRule?: FillRule;
}; };
const rotate = keyframes` const rotate = keyframes`
@ -119,37 +124,28 @@ const animation = (props: {
return undefined; return undefined;
}; };
const Svg = React.memo(styled.svg<StyledSvgProps>` const Svg = memo(styled.svg<StyledSvgProps>`
width: ${props => props.width}; width: ${props => props.width};
transform: ${props => `rotate(${props.iconRotation}deg)`}; transform: ${props => `rotate(${props.iconRotation}deg)`};
${props => animation(props)}; ${props => animation(props)};
border-radius: ${props => props.borderRadius}; border-radius: ${props => props.borderRadius};
background-color: ${props => background-color: ${props =>
props.backgroundColor ? props.backgroundColor : '--button-icon-background-color'}; props.backgroundColor ? props.backgroundColor : 'var(--button-icon-background-color)'};
border-radius: ${props => (props.borderRadius ? props.borderRadius : '')};
filter: ${props => (props.noScale ? `drop-shadow(0px 0px 4px ${props.iconColor})` : '')}; filter: ${props => (props.noScale ? `drop-shadow(0px 0px 4px ${props.iconColor})` : '')};
fill: ${props => (props.iconColor ? props.iconColor : '--button-icon-stroke-color')}; fill: ${props => (props.iconColor ? props.iconColor : 'var(--button-icon-stroke-color)')};
padding: ${props => (props.iconPadding ? props.iconPadding : '')}; padding: ${props => (props.iconPadding ? props.iconPadding : '')};
transition: inherit; transition: inherit;
`); `);
const SessionSvg = (props: { const SessionSvg = (
viewBox: string; props: StyledSvgProps & {
path: string | Array<string>; viewBox: string;
width: string | number; path: string | Array<string>;
height: string | number; style?: CSSProperties;
iconRotation: number; dataTestId?: string;
iconColor?: string; }
rotateDuration?: number; ) => {
glowDuration?: number; const colorSvg = props.iconColor ? props.iconColor : 'var(--button-icon-stroke-color)';
glowStartDelay?: number;
noScale?: boolean;
borderRadius?: string;
backgroundColor?: string;
iconPadding?: string;
dataTestId?: string;
}) => {
const colorSvg = props.iconColor ? props.iconColor : '--button-icon-stroke-color';
const pathArray = props.path instanceof Array ? props.path : [props.path]; const pathArray = props.path instanceof Array ? props.path : [props.path];
const propsToPick = { const propsToPick = {
width: props.width, width: props.width,
@ -164,6 +160,10 @@ const SessionSvg = (props: {
backgroundColor: props.backgroundColor, backgroundColor: props.backgroundColor,
borderRadius: props.borderRadius, borderRadius: props.borderRadius,
iconPadding: props.iconPadding, iconPadding: props.iconPadding,
fill: props.fill,
clipRule: props.clipRule,
fillRule: props.filleRule,
style: props.style,
dataTestId: props.dataTestId, dataTestId: props.dataTestId,
}; };
@ -187,6 +187,7 @@ export const SessionIcon = (props: SessionIconProps) => {
noScale, noScale,
backgroundColor, backgroundColor,
iconPadding, iconPadding,
style,
dataTestId, dataTestId,
} = props; } = props;
let { iconSize, iconRotation } = props; let { iconSize, iconRotation } = props;
@ -196,6 +197,9 @@ export const SessionIcon = (props: SessionIconProps) => {
const iconDimensions = getIconDimensionFromIconSize(iconSize); const iconDimensions = getIconDimensionFromIconSize(iconSize);
const iconDef = icons[iconType]; const iconDef = icons[iconType];
const ratio = iconDef?.ratio || 1; const ratio = iconDef?.ratio || 1;
const fill = iconDef?.fill || undefined;
const clipRule = iconDef?.clipRule || 'nonzero';
const fillRule = iconDef?.fillRule || 'nonzero';
return ( return (
<SessionSvg <SessionSvg
@ -212,6 +216,10 @@ export const SessionIcon = (props: SessionIconProps) => {
iconColor={iconColor} iconColor={iconColor}
backgroundColor={backgroundColor} backgroundColor={backgroundColor}
iconPadding={iconPadding} iconPadding={iconPadding}
fill={fill}
clipRule={clipRule}
filleRule={fillRule}
style={style}
dataTestId={dataTestId} dataTestId={dataTestId}
/> />
); );

@ -68,13 +68,21 @@ function IconMessageStatus({ status }: { status: LastMessageStatusType }) {
const nonErrorIconColor = 'var(--text-secondary-color'; const nonErrorIconColor = 'var(--text-secondary-color';
switch (status) { switch (status) {
case 'error': case 'error':
return <SessionIcon iconColor={'var(--danger-color'} iconType="error" iconSize="tiny" />; return (
<SessionIcon
iconColor={'var(--danger-color'}
iconType="error"
iconSize="tiny"
style={{ flexShrink: 0 }}
/>
);
case 'read': case 'read':
return ( return (
<SessionIcon <SessionIcon
iconColor={nonErrorIconColor} iconColor={nonErrorIconColor}
iconType="doubleCheckCircleFilled" iconType="doubleCheckCircleFilled"
iconSize="tiny" iconSize="tiny"
style={{ flexShrink: 0 }}
/> />
); );
case 'sending': case 'sending':
@ -84,10 +92,18 @@ function IconMessageStatus({ status }: { status: LastMessageStatusType }) {
iconColor={nonErrorIconColor} iconColor={nonErrorIconColor}
iconType="sending" iconType="sending"
iconSize="tiny" iconSize="tiny"
style={{ flexShrink: 0 }}
/> />
); );
case 'sent': case 'sent':
return <SessionIcon iconColor={nonErrorIconColor} iconType="circleCheck" iconSize="tiny" />; return (
<SessionIcon
iconColor={nonErrorIconColor}
iconType="circleCheck"
iconSize="tiny"
style={{ flexShrink: 0 }}
/>
);
case undefined: case undefined:
return null; return null;
default: default:

@ -251,7 +251,9 @@ export async function handleSwarmDataMessage({
} }
let msgModel = let msgModel =
isSyncedMessage || (envelope.senderIdentity && isUsFromCache(envelope.senderIdentity)) isSyncedMessage ||
(envelope.senderIdentity && isUsFromCache(envelope.senderIdentity)) ||
(envelope.source && isUsFromCache(envelope.source))
? createSwarmMessageSentFromUs({ ? createSwarmMessageSentFromUs({
conversationId: convoIdToAddTheMessageTo, conversationId: convoIdToAddTheMessageTo,
messageHash, messageHash,

@ -17,6 +17,15 @@ export const DURATION = {
WEEKS: days * 7, WEEKS: days * 7,
}; };
export const FILESIZE = {
/** 1KB */
KB: 1024,
/** 1MB */
MB: 1024 * 1024,
/** 1GB */
GB: 1024 * 1024 * 1024,
};
export const TTL_DEFAULT = { export const TTL_DEFAULT = {
/** 20 seconds */ /** 20 seconds */
TYPING_MESSAGE: 20 * DURATION.SECONDS, TYPING_MESSAGE: 20 * DURATION.SECONDS,

@ -3,6 +3,7 @@ import imageType from 'image-type';
import { arrayBufferToBlob } from 'blob-util'; import { arrayBufferToBlob } from 'blob-util';
import loadImage from 'blueimp-load-image'; import loadImage from 'blueimp-load-image';
import fileSize from 'filesize';
import { StagedAttachmentType } from '../components/conversation/composition/CompositionBox'; import { StagedAttachmentType } from '../components/conversation/composition/CompositionBox';
import { SignalService } from '../protobuf'; import { SignalService } from '../protobuf';
import { getDecryptedMediaUrl } from '../session/crypto/DecryptedAttachmentsManager'; import { getDecryptedMediaUrl } from '../session/crypto/DecryptedAttachmentsManager';
@ -12,7 +13,7 @@ import { IMAGE_GIF, IMAGE_JPEG, IMAGE_PNG, IMAGE_TIFF, IMAGE_UNKNOWN } from '../
import { getAbsoluteAttachmentPath, processNewAttachment } from '../types/MessageAttachment'; import { getAbsoluteAttachmentPath, processNewAttachment } from '../types/MessageAttachment';
import { THUMBNAIL_SIDE } from '../types/attachments/VisualAttachment'; import { THUMBNAIL_SIDE } from '../types/attachments/VisualAttachment';
import { MAX_ATTACHMENT_FILESIZE_BYTES } from '../session/constants'; import { FILESIZE, MAX_ATTACHMENT_FILESIZE_BYTES } from '../session/constants';
import { perfEnd, perfStart } from '../session/utils/Performance'; import { perfEnd, perfStart } from '../session/utils/Performance';
/** /**
@ -53,7 +54,7 @@ export async function autoScaleForAvatar<T extends { contentType: string; blob:
) { ) {
const maxMeasurements = { const maxMeasurements = {
maxSide: AVATAR_MAX_SIDE, maxSide: AVATAR_MAX_SIDE,
maxSize: 1000 * 1024, maxSize: 5 * FILESIZE.MB,
}; };
// we can only upload jpeg, gif, or png as avatar/opengroup // we can only upload jpeg, gif, or png as avatar/opengroup
@ -79,7 +80,7 @@ export async function autoScaleForAvatar<T extends { contentType: string; blob:
export async function autoScaleForIncomingAvatar(incomingAvatar: ArrayBuffer) { export async function autoScaleForIncomingAvatar(incomingAvatar: ArrayBuffer) {
const maxMeasurements = { const maxMeasurements = {
maxSide: AVATAR_MAX_SIDE, maxSide: AVATAR_MAX_SIDE,
maxSize: 1000 * 1024, maxSize: 5 * FILESIZE.MB,
}; };
// the avatar url send in a message does not contain anything related to the avatar MIME type, so // the avatar url send in a message does not contain anything related to the avatar MIME type, so
@ -186,7 +187,7 @@ export async function autoScale<T extends { contentType: string; blob: Blob }>(
} }
if (blob.type === IMAGE_GIF && blob.size > maxSize) { if (blob.type === IMAGE_GIF && blob.size > maxSize) {
throw new Error(`GIF is too large, required size is ${maxSize}`); throw new Error(`GIF is too large. Max size: ${fileSize(maxSize, { base: 10, round: 0 })}`);
} }
perfStart(`loadimage-*${blob.size}`); perfStart(`loadimage-*${blob.size}`);

Loading…
Cancel
Save