@ -25,10 +25,7 @@ import { SessionQuotedMessageComposition } from './SessionQuotedMessageCompositi
import { Mention , MentionsInput } from 'react-mentions' ;
import { Mention , MentionsInput } from 'react-mentions' ;
import { CaptionEditor } from '../../CaptionEditor' ;
import { CaptionEditor } from '../../CaptionEditor' ;
import { getConversationController } from '../../../session/conversations' ;
import { getConversationController } from '../../../session/conversations' ;
import {
import { ReduxConversationType } from '../../../state/ducks/conversations' ;
ReduxConversationType ,
updateDraftForConversation ,
} from '../../../state/ducks/conversations' ;
import { SessionMemberListItem } from '../SessionMemberListItem' ;
import { SessionMemberListItem } from '../SessionMemberListItem' ;
import autoBind from 'auto-bind' ;
import autoBind from 'auto-bind' ;
import { SessionSettingCategory } from '../settings/SessionSettings' ;
import { SessionSettingCategory } from '../settings/SessionSettings' ;
@ -45,7 +42,6 @@ import {
hasLinkPreviewPopupBeenDisplayed ,
hasLinkPreviewPopupBeenDisplayed ,
} from '../../../data/data' ;
} from '../../../data/data' ;
import {
import {
getDraftForCurrentConversation ,
getMentionsInput ,
getMentionsInput ,
getQuotedMessage ,
getQuotedMessage ,
getSelectedConversation ,
getSelectedConversation ,
@ -142,10 +138,17 @@ const SendMessageButton = (props: { onClick: () => void }) => {
) ;
) ;
} ;
} ;
// keep this draft state local to not have to do a redux state update (a bit slow with our large state for soem computers)
const draftsForConversations : Array < { conversationKey : string ; draft : string } > = new Array ( ) ;
function updateDraftForConversation ( action : { conversationKey : string ; draft : string } ) {
const { conversationKey , draft } = action ;
const foundAtIndex = draftsForConversations . findIndex ( c = > c . conversationKey === conversationKey ) ;
foundAtIndex === - 1
? draftsForConversations . push ( { conversationKey , draft } )
: ( draftsForConversations [ foundAtIndex ] = action ) ;
}
interface Props {
interface Props {
sendMessage : ( msg : SendMessageType ) = > void ;
sendMessage : ( msg : SendMessageType ) = > void ;
draft : string ;
onLoadVoiceNoteView : any ;
onLoadVoiceNoteView : any ;
onExitVoiceNoteView : any ;
onExitVoiceNoteView : any ;
selectedConversationKey : string ;
selectedConversationKey : string ;
@ -157,7 +160,7 @@ interface Props {
interface State {
interface State {
showRecordingView : boolean ;
showRecordingView : boolean ;
draft : string ;
showEmojiPanel : boolean ;
showEmojiPanel : boolean ;
voiceRecording? : Blob ;
voiceRecording? : Blob ;
ignoredLink? : string ; // set the the ignored url when users closed the link preview
ignoredLink? : string ; // set the the ignored url when users closed the link preview
@ -185,10 +188,11 @@ const sendMessageStyle = {
minHeight : '24px' ,
minHeight : '24px' ,
width : '100%' ,
width : '100%' ,
} ;
} ;
const getDefaultState = ( newConvoId? : string ) = > {
const getDefaultState = ( ) = > {
return {
return {
message : '' ,
draft :
( newConvoId && draftsForConversations . find ( c = > c . conversationKey === newConvoId ) ? . draft ) ||
'' ,
voiceRecording : undefined ,
voiceRecording : undefined ,
showRecordingView : false ,
showRecordingView : false ,
showEmojiPanel : false ,
showEmojiPanel : false ,
@ -238,7 +242,7 @@ class SessionCompositionBoxInner extends React.Component<Props, State> {
public componentDidUpdate ( prevProps : Props , _prevState : State ) {
public componentDidUpdate ( prevProps : Props , _prevState : State ) {
// reset the state on new conversation key
// reset the state on new conversation key
if ( prevProps . selectedConversationKey !== this . props . selectedConversationKey ) {
if ( prevProps . selectedConversationKey !== this . props . selectedConversationKey ) {
this . setState ( getDefaultState ( ) , this . focusCompositionBox ) ;
this . setState ( getDefaultState ( this . props . selectedConversationKey ) , this . focusCompositionBox ) ;
this . lastBumpTypingMessageLength = 0 ;
this . lastBumpTypingMessageLength = 0 ;
} else if ( this . props . stagedAttachments ? . length !== prevProps . stagedAttachments ? . length ) {
} else if ( this . props . stagedAttachments ? . length !== prevProps . stagedAttachments ? . length ) {
// if number of staged attachment changed, focus the composition box for a more natural UI
// if number of staged attachment changed, focus the composition box for a more natural UI
@ -433,7 +437,7 @@ class SessionCompositionBoxInner extends React.Component<Props, State> {
private renderTextArea() {
private renderTextArea() {
const { i18n } = window ;
const { i18n } = window ;
const { draft } = this . prop s;
const { draft } = this . state ;
if ( ! this . props . selectedConversation ) {
if ( ! this . props . selectedConversation ) {
return null ;
return null ;
@ -585,7 +589,7 @@ class SessionCompositionBoxInner extends React.Component<Props, State> {
return < > < / > ;
return < > < / > ;
}
}
// we try to match the first link found in the current message
// we try to match the first link found in the current message
const links = window . Signal . LinkPreviews . findLinks ( this . prop s. draft , undefined ) ;
const links = window . Signal . LinkPreviews . findLinks ( this . state . draft , undefined ) ;
if ( ! links || links . length === 0 || ignoredLink === links [ 0 ] ) {
if ( ! links || links . length === 0 || ignoredLink === links [ 0 ] ) {
if ( this . state . stagedLinkPreview ) {
if ( this . state . stagedLinkPreview ) {
this . setState ( {
this . setState ( {
@ -809,12 +813,12 @@ class SessionCompositionBoxInner extends React.Component<Props, State> {
}
}
private async onKeyUp() {
private async onKeyUp() {
const { draft } = this . prop s;
const { draft } = this . state ;
// Called whenever the user changes the message composition field. But only
// Called whenever the user changes the message composition field. But only
// fires if there's content in the message field after the change.
// fires if there's content in the message field after the change.
// Also, check for a message length change before firing it up, to avoid
// Also, check for a message length change before firing it up, to avoid
// catching ESC, tab, or whatever which is not typing
// catching ESC, tab, or whatever which is not typing
if ( draft . length && draft . length !== this . lastBumpTypingMessageLength ) {
if ( draft && draft . length && draft . length !== this . lastBumpTypingMessageLength ) {
const conversationModel = getConversationController ( ) . get ( this . props . selectedConversationKey ) ;
const conversationModel = getConversationController ( ) . get ( this . props . selectedConversationKey ) ;
if ( ! conversationModel ) {
if ( ! conversationModel ) {
return ;
return ;
@ -852,7 +856,7 @@ class SessionCompositionBoxInner extends React.Component<Props, State> {
return replacedMentions ;
return replacedMentions ;
} ;
} ;
const messagePlaintext = cleanMentions ( this . parseEmojis ( this . prop s. draft ) ) ;
const messagePlaintext = cleanMentions ( this . parseEmojis ( this . state . draft ) ) ;
const { selectedConversation } = this . props ;
const { selectedConversation } = this . props ;
@ -924,13 +928,12 @@ class SessionCompositionBoxInner extends React.Component<Props, State> {
showEmojiPanel : false ,
showEmojiPanel : false ,
stagedLinkPreview : undefined ,
stagedLinkPreview : undefined ,
ignoredLink : undefined ,
ignoredLink : undefined ,
draft : '' ,
} ) ;
updateDraftForConversation ( {
conversationKey : this.props.selectedConversationKey ,
draft : '' ,
} ) ;
} ) ;
window . inboxStore ? . dispatch (
updateDraftForConversation ( {
conversationKey : this.props.selectedConversationKey ,
draft : '' ,
} )
) ;
} catch ( e ) {
} catch ( e ) {
// Message sending failed
// Message sending failed
window ? . log ? . error ( e ) ;
window ? . log ? . error ( e ) ;
@ -1022,12 +1025,8 @@ class SessionCompositionBoxInner extends React.Component<Props, State> {
private onChange ( event : any ) {
private onChange ( event : any ) {
const draft = event . target . value ? ? '' ;
const draft = event . target . value ? ? '' ;
window . inboxStore ? . dispatch (
this . setState ( { draft } ) ;
updateDraftForConversation ( {
updateDraftForConversation ( { conversationKey : this.props.selectedConversationKey , draft } ) ;
conversationKey : this.props.selectedConversationKey ,
draft ,
} )
) ;
}
}
private getSelectionBasedOnMentions ( index : number ) {
private getSelectionBasedOnMentions ( index : number ) {
@ -1035,7 +1034,7 @@ class SessionCompositionBoxInner extends React.Component<Props, State> {
// this is kind of a pain as the mentions box has two inputs, one with the real text, and one with the extracted mentions
// this is kind of a pain as the mentions box has two inputs, one with the real text, and one with the extracted mentions
// the index shown to the user is actually just the visible part of the mentions (so the part between ᅲ...ᅭ
// the index shown to the user is actually just the visible part of the mentions (so the part between ᅲ...ᅭ
const matches = this . prop s. draft . match ( this . mentionsRegex ) ;
const matches = this . state . draft . match ( this . mentionsRegex ) ;
let lastMatchStartIndex = 0 ;
let lastMatchStartIndex = 0 ;
let lastMatchEndIndex = 0 ;
let lastMatchEndIndex = 0 ;
@ -1049,7 +1048,7 @@ class SessionCompositionBoxInner extends React.Component<Props, State> {
const displayNameEnd = match . lastIndexOf ( '\uFFD2' ) ;
const displayNameEnd = match . lastIndexOf ( '\uFFD2' ) ;
const displayName = match . substring ( displayNameStart , displayNameEnd ) ;
const displayName = match . substring ( displayNameStart , displayNameEnd ) ;
const currentMatchStartIndex = this . prop s. draft . indexOf ( match ) + lastMatchStartIndex ;
const currentMatchStartIndex = this . state . draft . indexOf ( match ) + lastMatchStartIndex ;
lastMatchStartIndex = currentMatchStartIndex ;
lastMatchStartIndex = currentMatchStartIndex ;
lastMatchEndIndex = currentMatchStartIndex + match . length ;
lastMatchEndIndex = currentMatchStartIndex + match . length ;
@ -1093,7 +1092,7 @@ class SessionCompositionBoxInner extends React.Component<Props, State> {
return ;
return ;
}
}
const { draft } = this . prop s;
const { draft } = this . state ;
const currentSelectionStart = Number ( messageBox . selectionStart ) ;
const currentSelectionStart = Number ( messageBox . selectionStart ) ;
@ -1103,12 +1102,11 @@ class SessionCompositionBoxInner extends React.Component<Props, State> {
const end = draft . slice ( realSelectionStart ) ;
const end = draft . slice ( realSelectionStart ) ;
const newMessage = ` ${ before } ${ colons } ${ end } ` ;
const newMessage = ` ${ before } ${ colons } ${ end } ` ;
window . inboxStore ? . dispatch (
this . setState ( { draft : newMessage } ) ;
updateDraftForConversation ( {
updateDraftForConversation ( {
conversationKey : this.props.selectedConversationKey ,
conversationKey : this.props.selectedConversationKey ,
draft : newMessage ,
draft : newMessage ,
} )
} ) ;
) ;
// update our selection because updating text programmatically
// update our selection because updating text programmatically
// will put the selection at the end of the textarea
// will put the selection at the end of the textarea
@ -1138,7 +1136,6 @@ const mapStateToProps = (state: StateType) => {
quotedMessageProps : getQuotedMessage ( state ) ,
quotedMessageProps : getQuotedMessage ( state ) ,
selectedConversation : getSelectedConversation ( state ) ,
selectedConversation : getSelectedConversation ( state ) ,
selectedConversationKey : getSelectedConversationKey ( state ) ,
selectedConversationKey : getSelectedConversationKey ( state ) ,
draft : getDraftForCurrentConversation ( state ) ,
theme : getTheme ( state ) ,
theme : getTheme ( state ) ,
} ;
} ;
} ;
} ;