264 lines
12 KiB
Mathematica
264 lines
12 KiB
Mathematica
|
|
//
|
|||
|
|
// TUISearchResultCell.m
|
|||
|
|
// Pods
|
|||
|
|
//
|
|||
|
|
// Created by harvy on 2020/12/24.
|
|||
|
|
// Copyright © 2023 Tencent. All rights reserved.
|
|||
|
|
//
|
|||
|
|
|
|||
|
|
#import "TUISearchResultCell.h"
|
|||
|
|
#import <TIMCommon/TIMCommonModel.h>
|
|||
|
|
#import <TIMCommon/TIMDefine.h>
|
|||
|
|
#import <TUICore/TUIThemeManager.h>
|
|||
|
|
#import "TUISearchResultCellModel.h"
|
|||
|
|
@interface TUISearchResultCell ()
|
|||
|
|
@property(nonatomic, strong) UILabel *detail_title;
|
|||
|
|
@property(nonatomic, strong) UIView *separtorView;
|
|||
|
|
@property(nonatomic, strong) TUISearchResultCellModel *cellModel;
|
|||
|
|
|
|||
|
|
@end
|
|||
|
|
|
|||
|
|
@implementation TUISearchResultCell
|
|||
|
|
|
|||
|
|
- (instancetype)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier {
|
|||
|
|
if (self = [super initWithStyle:style reuseIdentifier:reuseIdentifier]) {
|
|||
|
|
[self setupViews];
|
|||
|
|
}
|
|||
|
|
return self;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
- (void)setupViews {
|
|||
|
|
self.contentView.backgroundColor = TIMCommonDynamicColor(@"form_bg_color", @"#FFFFFF");
|
|||
|
|
|
|||
|
|
_avatarView = [[UIImageView alloc] init];
|
|||
|
|
[self.contentView addSubview:_avatarView];
|
|||
|
|
|
|||
|
|
_title_label = [[UILabel alloc] init];
|
|||
|
|
_title_label.text = @"";
|
|||
|
|
_title_label.textColor = TIMCommonDynamicColor(@"form_title_color", @"#000000");
|
|||
|
|
_title_label.font = [UIFont systemFontOfSize:14.0];
|
|||
|
|
_title_label.rtlAlignment = TUITextRTLAlignmentLeading;
|
|||
|
|
[self.contentView addSubview:_title_label];
|
|||
|
|
|
|||
|
|
_detail_title = [[UILabel alloc] init];
|
|||
|
|
_detail_title.text = @"";
|
|||
|
|
_detail_title.textColor = TIMCommonDynamicColor(@"form_subtitle_color", @"#888888");
|
|||
|
|
_detail_title.font = [UIFont systemFontOfSize:12.0];
|
|||
|
|
_detail_title.rtlAlignment = TUITextRTLAlignmentLeading;
|
|||
|
|
[self.contentView addSubview:_detail_title];
|
|||
|
|
|
|||
|
|
_separtorView = [[UIView alloc] init];
|
|||
|
|
_separtorView.backgroundColor = TIMCommonDynamicColor(@"separator_color", @"#DBDBDB");
|
|||
|
|
[self.contentView addSubview:_separtorView];
|
|||
|
|
|
|||
|
|
self.selectionStyle = UITableViewCellSelectionStyleNone;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
- (void)layoutSubviews {
|
|||
|
|
[super layoutSubviews];
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
- (void)fillWithData:(TUISearchResultCellModel *)cellModel {
|
|||
|
|
self.cellModel = cellModel;
|
|||
|
|
|
|||
|
|
self.title_label.text = nil;
|
|||
|
|
self.title_label.attributedText = nil;
|
|||
|
|
self.detail_title.text = nil;
|
|||
|
|
self.detail_title.attributedText = nil;
|
|||
|
|
|
|||
|
|
self.title_label.text = cellModel.title;
|
|||
|
|
if (cellModel.titleAttributeString) {
|
|||
|
|
self.title_label.attributedText = cellModel.titleAttributeString;
|
|||
|
|
}
|
|||
|
|
self.detail_title.text = cellModel.details;
|
|||
|
|
if (cellModel.detailsAttributeString) {
|
|||
|
|
self.detail_title.attributedText = cellModel.detailsAttributeString;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/**
|
|||
|
|
* Setup default avatar
|
|||
|
|
*/
|
|||
|
|
if (cellModel.groupID.length > 0) {
|
|||
|
|
/**
|
|||
|
|
* If it is a group, change the group default avatar to the last used avatar
|
|||
|
|
*/
|
|||
|
|
UIImage *avatar = nil;
|
|||
|
|
if (TUIConfig.defaultConfig.enableGroupGridAvatar) {
|
|||
|
|
NSString *key = [NSString stringWithFormat:@"TUIConversationLastGroupMember_%@", cellModel.groupID];
|
|||
|
|
NSInteger member = [NSUserDefaults.standardUserDefaults integerForKey:key];
|
|||
|
|
avatar = [TUIGroupAvatar getCacheAvatarForGroup:cellModel.groupID number:(UInt32)member];
|
|||
|
|
}
|
|||
|
|
cellModel.avatarImage = avatar ? avatar : DefaultGroupAvatarImageByGroupType(cellModel.groupType);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
@weakify(self);
|
|||
|
|
[[RACObserve(cellModel, avatarUrl) takeUntil:self.rac_prepareForReuseSignal] subscribeNext:^(NSString *faceUrl) {
|
|||
|
|
@strongify(self);
|
|||
|
|
if (cellModel.groupID.length > 0) {
|
|||
|
|
/**
|
|||
|
|
* Group avatar
|
|||
|
|
*/
|
|||
|
|
if (IS_NOT_EMPTY_NSSTRING(faceUrl)) {
|
|||
|
|
/**
|
|||
|
|
* The group avatar has been manually set externally
|
|||
|
|
*/
|
|||
|
|
[self.avatarView sd_setImageWithURL:[NSURL URLWithString:faceUrl] placeholderImage:self.cellModel.avatarImage];
|
|||
|
|
} else {
|
|||
|
|
/**
|
|||
|
|
* The group avatar has not been set externally. If the synthetic avatar is allowed, the synthetic avatar will be used; otherwise, the default
|
|||
|
|
* avatar will be used.
|
|||
|
|
*/
|
|||
|
|
if (TUIConfig.defaultConfig.enableGroupGridAvatar) {
|
|||
|
|
/**
|
|||
|
|
* If the synthetic avatar is allowed, the synthetic avatar will be used
|
|||
|
|
* 1. Asynchronously obtain the cached synthetic avatar according to the number of group members
|
|||
|
|
* 2. If the cache is hit, use the cached synthetic avatar directly
|
|||
|
|
* 3. If the cache is not hit, recompose a new avatar
|
|||
|
|
*
|
|||
|
|
* Note:
|
|||
|
|
* 1. Since "asynchronously obtaining cached avatars" and "synthesizing avatars" take a long time, it is easy to cause cell reuse problems, so
|
|||
|
|
* it is necessary to confirm whether to assign values directly according to groupID.
|
|||
|
|
* 2. Use SDWebImage to implement placeholder, because SDWebImage has already dealt with the problem of cell reuse
|
|||
|
|
*/
|
|||
|
|
|
|||
|
|
// 1. Obtain group avatar from cache
|
|||
|
|
|
|||
|
|
// fix: Th getCacheGroupAvatar needs to request the
|
|||
|
|
// network. When the network is disconnected, since the headImageView is not set, the current conversation sends a message, the conversation
|
|||
|
|
// is moved up, and the avatar of the first conversation is reused, resulting in confusion of the avatar.
|
|||
|
|
[self.avatarView sd_setImageWithURL:nil placeholderImage:cellModel.avatarImage];
|
|||
|
|
[TUIGroupAvatar
|
|||
|
|
getCacheGroupAvatar:cellModel.groupID
|
|||
|
|
callback:^(UIImage *avatar, NSString *groupID) {
|
|||
|
|
@strongify(self);
|
|||
|
|
if ([groupID isEqualToString:self.cellModel.groupID]) {
|
|||
|
|
// 1.1 When the callback is invoked, the cell is not reused
|
|||
|
|
|
|||
|
|
if (avatar != nil) {
|
|||
|
|
// 2. Hit the cache and assign directly
|
|||
|
|
[self.avatarView sd_setImageWithURL:nil placeholderImage:avatar];
|
|||
|
|
} else {
|
|||
|
|
// 3. Synthesize new avatars asynchronously without hitting cache
|
|||
|
|
|
|||
|
|
[self.avatarView sd_setImageWithURL:nil placeholderImage:cellModel.avatarImage];
|
|||
|
|
[TUIGroupAvatar
|
|||
|
|
fetchGroupAvatars:cellModel.groupID
|
|||
|
|
placeholder:cellModel.avatarImage
|
|||
|
|
callback:^(BOOL success, UIImage *image, NSString *groupID) {
|
|||
|
|
@strongify(self);
|
|||
|
|
if ([groupID isEqualToString:self.cellModel.groupID]) {
|
|||
|
|
// When the callback is invoked, the cell is not reused
|
|||
|
|
[self.avatarView
|
|||
|
|
sd_setImageWithURL:nil
|
|||
|
|
placeholderImage:success ? image
|
|||
|
|
: DefaultGroupAvatarImageByGroupType(self.cellModel.groupType)];
|
|||
|
|
} else {
|
|||
|
|
// callback, When the callback is invoked, the cell has been reused to other groupIDs.
|
|||
|
|
// Since a new callback will be triggered when the new groupID synthesizes new avatar, it is
|
|||
|
|
// ignored here
|
|||
|
|
}
|
|||
|
|
}];
|
|||
|
|
}
|
|||
|
|
} else {
|
|||
|
|
// 1.2 When the callback is invoked, the cell has been reused to other groupIDs. Since a new callback will be triggered
|
|||
|
|
// when the new groupID gets the cache, it is ignored here
|
|||
|
|
}
|
|||
|
|
}];
|
|||
|
|
} else {
|
|||
|
|
/**
|
|||
|
|
* Synthetic avatars are not allowed, use the default avatar directly
|
|||
|
|
*/
|
|||
|
|
[self.avatarView sd_setImageWithURL:nil placeholderImage:cellModel.avatarImage];
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
} else {
|
|||
|
|
/**
|
|||
|
|
* Personal avatar
|
|||
|
|
*/
|
|||
|
|
[self.avatarView sd_setImageWithURL:[NSURL URLWithString:faceUrl] placeholderImage:self.cellModel.avatarImage];
|
|||
|
|
}
|
|||
|
|
}];
|
|||
|
|
|
|||
|
|
// tell constraints they need updating
|
|||
|
|
[self setNeedsUpdateConstraints];
|
|||
|
|
|
|||
|
|
// update constraints now so we can animate the change
|
|||
|
|
[self updateConstraintsIfNeeded];
|
|||
|
|
|
|||
|
|
[self layoutIfNeeded];
|
|||
|
|
}
|
|||
|
|
+ (BOOL)requiresConstraintBasedLayout {
|
|||
|
|
return YES;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// this is Apple's recommended place for adding/updating constraints
|
|||
|
|
- (void)updateConstraints {
|
|||
|
|
|
|||
|
|
[super updateConstraints];
|
|||
|
|
|
|||
|
|
CGSize headSize = CGSizeMake(kScale390(40), kScale390(40));
|
|||
|
|
|
|||
|
|
[self.avatarView mas_remakeConstraints:^(MASConstraintMaker *make) {
|
|||
|
|
make.size.mas_equalTo(headSize);
|
|||
|
|
make.centerY.mas_equalTo(self.contentView.mas_centerY);
|
|||
|
|
make.leading.mas_equalTo(kScale390(10));
|
|||
|
|
}];
|
|||
|
|
|
|||
|
|
if ([TUIConfig defaultConfig].avatarType == TAvatarTypeRounded) {
|
|||
|
|
self.avatarView.layer.masksToBounds = YES;
|
|||
|
|
self.avatarView.layer.cornerRadius = headSize.height / 2;
|
|||
|
|
} else if ([TUIConfig defaultConfig].avatarType == TAvatarTypeRadiusCorner) {
|
|||
|
|
self.avatarView.layer.masksToBounds = YES;
|
|||
|
|
self.avatarView.layer.cornerRadius = [TUIConfig defaultConfig].avatarCornerRadius;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
NSString *title = self.title_label.text;
|
|||
|
|
if (title.length == 0) {
|
|||
|
|
title = self.title_label.attributedText.string;
|
|||
|
|
}
|
|||
|
|
NSString *detail = self.detail_title.text;
|
|||
|
|
if (detail.length == 0) {
|
|||
|
|
detail = self.detail_title.attributedText.string;
|
|||
|
|
}
|
|||
|
|
[self.title_label sizeToFit];
|
|||
|
|
[self.detail_title sizeToFit];
|
|||
|
|
if (title.length && self.detail_title.text.length) {
|
|||
|
|
[self.title_label mas_remakeConstraints:^(MASConstraintMaker *make) {
|
|||
|
|
make.top.mas_equalTo(self.avatarView.mas_top);
|
|||
|
|
make.leading.mas_equalTo(self.avatarView.mas_trailing).mas_offset(10);
|
|||
|
|
make.trailing.mas_equalTo(self.contentView.mas_trailing).mas_offset(-10);
|
|||
|
|
make.height.mas_greaterThanOrEqualTo(self.title_label.frame.size.height);
|
|||
|
|
}];
|
|||
|
|
|
|||
|
|
[self.detail_title mas_remakeConstraints:^(MASConstraintMaker *make) {
|
|||
|
|
make.bottom.mas_equalTo(self.avatarView.mas_bottom);
|
|||
|
|
make.leading.mas_equalTo(self.avatarView.mas_trailing).mas_offset(10);
|
|||
|
|
make.trailing.mas_equalTo(self.contentView.mas_trailing).mas_offset(-10);
|
|||
|
|
make.height.mas_greaterThanOrEqualTo(self.detail_title.frame.size.height);
|
|||
|
|
}];
|
|||
|
|
} else {
|
|||
|
|
[self.title_label mas_remakeConstraints:^(MASConstraintMaker *make) {
|
|||
|
|
make.centerY.mas_equalTo(self.avatarView.mas_centerY);
|
|||
|
|
make.leading.mas_equalTo(self.avatarView.mas_trailing).mas_offset(10);
|
|||
|
|
make.trailing.mas_equalTo(self.contentView.mas_trailing).mas_offset(-10);
|
|||
|
|
make.height.mas_greaterThanOrEqualTo(self.title_label.frame.size.height);
|
|||
|
|
}];
|
|||
|
|
|
|||
|
|
[self.detail_title mas_remakeConstraints:^(MASConstraintMaker *make) {
|
|||
|
|
make.centerY.mas_equalTo(self.avatarView.mas_centerY);
|
|||
|
|
make.leading.mas_equalTo(self.avatarView.mas_trailing).mas_offset(10);
|
|||
|
|
make.trailing.mas_equalTo(self.contentView.mas_trailing).mas_offset(-10);
|
|||
|
|
make.height.mas_greaterThanOrEqualTo(self.detail_title.frame.size.height);
|
|||
|
|
}];
|
|||
|
|
}
|
|||
|
|
[self.separtorView mas_remakeConstraints:^(MASConstraintMaker *make) {
|
|||
|
|
make.leading.mas_equalTo(self.avatarView.mas_trailing);
|
|||
|
|
make.bottom.mas_equalTo(self.contentView.mas_bottom).mas_offset(-1);
|
|||
|
|
make.width.mas_equalTo(self.contentView);
|
|||
|
|
make.height.mas_equalTo(1);
|
|||
|
|
}];
|
|||
|
|
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
@end
|