This commit is contained in:
启星
2025-08-08 10:49:36 +08:00
parent 6400cf78bb
commit b5ce3d580a
8780 changed files with 978183 additions and 0 deletions

View File

@@ -0,0 +1,21 @@
//
// TUISearchResultCell.h
// Pods
//
// Created by harvy on 2020/12/24.
// Copyright © 2023 Tencent. All rights reserved.
//
#import <UIKit/UIKit.h>
@class TUISearchResultCellModel;
NS_ASSUME_NONNULL_BEGIN
@interface TUISearchResultCell : UITableViewCell
@property(nonatomic, strong) UIImageView *avatarView;
@property(nonatomic, strong) UILabel *title_label;
- (void)fillWithData:(TUISearchResultCellModel *)cellModel;
@end
NS_ASSUME_NONNULL_END

View File

@@ -0,0 +1,263 @@
//
// 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