mirror of https://github.com/oxen-io/session-ios
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
289 lines
9.7 KiB
Objective-C
289 lines
9.7 KiB
Objective-C
//
|
|
// Copyright (c) 2018 Open Whisper Systems. All rights reserved.
|
|
//
|
|
|
|
#import "OWSMessageFooterView.h"
|
|
#import "DateUtil.h"
|
|
#import "OWSMessageTimerView.h"
|
|
#import "Signal-Swift.h"
|
|
#import <QuartzCore/QuartzCore.h>
|
|
|
|
NS_ASSUME_NONNULL_BEGIN
|
|
|
|
@interface OWSMessageFooterView ()
|
|
|
|
@property (nonatomic) UILabel *timestampLabel;
|
|
@property (nonatomic) UIImageView *statusIndicatorImageView;
|
|
@property (nonatomic) OWSMessageTimerView *messageTimerView;
|
|
|
|
@end
|
|
|
|
@implementation OWSMessageFooterView
|
|
|
|
// `[UIView init]` invokes `[self initWithFrame:...]`.
|
|
- (instancetype)initWithFrame:(CGRect)frame
|
|
{
|
|
if (self = [super initWithFrame:frame]) {
|
|
[self commontInit];
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void)commontInit
|
|
{
|
|
// Ensure only called once.
|
|
OWSAssert(!self.timestampLabel);
|
|
|
|
self.layoutMargins = UIEdgeInsetsZero;
|
|
|
|
self.axis = UILayoutConstraintAxisHorizontal;
|
|
self.spacing = self.hSpacing;
|
|
self.alignment = UIStackViewAlignmentCenter;
|
|
self.distribution = UIStackViewDistributionEqualSpacing;
|
|
|
|
UIStackView *leftStackView = [UIStackView new];
|
|
leftStackView.axis = UILayoutConstraintAxisHorizontal;
|
|
leftStackView.spacing = self.hSpacing;
|
|
leftStackView.alignment = UIStackViewAlignmentCenter;
|
|
[self addArrangedSubview:leftStackView];
|
|
|
|
self.timestampLabel = [UILabel new];
|
|
[leftStackView addArrangedSubview:self.timestampLabel];
|
|
|
|
self.messageTimerView = [OWSMessageTimerView new];
|
|
[self.messageTimerView setContentHuggingHigh];
|
|
[leftStackView addArrangedSubview:self.messageTimerView];
|
|
|
|
self.statusIndicatorImageView = [UIImageView new];
|
|
[self.statusIndicatorImageView setContentHuggingHigh];
|
|
[self addArrangedSubview:self.statusIndicatorImageView];
|
|
|
|
self.userInteractionEnabled = NO;
|
|
}
|
|
|
|
- (void)configureFonts
|
|
{
|
|
self.timestampLabel.font = UIFont.ows_dynamicTypeCaption1Font;
|
|
}
|
|
|
|
- (CGFloat)hSpacing
|
|
{
|
|
// TODO: Review constant.
|
|
return 8.f;
|
|
}
|
|
|
|
- (CGFloat)maxImageWidth
|
|
{
|
|
return 18.f;
|
|
}
|
|
|
|
- (CGFloat)imageHeight
|
|
{
|
|
return 12.f;
|
|
}
|
|
|
|
#pragma mark - Load
|
|
|
|
- (void)configureWithConversationViewItem:(ConversationViewItem *)viewItem
|
|
isOverlayingMedia:(BOOL)isOverlayingMedia
|
|
conversationStyle:(ConversationStyle *)conversationStyle
|
|
isIncoming:(BOOL)isIncoming
|
|
{
|
|
OWSAssert(viewItem);
|
|
OWSAssert(conversationStyle);
|
|
|
|
[self configureLabelsWithConversationViewItem:viewItem];
|
|
|
|
UIColor *textColor;
|
|
if (isOverlayingMedia) {
|
|
textColor = [UIColor whiteColor];
|
|
} else {
|
|
textColor = [conversationStyle bubbleSecondaryTextColorWithIsIncoming:isIncoming];
|
|
}
|
|
self.timestampLabel.textColor = textColor;
|
|
|
|
if ([self isDisappearingMessage:viewItem]) {
|
|
TSMessage *message = (TSMessage *)viewItem.interaction;
|
|
uint64_t expirationTimestamp = message.expiresAt;
|
|
uint32_t expiresInSeconds = message.expiresInSeconds;
|
|
[self.messageTimerView configureWithExpirationTimestamp:expirationTimestamp
|
|
initialDurationSeconds:expiresInSeconds
|
|
tintColor:textColor];
|
|
self.messageTimerView.hidden = NO;
|
|
} else {
|
|
self.messageTimerView.hidden = YES;
|
|
}
|
|
|
|
if (viewItem.interaction.interactionType == OWSInteractionType_OutgoingMessage) {
|
|
TSOutgoingMessage *outgoingMessage = (TSOutgoingMessage *)viewItem.interaction;
|
|
|
|
UIImage *_Nullable statusIndicatorImage = nil;
|
|
MessageReceiptStatus messageStatus =
|
|
[MessageRecipientStatusUtils recipientStatusWithOutgoingMessage:outgoingMessage];
|
|
switch (messageStatus) {
|
|
case MessageReceiptStatusUploading:
|
|
case MessageReceiptStatusSending:
|
|
statusIndicatorImage = [UIImage imageNamed:@"message_status_sending"];
|
|
[self animateSpinningIcon];
|
|
break;
|
|
case MessageReceiptStatusSent:
|
|
case MessageReceiptStatusSkipped:
|
|
statusIndicatorImage = [UIImage imageNamed:@"message_status_sent"];
|
|
break;
|
|
case MessageReceiptStatusDelivered:
|
|
statusIndicatorImage = [UIImage imageNamed:@"message_status_delivered"];
|
|
break;
|
|
case MessageReceiptStatusRead:
|
|
statusIndicatorImage = [UIImage imageNamed:@"message_status_read"];
|
|
break;
|
|
case MessageReceiptStatusFailed:
|
|
// No status indicator icon.
|
|
break;
|
|
}
|
|
|
|
if (statusIndicatorImage) {
|
|
OWSAssert(statusIndicatorImage.size.width <= self.maxImageWidth);
|
|
self.statusIndicatorImageView.image =
|
|
[statusIndicatorImage imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate];
|
|
self.statusIndicatorImageView.tintColor = textColor;
|
|
self.statusIndicatorImageView.hidden = NO;
|
|
} else {
|
|
self.statusIndicatorImageView.image = nil;
|
|
self.statusIndicatorImageView.hidden = YES;
|
|
}
|
|
} else {
|
|
self.statusIndicatorImageView.image = nil;
|
|
self.statusIndicatorImageView.hidden = YES;
|
|
}
|
|
}
|
|
|
|
- (void)animateSpinningIcon
|
|
{
|
|
CABasicAnimation *animation;
|
|
animation = [CABasicAnimation animationWithKeyPath:@"transform.rotation.z"];
|
|
animation.toValue = @(M_PI * 2.0);
|
|
const CGFloat kPeriodSeconds = 1.f;
|
|
animation.duration = kPeriodSeconds;
|
|
animation.cumulative = YES;
|
|
animation.repeatCount = HUGE_VALF;
|
|
|
|
[self.statusIndicatorImageView.layer addAnimation:animation forKey:@"animation"];
|
|
}
|
|
|
|
- (BOOL)isFailedOutgoingMessage:(ConversationViewItem *)viewItem
|
|
{
|
|
OWSAssert(viewItem);
|
|
|
|
if (viewItem.interaction.interactionType != OWSInteractionType_OutgoingMessage) {
|
|
return NO;
|
|
}
|
|
|
|
TSOutgoingMessage *outgoingMessage = (TSOutgoingMessage *)viewItem.interaction;
|
|
MessageReceiptStatus messageStatus =
|
|
[MessageRecipientStatusUtils recipientStatusWithOutgoingMessage:outgoingMessage];
|
|
return messageStatus == MessageReceiptStatusFailed;
|
|
}
|
|
|
|
- (BOOL)isDisappearingMessage:(ConversationViewItem *)viewItem
|
|
{
|
|
OWSAssert(viewItem);
|
|
|
|
if (viewItem.interaction.interactionType != OWSInteractionType_OutgoingMessage
|
|
&& viewItem.interaction.interactionType != OWSInteractionType_IncomingMessage) {
|
|
return NO;
|
|
}
|
|
|
|
TSMessage *message = (TSMessage *)viewItem.interaction;
|
|
return message.isExpiringMessage;
|
|
}
|
|
|
|
- (void)configureLabelsWithConversationViewItem:(ConversationViewItem *)viewItem
|
|
{
|
|
OWSAssert(viewItem);
|
|
|
|
[self configureFonts];
|
|
|
|
NSString *timestampLabelText;
|
|
if ([self isFailedOutgoingMessage:viewItem]) {
|
|
timestampLabelText
|
|
= NSLocalizedString(@"MESSAGE_STATUS_SEND_FAILED", @"Label indicating that a message failed to send.");
|
|
} else {
|
|
timestampLabelText = [DateUtil formatMessageTimestamp:viewItem.interaction.timestamp];
|
|
}
|
|
|
|
self.timestampLabel.text = timestampLabelText.localizedUppercaseString;
|
|
}
|
|
|
|
- (CGSize)measureWithConversationViewItem:(ConversationViewItem *)viewItem
|
|
{
|
|
OWSAssert(viewItem);
|
|
|
|
[self configureLabelsWithConversationViewItem:viewItem];
|
|
|
|
CGSize result = CGSizeZero;
|
|
result.height = MAX(self.timestampLabel.font.lineHeight, self.imageHeight);
|
|
|
|
// Measure the actual current width, to be safe.
|
|
CGFloat timestampLabelWidth = [self.timestampLabel sizeThatFits:CGSizeZero].width;
|
|
|
|
// Measuring the timestamp label's width is non-trivial since its
|
|
// contents can be relative the current time. We avoid having
|
|
// message bubbles' "visually vibrate" as their timestamp labels
|
|
// vary in width. So we try to leave enough space for all possible
|
|
// contents of this label _for the first hour of its lifetime_, when
|
|
// the timestamp is particularly volatile.
|
|
if ([DateUtil isTimestampFromLastHour:viewItem.interaction.timestamp]) {
|
|
// Measure the "now" case.
|
|
self.timestampLabel.text = [DateUtil exemplaryNowTimeFormat];
|
|
timestampLabelWidth = MAX(timestampLabelWidth, [self.timestampLabel sizeThatFits:CGSizeZero].width);
|
|
// Measure the "relative time" case.
|
|
// Since this case varies with time, we multiply to leave
|
|
// space for the worst case (whose exact value, due to localization,
|
|
// is unpredictable).
|
|
self.timestampLabel.text = [DateUtil exemplaryMinutesTimeFormat];
|
|
timestampLabelWidth = MAX(timestampLabelWidth,
|
|
[self.timestampLabel sizeThatFits:CGSizeZero].width + self.timestampLabel.font.lineHeight * 0.5f);
|
|
|
|
// Re-configure the labels with the current appropriate value in case
|
|
// we are configuring this view for display.
|
|
[self configureLabelsWithConversationViewItem:viewItem];
|
|
}
|
|
|
|
result.width = timestampLabelWidth;
|
|
if (viewItem.interaction.interactionType == OWSInteractionType_OutgoingMessage) {
|
|
if (![self isFailedOutgoingMessage:viewItem]) {
|
|
result.width += (self.maxImageWidth + self.hSpacing);
|
|
}
|
|
}
|
|
|
|
if ([self isDisappearingMessage:viewItem]) {
|
|
result.width += ([OWSMessageTimerView measureSize].width + self.hSpacing);
|
|
}
|
|
|
|
return CGSizeCeil(result);
|
|
}
|
|
|
|
- (nullable NSString *)messageStatusTextForConversationViewItem:(ConversationViewItem *)viewItem
|
|
{
|
|
OWSAssert(viewItem);
|
|
if (viewItem.interaction.interactionType != OWSInteractionType_OutgoingMessage) {
|
|
return nil;
|
|
}
|
|
|
|
TSOutgoingMessage *outgoingMessage = (TSOutgoingMessage *)viewItem.interaction;
|
|
NSString *statusMessage = [MessageRecipientStatusUtils receiptMessageWithOutgoingMessage:outgoingMessage];
|
|
return statusMessage;
|
|
}
|
|
|
|
- (void)prepareForReuse
|
|
{
|
|
[self.statusIndicatorImageView.layer removeAllAnimations];
|
|
|
|
[self.messageTimerView prepareForReuse];
|
|
}
|
|
|
|
@end
|
|
|
|
NS_ASSUME_NONNULL_END
|