@ -847,7 +847,8 @@ extension ConversationVC:
func handleItemTapped (
_ cellViewModel : MessageViewModel ,
gestureRecognizer : UITapGestureRecognizer ,
cell : UITableViewCell ,
cellLocation : CGPoint ,
using dependencies : Dependencies = Dependencies ( )
) {
guard cellViewModel . variant != . standardOutgoing || ( cellViewModel . state != . failed && cellViewModel . state != . failedToSync ) else {
@ -900,20 +901,10 @@ extension ConversationVC:
case . voiceMessage : viewModel . playOrPauseAudio ( for : cellViewModel )
case . mediaMessage :
guard
let sectionIndex : Int = self . viewModel . interactionData
. firstIndex ( where : { $0 . model = = . messages } ) ,
let messageIndex : Int = self . viewModel . interactionData [ sectionIndex ]
. elements
. firstIndex ( where : { $0 . id = = cellViewModel . id } ) ,
let cell = tableView . cellForRow ( at : IndexPath ( row : messageIndex , section : sectionIndex ) ) as ? VisibleMessageCell ,
let albumView : MediaAlbumView = cell . albumView
else { return }
let locationInCell : CGPoint = gestureRecognizer . location ( in : cell )
guard let albumView : MediaAlbumView = ( cell as ? VisibleMessageCell ) ? . albumView else { return }
// F i g u r e o u t w h i c h o f t h e m e d i a v i e w s w a s t a p p e d
let locationInAlbumView : CGPoint = cell . convert ( locationInCell , to : albumView )
let locationInAlbumView : CGPoint = cell . convert ( cellLocation , to : albumView )
guard let mediaView = albumView . mediaView ( forLocation : locationInAlbumView ) else { return }
switch mediaView . attachment . state {
@ -1034,26 +1025,51 @@ extension ConversationVC:
navigationController ? . present ( shareVC , animated : true , completion : nil )
case . textOnlyMessage :
if let quote : Quote = cellViewModel . quote {
// S c r o l l t o t h e o r i g i n a l q u o t e d m e s s a g e
let maybeOriginalInteractionInfo : Interaction . TimestampInfo ? = Storage . shared . read { db in
try quote . originalInteraction
. select ( . id , . timestampMs )
. asRequest ( of : Interaction . TimestampInfo . self )
. fetchOne ( db )
}
guard let visibleCell : VisibleMessageCell = cell as ? VisibleMessageCell else { return }
let quotePoint : CGPoint = visibleCell . convert ( cellLocation , to : visibleCell . quoteView )
let linkPreviewPoint : CGPoint = visibleCell . convert ( cellLocation , to : visibleCell . linkPreviewView ? . previewView )
let tappableLabelPoint : CGPoint = visibleCell . convert ( cellLocation , to : visibleCell . bodyTappableLabel )
let containsLinks : Bool = (
// I f t h e r e i s o n l y a s i n g l e l i n k a n d i t m a t c h e s t h e L i n k P r e v i e w t h e n c o n s i d e r t h i s _ j u s t _ a
// L i n k P r e v i e w
visibleCell . bodyTappableLabel ? . containsLinks = = true && (
( visibleCell . bodyTappableLabel ? . links . count ? ? 0 ) > 1 ||
visibleCell . bodyTappableLabel ? . links [ cellViewModel . linkPreview ? . url ? ? " " ] = = nil
)
)
let quoteViewContainsTouch : Bool = ( visibleCell . quoteView ? . bounds . contains ( quotePoint ) = = true )
let linkPreviewViewContainsTouch : Bool = ( visibleCell . linkPreviewView ? . previewView . bounds . contains ( linkPreviewPoint ) = = true )
switch ( containsLinks , quoteViewContainsTouch , linkPreviewViewContainsTouch , cellViewModel . quote , cellViewModel . linkPreview ) {
// I f t h e m e s s a g e c o n t a i n s b o t h l i n k s a n d a q u o t e , a n d t h e u s e r t a p p e d o n t h e q u o t e ; O R t h e
// m e s s a g e o n l y c o n t a i n e d a q u o t e , t h e n s c r o l l t o t h e q u o t e
case ( true , true , _ , . some ( let quote ) , _ ) , ( false , _ , _ , . some ( let quote ) , _ ) :
let maybeOriginalInteractionInfo : Interaction . TimestampInfo ? = Storage . shared . read { db in
try quote . originalInteraction
. select ( . id , . timestampMs )
. asRequest ( of : Interaction . TimestampInfo . self )
. fetchOne ( db )
}
guard let interactionInfo : Interaction . TimestampInfo = maybeOriginalInteractionInfo else {
return
}
self . scrollToInteractionIfNeeded ( with : interactionInfo , focusBehaviour : . highlight )
guard let interactionInfo : Interaction . TimestampInfo = maybeOriginalInteractionInfo else {
return
}
// I f t h e m e s s a g e c o n t a i n s b o t h l i n k s a n d a L i n k P r e v i e w , a n d t h e u s e r t a p p e d o n
// t h e L i n k P r e v i e w ; O R t h e m e s s a g e o n l y c o n t a i n e d a L i n k P r e v i e w , t h e n o p e n t h e l i n k
case ( true , _ , true , _ , . some ( let linkPreview ) ) , ( false , _ , _ , _ , . some ( let linkPreview ) ) :
switch linkPreview . variant {
case . standard : openUrl ( linkPreview . url )
case . openGroupInvitation : joinOpenGroup ( name : linkPreview . title , url : linkPreview . url )
}
self . scrollToInteractionIfNeeded ( with : interactionInfo , focusBehaviour : . highlight )
}
else if let linkPreview : LinkPreview = cellViewModel . linkPreview {
switch linkPreview . variant {
case . standard : openUrl ( linkPreview . url )
case . openGroupInvitation : joinOpenGroup ( name : linkPreview . title , url : linkPreview . url )
}
// I f t h e m e s s a g e c o n t a i n e d l i n k s t h e n i n t e r a c t w i t h t h e m d i r e c t l y
case ( true , _ , _ , _ , _ ) : visibleCell . bodyTappableLabel ? . handleTouch ( at : tappableLabelPoint )
default : break
}
default : break