@ -81,12 +81,18 @@
@ import Photos ;
#define kYapDatabaseRangeLength 50
#define kYapDatabaseRangeMaxLength 300
#define kYapDatabaseRangeMinLength 20
#define JSQ_TOOLBAR_ICON_HEIGHT 22
#define JSQ_TOOLBAR_ICON_WIDTH 22
#define JSQ_IMAGE_INSET 5
/ / Always load up to 50 messsages when user arrives .
static const int kYapDatabasePageSize = 50 ;
/ / Never show more than 50 * 50 = 2 , 500 messages in conversation view at a time .
static const int kYapDatabaseMaxPageCount = 50 ;
/ / Never show more than 6 * 50 = 300 messages in conversation view when user
/ / arrives .
static const int kYapDatabaseMaxInitialPageCount = 6 ;
static const int kYapDatabaseRangeMaxLength = kYapDatabasePageSize * kYapDatabaseMaxPageCount ;
static const int kYapDatabaseRangeMinLength = 0 ;
static const int JSQ_TOOLBAR_ICON_HEIGHT = 22 ;
static const int JSQ_TOOLBAR_ICON_WIDTH = 22 ;
static const int JSQ_IMAGE_INSET = 5 ;
static NSTimeInterval const kTSMessageSentDateShowTimeInterval = 5 * 60 ;
@ -97,42 +103,6 @@ typedef enum : NSUInteger {
kMediaTypeVideo ,
} kMediaTypes ;
#pragma mark -
@ interface OWSMessagesCollectionViewFlowLayout : JSQMessagesCollectionViewFlowLayout
@ property ( nonatomic ) BOOL ignoreLayout ;
@ end
#pragma mark -
@ implementation OWSMessagesCollectionViewFlowLayout
- ( void ) prepareLayout
{
if ( self . ignoreLayout ) {
DDLogInfo ( @ "%@ ignoring layout.", self.tag);
return ;
}
[ super prepareLayout ] ;
}
#pragma mark - Logging
+ ( NSString * ) tag
{
return [ NSString stringWithFormat : @ "[ %@]", self.class];
}
- ( NSString * ) tag
{
return self . class . tag ;
}
@ end
#pragma mark -
@ protocol OWSTextViewPasteDelegate < NSObject >
@ -799,7 +769,23 @@ typedef enum : NSUInteger {
[ self . messageMappings updateWithTransaction : transaction ] ;
} ] ;
self . page = 0 ;
[ self updateRangeOptionsForPage : self . page ] ;
if ( self . offersAndIndicators . unreadIndicatorPosition != nil ) {
long unreadIndicatorPosition = [ self . offersAndIndicators . unreadIndicatorPosition longValue ] ;
/ / If there is an unread indicator , increase the initial load window
/ / to include it .
OWSAssert ( unreadIndicatorPosition > 0 ) ;
OWSAssert ( unreadIndicatorPosition < = kYapDatabaseRangeMaxLength ) ;
/ / We ' d like to include at least N seen messages , if possible ,
/ / to give the user the context of where they left off the conversation .
const int kPreferredSeenMessageCount = 1 ;
self . page = ( NSUInteger ) MAX ( 0 ,
MIN ( kYapDatabaseMaxInitialPageCount - 1 ,
( unreadIndicatorPosition + kPreferredSeenMessageCount ) / kYapDatabasePageSize ) ) ;
}
[ self updateMessageMappingRangeOptions ] ;
[ self updateLoadEarlierVisible ] ;
[ self . collectionView reloadData ] ;
}
@ -881,6 +867,7 @@ typedef enum : NSUInteger {
/ / invalidate layout
[ self . collectionView . collectionViewLayout
invalidateLayoutWithContext : [ JSQMessagesCollectionViewFlowLayoutInvalidationContext context ] ] ;
self . collectionView . collectionViewLayout . bubbleSizeCalculator = [ [ OWSMessagesBubblesSizeCalculator alloc ] init ] ;
}
}
@ -982,12 +969,17 @@ typedef enum : NSUInteger {
- ( void ) viewWillAppear : ( BOOL ) animated
{
/ / Ignore layout requests in viewWillAppear .
/ / JSQMessagesView forces layout then invalidates the layout .
/ / Besides , we ' ll be changing the contents of the view below .
( ( OWSMessagesCollectionViewFlowLayout * ) self . collectionView . collectionViewLayout ) . ignoreLayout = YES ;
/ / We need to update the dynamic interactions before we do any layout .
[ self ensureThreadOffersAndIndicators ] ;
/ / Triggering modified notification renders "call notification " when leaving full screen call view
[ self . thread touch ] ;
[ self ensureBlockStateIndicator ] ;
[ self resetContentAndLayout ] ;
[ super viewWillAppear : animated ] ;
( ( OWSMessagesCollectionViewFlowLayout * ) self . collectionView . collectionViewLayout ) . ignoreLayout = NO ;
/ / In case we ' re dismissing a CNContactViewController which requires default system appearance
[ UIUtil applySignalAppearence ] ;
@ -1001,16 +993,11 @@ typedef enum : NSUInteger {
[ self toggleObservers : YES ] ;
[ self ensureThreadOffersAndIndicators ] ;
/ / Triggering modified notification renders "call notification " when leaving full screen call view
[ self . thread touch ] ;
/ / restart any animations that were stopped e . g . while inspecting the contact info screens .
[ self startExpirationTimerAnimations ] ;
/ / We should have already requested contact access at this point , so this should be a no - op
/ / unless it ever becomes possible to to load this VC without going via the SignalsViewController
/ / unless it ever becomes possible to load this VC without going via the SignalsViewController .
[ self . contactsManager requestSystemContactsOnce ] ;
OWSDisappearingMessagesConfiguration * configuration =
@ -1031,15 +1018,9 @@ typedef enum : NSUInteger {
action : shareSelector ] ,
] ;
[ self ensureBlockStateIndicator ] ;
[ self resetContentAndLayout ] ;
[ ( ( OWSMessagesToolbarContentView * ) self . inputToolbar . contentView ) ensureSubviews ] ;
[ self . collectionView . collectionViewLayout
invalidateLayoutWithContext : [ JSQMessagesCollectionViewFlowLayoutInvalidationContext context ] ] ;
[ self . scrollLaterTimer invalidate ] ;
/ / We want to scroll to the bottom _after_ the layout has been updated .
self . scrollLaterTimer = [ NSTimer weakScheduledTimerWithTimeInterval : 0.001 f
@ -1069,18 +1050,40 @@ typedef enum : NSUInteger {
NSIndexPath * _Nullable indexPath = [ self indexPathOfUnreadMessagesIndicator ] ;
if ( indexPath ) {
[ self . collectionView scrollToItemAtIndexPath : indexPath
atScrollPosition : UICollectionViewScrollPositionTop
animated : NO ] ;
if ( indexPath . section == 0 && indexPath . row == 0 ) {
[ self . collectionView setContentOffset : CGPointZero animated : NO ] ;
} else {
[ self . collectionView scrollToItemAtIndexPath : indexPath
atScrollPosition : UICollectionViewScrollPositionTop
animated : NO ] ;
}
} else {
[ self scrollToBottomAnimated : NO ] ;
}
}
- ( void ) scrollToUnreadIndicatorAnimated
{
[ self . scrollLaterTimer invalidate ] ;
self . scrollLaterTimer = nil ;
NSIndexPath * _Nullable indexPath = [ self indexPathOfUnreadMessagesIndicator ] ;
if ( indexPath ) {
if ( indexPath . section == 0 && indexPath . row == 0 ) {
[ self . collectionView setContentOffset : CGPointZero animated : YES ] ;
} else {
[ self . collectionView scrollToItemAtIndexPath : indexPath
atScrollPosition : UICollectionViewScrollPositionTop
animated : YES ] ;
}
}
}
- ( void ) resetContentAndLayout
{
/ / Avoid layout corrupt issues and out - of - date message subtitles .
[ self . collectionView . collectionViewLayout invalidateLayout ] ;
[ self . collectionView . collectionViewLayout
invalidateLayoutWithContext : [ JSQMessagesCollectionViewFlowLayoutInvalidationContext context ] ] ;
[ self . collectionView reloadData ] ;
}
@ -1536,7 +1539,6 @@ typedef enum : NSUInteger {
/ / Overiding JSQMVC layout defaults
- ( void ) initializeCollectionViewLayout
{
self . collectionView . collectionViewLayout = [ OWSMessagesCollectionViewFlowLayout new ] ;
[ self . collectionView . collectionViewLayout setMessageBubbleFont : [ UIFont ows_dynamicTypeBodyFont ] ] ;
self . collectionView . showsVerticalScrollIndicator = NO ;
@ -1560,7 +1562,6 @@ typedef enum : NSUInteger {
self . outgoingBubbleImageData = [ bubbleFactory outgoingMessagesBubbleImageWithColor : [ UIColor ows_materialBlueColor ] ] ;
self . currentlyOutgoingBubbleImageData = [ bubbleFactory outgoingMessagesBubbleImageWithColor : [ UIColor ows_fadedBlueColor ] ] ;
self . outgoingMessageFailedImageData = [ bubbleFactory outgoingMessagesBubbleImageWithColor : [ UIColor grayColor ] ] ;
}
#pragma mark - Identity
@ -1895,26 +1896,33 @@ typedef enum : NSUInteger {
case TSCallAdapter : {
OWSCall * call = ( OWSCall * ) message ;
cell = [ self loadCallCellForCall : call atIndexPath : indexPath ] ;
} break ;
break ;
}
case TSInfoMessageAdapter : {
cell = [ self loadInfoMessageCellForMessage : ( TSMessageAdapter * ) message atIndexPath : indexPath ] ;
} break ;
break ;
}
case TSErrorMessageAdapter : {
cell = [ self loadErrorMessageCellForMessage : ( TSMessageAdapter * ) message atIndexPath : indexPath ] ;
} break ;
break ;
}
case TSIncomingMessageAdapter : {
cell = [ self loadIncomingMessageCellForMessage : message atIndexPath : indexPath ] ;
} break ;
break ;
}
case TSOutgoingMessageAdapter : {
cell = [ self loadOutgoingCellForMessage : message atIndexPath : indexPath ] ;
} break ;
break ;
}
case TSUnreadIndicatorAdapter : {
cell = [ self loadUnreadIndicatorCell : indexPath ] ;
} break ;
cell = [ self loadUnreadIndicatorCell : indexPath interaction : message . interaction ] ;
break ;
}
default : {
DDLogWarn ( @ "using default cell constructor for message : %@", message);
cell = ( JSQMessagesCollectionViewCell * ) [ super collectionView : collectionView cellForItemAtIndexPath : indexPath ] ;
} break ;
break ;
}
}
cell . delegate = collectionView ;
@ -1981,12 +1989,18 @@ typedef enum : NSUInteger {
}
- ( JSQMessagesCollectionViewCell * ) loadUnreadIndicatorCell : ( NSIndexPath * ) indexPath
interaction : ( TSInteraction * ) interaction
{
OWSAssert ( indexPath ) ;
OWSAssert ( interaction ) ;
OWSAssert ( [ interaction isKindOfClass : [ TSUnreadIndicatorInteraction class ] ] ) ;
TSUnreadIndicatorInteraction * unreadIndicator = ( TSUnreadIndicatorInteraction * ) interaction ;
OWSUnreadIndicatorCell * cell =
[ self . collectionView dequeueReusableCellWithReuseIdentifier : [ OWSUnreadIndicatorCell cellReuseIdentifier ]
forIndexPath : indexPath ] ;
cell . interaction = unreadIndicator ;
[ cell configure ] ;
return cell ;
@ -2528,25 +2542,73 @@ typedef enum : NSUInteger {
- ( void ) collectionView : ( JSQMessagesCollectionView * ) collectionView
header : ( JSQMessagesLoadEarlierHeaderView * ) headerView
didTapLoadEarlierMessagesButton : ( UIButton * ) sender {
if ( [ self shouldShowLoadEarlierMessages ] ) {
self . page + + ;
}
didTapLoadEarlierMessagesButton : ( UIButton * ) sender
{
[ self . scrollLaterTimer invalidate ] ;
self . scrollLaterTimer = nil ;
NSInteger item = ( NSInteger ) [ self scrollToItem ] ;
/ / We want to restore the current scroll state after we update the range , update
/ / the dynamic interactions and re - layout . Here we take a "before " snapshot .
CGFloat scrollDistanceToBottom = self . collectionView . contentSize . height - self . collectionView . contentOffset . y ;
self . page = MIN ( self . page + 1 , ( NSUInteger ) kYapDatabaseMaxPageCount - 1 ) ;
/ / To update a YapDatabaseViewMappings , you can call either :
/ /
/ / * [ YapDatabaseViewMappings updateWithTransaction ]
/ / * [ YapDatabaseViewMappings getSectionChanges : rowChanges : forNotifications : withMappings : ]
/ /
/ / ...but you can't call both.
/ /
/ / If ensureThreadOffersAndIndicators modifies the database ,
/ / the mappings will be updated by yapDatabaseModified .
/ / This will leave the mapping range in a bad state .
/ / Therefore we temporarily disable observation of YapDatabaseModifiedNotification
/ / while updating the range and the dynamic interactions .
[ [ NSNotificationCenter defaultCenter ] removeObserver : self name : YapDatabaseModifiedNotification object : nil ] ;
/ / We need to update the dynamic interactions after loading earlier messages ,
/ / since the unseen indicator may need to move or change .
[ self ensureThreadOffersAndIndicators ] ;
[ self updateRangeOptionsForPage : self . page ] ;
[ self update MessageMapping RangeOptions] ;
/ / We need to `beginLongLivedReadTransaction ` before we update our
/ / mapping in order to jump to the most recent commit .
[ self . uiDatabaseConnection beginLongLivedReadTransaction ] ;
[ self . uiDatabaseConnection readWithBlock : ^( YapDatabaseReadTransaction * transaction ) {
[ self . messageMappings updateWithTransaction : transaction ] ;
[ self . messageMappings updateWithTransaction : transaction ] ;
} ] ;
[ self updateLayoutForEarlierMessagesWithOffset : item ] ;
[ self . collectionView . collectionViewLayout
invalidateLayoutWithContext : [ JSQMessagesCollectionViewFlowLayoutInvalidationContext context ] ] ;
[ self . collectionView reloadData ] ;
[ [ NSNotificationCenter defaultCenter ] addObserver : self
selector : @ selector ( yapDatabaseModified : )
name : YapDatabaseModifiedNotification
object : nil ] ;
[ self . collectionView . collectionViewLayout
invalidateLayoutWithContext : [ JSQMessagesCollectionViewFlowLayoutInvalidationContext context ] ] ;
[ self . collectionView reloadData ] ;
[ self . collectionView layoutSubviews ] ;
self . collectionView . contentOffset = CGPointMake ( 0 , self . collectionView . contentSize . height - scrollDistanceToBottom ) ;
[ self . scrollLaterTimer invalidate ] ;
/ / We want to scroll to the bottom _after_ the layout has been updated .
self . scrollLaterTimer = [ NSTimer weakScheduledTimerWithTimeInterval : 0.001 f
target : self
selector : @ selector ( scrollToUnreadIndicatorAnimated )
userInfo : nil
repeats : NO ] ;
[ self updateLoadEarlierVisible ] ;
}
- ( BOOL ) shouldShowLoadEarlierMessages {
if ( self . page == kYapDatabaseMaxPageCount - 1 ) {
return NO ;
}
__block BOOL show = YES ;
[ self . uiDatabaseConnection readWithBlock : ^( YapDatabaseReadTransaction * transaction ) {
@ -2558,8 +2620,8 @@ typedef enum : NSUInteger {
}
- ( NSUInteger ) scrollToItem {
__block NSUInteger item =
kYapDatabaseRangeLength * ( self . page + 1 ) - [ self . messageMappings numberOfItemsInGroup : self . thread . uniqueId ] ;
__block NSUInteger item
= kYapDatabasePageSize * ( self . page + 1 ) - [ self . messageMappings numberOfItemsInGroup : self . thread . uniqueId ] ;
[ self . uiDatabaseConnection readWithBlock : ^( YapDatabaseReadTransaction * transaction ) {
@ -2568,7 +2630,7 @@ typedef enum : NSUInteger {
[ [ transaction ext : TSMessageDatabaseViewExtensionName ] numberOfItemsInGroup : self . thread . uniqueId ] ;
NSUInteger numberOfMessagesToLoad = numberOfTotalMessages - numberOfVisibleMessages ;
BOOL canLoadFullRange = numberOfMessagesToLoad > = kYapDatabase RangeLength ;
BOOL canLoadFullRange = numberOfMessagesToLoad > = kYapDatabase PageSize ;
if ( !canLoadFullRange ) {
item = numberOfMessagesToLoad ;
@ -2582,23 +2644,10 @@ typedef enum : NSUInteger {
[ self setShowLoadEarlierMessagesHeader : [ self shouldShowLoadEarlierMessages ] ] ;
}
- ( void ) updateLayoutForEarlierMessagesWithOffset : ( NSInteger ) offset {
[ self . collectionView . collectionViewLayout
invalidateLayoutWithContext : [ JSQMessagesCollectionViewFlowLayoutInvalidationContext context ] ] ;
[ self . collectionView reloadData ] ;
[ self . scrollLaterTimer invalidate ] ;
self . scrollLaterTimer = nil ;
[ self . collectionView scrollToItemAtIndexPath : [ NSIndexPath indexPathForItem : offset inSection : 0 ]
atScrollPosition : UICollectionViewScrollPositionTop
animated : NO ] ;
[ self updateLoadEarlierVisible ] ;
}
- ( void ) updateRangeOptionsForPage : ( NSUInteger ) page {
- ( void ) updateMessageMappingRangeOptions
{
YapDatabaseViewRangeOptions * rangeOptions =
[ YapDatabaseViewRangeOptions flexibleRangeWithLength : kYapDatabase RangeLength * ( page + 1 )
[ YapDatabaseViewRangeOptions flexibleRangeWithLength : kYapDatabasePageSize * ( self . page + 1 )
offset : 0
from : YapDatabaseViewEnd ] ;
@ -2903,15 +2952,18 @@ typedef enum : NSUInteger {
{
OWSAssert ( [ NSThread isMainThread ] ) ;
const int initialMaxRangeSize = kYapDatabasePageSize * kYapDatabaseMaxInitialPageCount ;
const int currentMaxRangeSize = ( int ) ( self . page + 1 ) * kYapDatabasePageSize ;
const int maxRangeSize = MAX ( initialMaxRangeSize , currentMaxRangeSize ) ;
self . offersAndIndicators =
[ ThreadUtil ensureThreadOffersAndIndicators : self . thread
storageManager : self . storageManager
contactsManager : self . contactsManager
blockingManager : self . blockingManager
hideUnreadMessagesIndicator : self . hasClearedUnreadMessagesIndicator
fixedUnreadIndicatorTimestamp : ( self . offersAndIndicators . unreadIndicator
? @ ( self . offersAndIndicators . unreadIndicator . timestamp )
: nil ) ] ;
firstUnseenInteractionTimestamp : self . offersAndIndicators . firstUnseenInteractionTimestamp
maxRangeSize : maxRangeSize ] ;
}
- ( void ) clearUnreadMessagesIndicator
@ -3370,55 +3422,56 @@ typedef enum : NSUInteger {
__block BOOL scrollToBottom = wasAtBottom ;
[ self . collectionView performBatchUpdates : ^{
for ( YapDatabaseViewRowChange * rowChange in messageRowChanges ) {
switch ( rowChange . type ) {
case YapDatabaseViewChangeDelete : {
[ self . collectionView deleteItemsAtIndexPaths : @ [ rowChange . indexPath ] ] ;
YapCollectionKey * collectionKey = rowChange . collectionKey ;
if ( collectionKey . key ) {
[ self . messageAdapterCache removeObjectForKey : collectionKey . key ] ;
}
break ;
}
case YapDatabaseViewChangeInsert : {
[ self . collectionView insertItemsAtIndexPaths : @ [ rowChange . newIndexPath ] ] ;
TSInteraction * interaction = [ self interactionAtIndexPath : rowChange . newIndexPath ] ;
if ( [ interaction isKindOfClass : [ TSOutgoingMessage class ] ] ) {
scrollToBottom = YES ;
shouldAnimateScrollToBottom = NO ;
}
break ;
}
case YapDatabaseViewChangeMove : {
[ self . collectionView deleteItemsAtIndexPaths : @ [ rowChange . indexPath ] ] ;
[ self . collectionView insertItemsAtIndexPaths : @ [ rowChange . newIndexPath ] ] ;
break ;
}
case YapDatabaseViewChangeUpdate : {
YapCollectionKey * collectionKey = rowChange . collectionKey ;
if ( collectionKey . key ) {
[ self . messageAdapterCache removeObjectForKey : collectionKey . key ] ;
}
[ self . collectionView reloadItemsAtIndexPaths : @ [ rowChange . indexPath ] ] ;
break ;
}
}
}
for ( YapDatabaseViewRowChange * rowChange in messageRowChanges ) {
switch ( rowChange . type ) {
case YapDatabaseViewChangeDelete : {
[ self . collectionView deleteItemsAtIndexPaths : @ [ rowChange . indexPath ] ] ;
YapCollectionKey * collectionKey = rowChange . collectionKey ;
if ( collectionKey . key ) {
[ self . messageAdapterCache removeObjectForKey : collectionKey . key ] ;
}
break ;
}
case YapDatabaseViewChangeInsert : {
[ self . collectionView insertItemsAtIndexPaths : @ [ rowChange . newIndexPath ] ] ;
TSInteraction * interaction = [ self interactionAtIndexPath : rowChange . newIndexPath ] ;
if ( [ interaction isKindOfClass : [ TSOutgoingMessage class ] ] ) {
scrollToBottom = YES ;
shouldAnimateScrollToBottom = NO ;
}
break ;
}
case YapDatabaseViewChangeMove : {
[ self . collectionView deleteItemsAtIndexPaths : @ [ rowChange . indexPath ] ] ;
[ self . collectionView insertItemsAtIndexPaths : @ [ rowChange . newIndexPath ] ] ;
break ;
}
case YapDatabaseViewChangeUpdate : {
YapCollectionKey * collectionKey = rowChange . collectionKey ;
if ( collectionKey . key ) {
[ self . messageAdapterCache removeObjectForKey : collectionKey . key ] ;
}
[ self . collectionView reloadItemsAtIndexPaths : @ [ rowChange . indexPath ] ] ;
break ;
}
}
}
}
completion : ^( BOOL success ) {
if ( !success ) {
[ self . collectionView . collectionViewLayout
invalidateLayoutWithContext : [ JSQMessagesCollectionViewFlowLayoutInvalidationContext context ] ] ;
[ self . collectionView reloadData ] ;
}
if ( scrollToBottom ) {
[ self . scrollLaterTimer invalidate ] ;
self . scrollLaterTimer = nil ;
[ self scrollToBottomAnimated : shouldAnimateScrollToBottom ] ;
}
if ( !success ) {
[ self . collectionView . collectionViewLayout
invalidateLayoutWithContext : [ JSQMessagesCollectionViewFlowLayoutInvalidationContext context ] ] ;
[ self . collectionView reloadData ] ;
}
if ( scrollToBottom ) {
[ self . scrollLaterTimer invalidate ] ;
self . scrollLaterTimer = nil ;
[ self scrollToBottomAnimated : shouldAnimateScrollToBottom ] ;
}
} ] ;
}
@ -3436,26 +3489,23 @@ typedef enum : NSUInteger {
return numberOfMessages ;
}
- ( TSInteraction * ) interactionAtIndexPath : ( NSIndexPath * ) indexPath {
__block TSInteraction * message = nil ;
- ( TSInteraction * ) interactionAtIndexPath : ( NSIndexPath * ) indexPath
{
OWSAssert ( indexPath ) ;
OWSAssert ( indexPath . section == 0 ) ;
OWSAssert ( self . messageMappings ) ;
__block TSInteraction * interaction ;
[ self . uiDatabaseConnection readWithBlock : ^( YapDatabaseReadTransaction * transaction ) {
YapDatabaseViewTransaction * viewTransaction = [ transaction ext : TSMessageDatabaseViewExtensionName ] ;
NSParameterAssert ( viewTransaction != nil ) ;
NSParameterAssert ( self . messageMappings != nil ) ;
NSParameterAssert ( indexPath != nil ) ;
NSUInteger row = ( NSUInteger ) indexPath . row ;
NSUInteger section = ( NSUInteger ) indexPath . section ;
NSUInteger numberOfItemsInSection __unused = [ self . messageMappings numberOfItemsInSection : section ] ;
NSAssert ( row < numberOfItemsInSection ,
@ "Cannot fetch message because row %d is >= numberOfItemsInSection %d",
( int ) row ,
( int ) numberOfItemsInSection ) ;
message = [ viewTransaction objectAtRow : row inSection : section withMappings : self . messageMappings ] ;
NSParameterAssert ( message != nil ) ;
YapDatabaseViewTransaction * viewTransaction = [ transaction ext : TSMessageDatabaseViewExtensionName ] ;
OWSAssert ( viewTransaction ) ;
interaction = [ viewTransaction objectAtRow : ( NSUInteger ) indexPath . row
inSection : ( NSUInteger ) indexPath . section
withMappings : self . messageMappings ] ;
OWSAssert ( interaction ) ;
} ] ;
return message ;
return interaction ;
}
- ( id < OWSMessageData > ) messageAtIndexPath : ( NSIndexPath * ) indexPath