Files
keyboard/keyBoard/Class/AiTalk/V/Chat/KBPersonaChatCell.m

990 lines
34 KiB
Mathematica
Raw Normal View History

2026-01-26 16:53:41 +08:00
//
// KBPersonaChatCell.m
// keyBoard
//
// Created by Kiro on 2026/1/26.
//
#import "KBPersonaChatCell.h"
#import "KBAiChatMessage.h"
2026-01-26 18:17:02 +08:00
#import "KBChatHistoryPageModel.h"
#import "AiVM.h"
2026-01-27 18:53:19 +08:00
#import "KBImagePositionButton.h"
2026-01-27 21:32:52 +08:00
#import "KBAICommentView.h"
2026-01-28 18:58:30 +08:00
#import "KBAIChatMessageCacheManager.h"
2026-01-26 16:53:41 +08:00
#import <Masonry/Masonry.h>
#import <SDWebImage/SDWebImage.h>
2026-01-27 21:32:52 +08:00
#import <LSTPopView/LSTPopView.h>
2026-01-29 16:03:21 +08:00
#import "AIPersonInfoVC.h"
2026-01-28 19:31:27 +08:00
///
static NSString * const KBChatSessionDidResetNotification = @"KBChatSessionDidResetNotification";
2026-01-26 20:36:51 +08:00
@interface KBPersonaChatCell () <KBChatTableViewDelegate>
2026-01-26 16:53:41 +08:00
///
@property (nonatomic, strong) UIImageView *backgroundImageView;
///
@property (nonatomic, strong) UIImageView *avatarImageView;
///
@property (nonatomic, strong) UILabel *nameLabel;
///
@property (nonatomic, strong) UILabel *openingLabel;
2026-01-28 17:21:19 +08:00
2026-01-26 16:53:41 +08:00
///
@property (nonatomic, strong) NSMutableArray<KBAiChatMessage *> *messages;
///
@property (nonatomic, assign) BOOL hasLoadedData;
2026-01-26 18:17:02 +08:00
///
@property (nonatomic, assign) BOOL isLoading;
2026-01-30 21:24:17 +08:00
@property (nonatomic, assign) BOOL canTriggerLoadMore;
2026-01-26 18:17:02 +08:00
///
@property (nonatomic, assign) NSInteger currentPage;
///
@property (nonatomic, assign) BOOL hasMoreHistory;
/// AiVM
@property (nonatomic, strong) AiVM *aiVM;
2026-01-27 18:53:19 +08:00
///
@property (nonatomic, strong) KBImagePositionButton *commentButton;
///
@property (nonatomic, strong) KBImagePositionButton *likeButton;
2026-01-27 21:32:52 +08:00
///
@property (nonatomic, weak) LSTPopView *popView;
2026-01-31 23:17:58 +08:00
@property (nonatomic, strong) NSMutableDictionary<NSString *, KBAiChatMessage *> *pendingAssistantMessages;
2026-01-26 16:53:41 +08:00
@end
@implementation KBPersonaChatCell
#pragma mark - Lifecycle
- (instancetype)initWithFrame:(CGRect)frame {
if (self = [super initWithFrame:frame]) {
[self setupUI];
2026-01-28 19:31:27 +08:00
//
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(handleChatSessionReset:)
name:KBChatSessionDidResetNotification
object:nil];
2026-01-26 16:53:41 +08:00
}
return self;
}
2026-01-28 19:31:27 +08:00
- (void)dealloc {
[[NSNotificationCenter defaultCenter] removeObserver:self];
}
2026-01-28 18:58:30 +08:00
/// Cell
2026-01-27 17:03:16 +08:00
- (void)prepareForReuse {
[super prepareForReuse];
//
[self.chatView stopPlayingAudio];
2026-01-28 18:58:30 +08:00
// hasLoadedData
2026-01-27 17:03:16 +08:00
self.isLoading = NO;
2026-01-30 21:24:17 +08:00
self.canTriggerLoadMore = YES;
2026-01-31 23:17:58 +08:00
[self.pendingAssistantMessages removeAllObjects];
2026-01-28 18:58:30 +08:00
// self.hasLoadedData = NO;
// Cell
2026-01-27 17:03:16 +08:00
}
2026-01-26 16:53:41 +08:00
#pragma mark - 1
- (void)setupUI {
//
[self.contentView addSubview:self.backgroundImageView];
[self.backgroundImageView mas_makeConstraints:^(MASConstraintMaker *make) {
make.edges.equalTo(self.contentView);
}];
//
UIView *maskView = [[UIView alloc] init];
maskView.backgroundColor = [[UIColor blackColor] colorWithAlphaComponent:0.3];
[self.contentView addSubview:maskView];
[maskView mas_makeConstraints:^(MASConstraintMaker *make) {
make.edges.equalTo(self.contentView);
}];
2026-01-27 17:49:45 +08:00
//
[self.contentView addSubview:self.openingLabel];
[self.openingLabel mas_makeConstraints:^(MASConstraintMaker *make) {
make.top.equalTo(self.contentView).offset(KB_NAV_TOTAL_HEIGHT);
make.left.equalTo(self.contentView).offset(40);
make.right.equalTo(self.contentView).offset(-40);
}];
2026-01-26 16:53:41 +08:00
//
[self.contentView addSubview:self.avatarImageView];
[self.avatarImageView mas_makeConstraints:^(MASConstraintMaker *make) {
2026-01-27 17:49:45 +08:00
make.bottom.equalTo(self.contentView).offset(-KB_TABBAR_HEIGHT - 50 - 20);
make.left.equalTo(self.contentView).offset(20);
make.size.mas_equalTo(CGSizeMake(54, 54));
2026-01-26 16:53:41 +08:00
}];
//
[self.contentView addSubview:self.nameLabel];
[self.nameLabel mas_makeConstraints:^(MASConstraintMaker *make) {
2026-01-27 17:49:45 +08:00
make.left.equalTo(self.avatarImageView.mas_right).offset(5);
make.centerY.equalTo(self.avatarImageView);
2026-01-26 16:53:41 +08:00
}];
2026-01-27 18:53:19 +08:00
//
[self.contentView addSubview:self.commentButton];
[self.commentButton mas_makeConstraints:^(MASConstraintMaker *make) {
make.right.equalTo(self.contentView).offset(-20);
make.centerY.equalTo(self.avatarImageView);
make.width.mas_equalTo(40);
make.height.mas_equalTo(50);
}];
// 20px
[self.contentView addSubview:self.likeButton];
[self.likeButton mas_makeConstraints:^(MASConstraintMaker *make) {
make.right.equalTo(self.commentButton.mas_left).offset(-20);
make.centerY.equalTo(self.avatarImageView);
make.width.mas_equalTo(40);
make.height.mas_equalTo(50);
}];
2026-01-26 16:53:41 +08:00
//
2026-01-26 20:36:51 +08:00
[self.contentView addSubview:self.chatView];
[self.chatView mas_makeConstraints:^(MASConstraintMaker *make) {
2026-01-27 17:49:45 +08:00
make.top.equalTo(self.contentView).offset(KB_NAV_TOTAL_HEIGHT);
make.left.right.equalTo(self.contentView);
make.bottom.equalTo(self.avatarImageView.mas_top).offset(-10);
2026-01-26 16:53:41 +08:00
}];
2026-01-27 17:49:45 +08:00
2026-01-26 16:53:41 +08:00
}
#pragma mark - Setter
- (void)setPersona:(KBPersonaModel *)persona {
_persona = persona;
//
2026-01-26 18:17:02 +08:00
self.isLoading = NO;
2026-01-30 21:24:17 +08:00
self.canTriggerLoadMore = YES;
2026-01-26 18:17:02 +08:00
self.currentPage = 1;
self.hasMoreHistory = YES;
2026-01-31 23:17:58 +08:00
[self.pendingAssistantMessages removeAllObjects];
2026-01-28 18:58:30 +08:00
//
// NSArray *cachedMessages = [[KBAIChatMessageCacheManager shared] messagesForCompanionId:persona.personaId];
// if (cachedMessages.count > 0) {
// self.messages = [cachedMessages mutableCopy];
// self.hasLoadedData = YES;
// NSLog(@"[Cell] ✅ 从缓存加载personaId=%ld, 消息数=%ld", (long)persona.personaId, (long)cachedMessages.count);
// } else {
2026-01-28 18:58:30 +08:00
self.messages = [NSMutableArray array];
self.hasLoadedData = NO;
NSLog(@"[Cell] ⚠️ 缓存已禁用personaId=%ld, 需要请求数据", (long)persona.personaId);
// }
2026-01-26 16:53:41 +08:00
// UI
[self.backgroundImageView sd_setImageWithURL:[NSURL URLWithString:persona.coverImageUrl]
placeholderImage:[UIImage imageNamed:@"placeholder_bg"]];
[self.avatarImageView sd_setImageWithURL:[NSURL URLWithString:persona.avatarUrl]
placeholderImage:[UIImage imageNamed:@"placeholder_avatar"]];
self.nameLabel.text = persona.name;
self.openingLabel.text = persona.shortDesc.length > 0 ? persona.shortDesc : persona.introText;
2026-01-27 17:03:16 +08:00
//
[self.chatView stopPlayingAudio];
2026-01-28 18:58:30 +08:00
2026-01-28 20:18:18 +08:00
//
[self ensureOpeningMessageAtTop];
NSLog(@"[KBPersonaChatCell] ========== setPersona 调试 ==========");
NSLog(@"[KBPersonaChatCell] personaId: %ld", (long)persona.personaId);
NSLog(@"[KBPersonaChatCell] messages.count: %ld", (long)self.messages.count);
NSLog(@"[KBPersonaChatCell] chatView.frame: %@", NSStringFromCGRect(self.chatView.frame));
NSLog(@"[KBPersonaChatCell] contentView.frame: %@", NSStringFromCGRect(self.contentView.frame));
2026-01-28 20:18:18 +08:00
//
2026-01-28 18:58:30 +08:00
if (self.messages.count > 0) {
2026-01-28 20:18:18 +08:00
//
[[KBAIChatMessageCacheManager shared] saveMessages:self.messages
forCompanionId:persona.personaId];
2026-01-28 18:58:30 +08:00
[self.chatView reloadWithMessages:self.messages
2026-01-28 19:31:27 +08:00
keepOffset:NO
scrollToBottom:YES];
2026-01-28 18:58:30 +08:00
} else {
[self.chatView clearMessages];
}
NSLog(@"[KBPersonaChatCell] ========== setPersona 结束 ==========");
2026-01-27 18:53:19 +08:00
[self.commentButton setTitle:persona.commentCount forState:UIControlStateNormal];
[self.likeButton setTitle:persona.likeCount forState:UIControlStateNormal];
self.likeButton.selected = persona.liked;
2026-01-26 16:53:41 +08:00
}
#pragma mark - 2
- (void)preloadDataIfNeeded {
2026-01-26 18:17:02 +08:00
if (self.hasLoadedData || self.isLoading) {
2026-01-26 16:53:41 +08:00
return;
}
2026-01-26 18:17:02 +08:00
[self loadChatHistory];
}
- (void)loadChatHistory {
if (self.isLoading || !self.hasMoreHistory) {
2026-01-26 20:36:51 +08:00
[self.chatView endLoadMoreWithHasMoreData:self.hasMoreHistory];
2026-01-26 18:17:02 +08:00
return;
}
2026-01-26 16:53:41 +08:00
2026-01-26 18:17:02 +08:00
self.isLoading = YES;
2026-01-26 20:36:51 +08:00
if (self.currentPage == 1) {
[self.chatView resetNoMoreData];
}
2026-01-26 18:17:02 +08:00
// 使 persona.personaId companionId
NSInteger companionId = self.persona.personaId;
__weak typeof(self) weakSelf = self;
[self.aiVM fetchChatHistoryWithCompanionId:companionId
pageNum:self.currentPage
2026-01-31 22:40:50 +08:00
pageSize:10
2026-01-26 18:17:02 +08:00
completion:^(KBChatHistoryPageModel *pageModel, NSError *error) {
2026-01-30 21:24:17 +08:00
__strong typeof(weakSelf) strongSelf = weakSelf;
if (!strongSelf) {
return;
}
2026-01-26 18:17:02 +08:00
if (error) {
NSLog(@"[KBPersonaChatCell] 加载聊天记录失败:%@", error.localizedDescription);
2026-01-30 21:24:17 +08:00
dispatch_async(dispatch_get_main_queue(), ^{
strongSelf.isLoading = NO;
[strongSelf.chatView endLoadMoreWithHasMoreData:strongSelf.hasMoreHistory];
if (strongSelf.currentPage == 1 && strongSelf.persona.introText.length > 0) {
[strongSelf showOpeningMessage];
}
});
2026-01-26 18:17:02 +08:00
return;
}
2026-01-30 21:24:17 +08:00
strongSelf.hasLoadedData = YES;
strongSelf.hasMoreHistory = pageModel.hasMore;
2026-01-26 18:17:02 +08:00
// KBAiChatMessage
NSMutableArray *newMessages = [NSMutableArray array];
for (KBChatHistoryModel *item in pageModel.records) {
KBAiChatMessage *message;
2026-01-26 20:36:51 +08:00
// sender
// sender = 1:
// sender = 2: AI
if (item.sender == KBChatSenderUser) {
//
2026-01-26 18:17:02 +08:00
message = [KBAiChatMessage userMessageWithText:item.content];
2026-01-26 20:36:51 +08:00
} else if (item.sender == KBChatSenderAssistant) {
// AI
message = [KBAiChatMessage assistantMessageWithText:item.content];
2026-01-26 18:17:02 +08:00
} else {
2026-01-26 20:36:51 +08:00
// AI
NSLog(@"[KBPersonaChatCell] 未知的 sender 类型:%ld", (long)item.sender);
2026-01-26 18:17:02 +08:00
message = [KBAiChatMessage assistantMessageWithText:item.content];
}
2026-01-26 20:36:51 +08:00
2026-01-26 18:17:02 +08:00
message.isComplete = YES;
2026-01-26 20:36:51 +08:00
message.needsTypewriterEffect = NO;
2026-01-31 22:40:50 +08:00
[newMessages addObject:message];
// [newMessages insertObject:message atIndex:0];
2026-01-26 18:17:02 +08:00
}
2026-01-28 20:18:18 +08:00
//
// dispatch_async currentPage
2026-01-30 21:24:17 +08:00
NSInteger loadedPage = strongSelf.currentPage;
if (loadedPage == 1) {
2026-01-26 18:17:02 +08:00
//
2026-01-30 21:24:17 +08:00
strongSelf.messages = newMessages;
[strongSelf ensureOpeningMessageAtTop];
2026-01-26 18:17:02 +08:00
} else {
2026-01-31 22:40:50 +08:00
//
2026-01-30 21:24:17 +08:00
[strongSelf ensureOpeningMessageAtTop];
2026-01-28 20:18:18 +08:00
if (newMessages.count > 0) {
2026-01-31 22:40:50 +08:00
if (strongSelf.chatView.inverted) {
NSInteger openingIndex = [strongSelf openingMessageIndexInMessages];
NSUInteger insertIndex = (openingIndex != NSNotFound) ? (NSUInteger)openingIndex : strongSelf.messages.count;
NSIndexSet *indexSet = [NSIndexSet indexSetWithIndexesInRange:NSMakeRange(insertIndex, newMessages.count)];
[strongSelf.messages insertObjects:newMessages atIndexes:indexSet];
} else {
NSUInteger insertIndex = [strongSelf hasOpeningMessageAtTop] ? 1 : 0;
NSIndexSet *indexSet = [NSIndexSet indexSetWithIndexesInRange:NSMakeRange(insertIndex, newMessages.count)];
[strongSelf.messages insertObjects:newMessages atIndexes:indexSet];
}
2026-01-28 20:18:18 +08:00
}
2026-01-26 18:17:02 +08:00
}
// UI
dispatch_async(dispatch_get_main_queue(), ^{
2026-01-30 21:24:17 +08:00
if (loadedPage == 1) {
NSLog(@"[KBPersonaChatCell] 刷新 UI - loadedPage: %ld, keepOffset: 0, scrollToBottom: 1",
(long)loadedPage);
[strongSelf.chatView reloadWithMessages:strongSelf.messages
keepOffset:NO
scrollToBottom:YES];
} else {
2026-01-31 22:40:50 +08:00
if (strongSelf.chatView.inverted) {
NSLog(@"[KBPersonaChatCell] 刷新 UI - loadedPage: %ld, appendHistory", (long)loadedPage);
KBAiChatMessage *openingMessage = [strongSelf openingMessageInMessages];
[strongSelf.chatView appendHistoryMessages:newMessages openingMessage:openingMessage];
} else {
NSLog(@"[KBPersonaChatCell] 刷新 UI - loadedPage: %ld, prependHistory",
(long)loadedPage);
KBAiChatMessage *openingMessage = [strongSelf hasOpeningMessageAtTop] ? strongSelf.messages.firstObject : nil;
[strongSelf.chatView prependHistoryMessages:newMessages openingMessage:openingMessage];
}
}
[strongSelf.chatView endLoadMoreWithHasMoreData:strongSelf.hasMoreHistory];
2026-01-28 18:58:30 +08:00
2026-01-28 20:18:18 +08:00
//
[[KBAIChatMessageCacheManager shared] saveMessages:strongSelf.messages
2026-01-28 18:58:30 +08:00
forCompanionId:companionId];
2026-01-30 21:24:17 +08:00
strongSelf.isLoading = NO;
2026-01-26 18:17:02 +08:00
});
2026-01-30 21:24:17 +08:00
strongSelf.currentPage++;
2026-01-28 18:58:30 +08:00
2026-01-26 18:17:02 +08:00
NSLog(@"[KBPersonaChatCell] 加载成功:第 %ld 页,%ld 条消息,还有更多:%@",
2026-01-30 21:24:17 +08:00
(long)strongSelf.currentPage - 1,
2026-01-26 18:17:02 +08:00
(long)newMessages.count,
pageModel.hasMore ? @"是" : @"否");
}];
}
- (void)loadMoreHistory {
if (!self.hasMoreHistory || self.isLoading) {
2026-01-26 20:36:51 +08:00
[self.chatView endLoadMoreWithHasMoreData:self.hasMoreHistory];
2026-01-26 18:17:02 +08:00
return;
2026-01-26 16:53:41 +08:00
}
2026-01-26 18:17:02 +08:00
[self loadChatHistory];
}
- (void)showOpeningMessage {
//
2026-01-28 20:18:18 +08:00
[self ensureOpeningMessageAtTop];
2026-01-26 18:17:02 +08:00
dispatch_async(dispatch_get_main_queue(), ^{
2026-01-26 20:36:51 +08:00
[self.chatView reloadWithMessages:self.messages
keepOffset:NO
scrollToBottom:YES];
2026-01-26 18:17:02 +08:00
});
2026-01-26 16:53:41 +08:00
}
2026-01-28 20:18:18 +08:00
- (BOOL)hasOpeningMessageAtTop {
if (self.messages.count == 0) {
return NO;
}
2026-01-31 22:40:50 +08:00
if (self.chatView.inverted) {
return [self isOpeningMessage:self.messages.lastObject];
}
2026-01-28 20:18:18 +08:00
return [self isOpeningMessage:self.messages.firstObject];
}
- (BOOL)isOpeningMessage:(KBAiChatMessage *)message {
if (!message) {
return NO;
}
NSString *introText = self.persona.introText ?: @"";
if (introText.length == 0) {
return NO;
}
return (message.type == KBAiChatMessageTypeAssistant) && [message.text isEqualToString:introText];
}
- (void)ensureOpeningMessageAtTop {
NSString *introText = self.persona.introText ?: @"";
if (introText.length == 0) {
return;
}
if (!self.messages) {
self.messages = [NSMutableArray array];
}
if ([self hasOpeningMessageAtTop]) {
return;
}
KBAiChatMessage *openingMsg = [KBAiChatMessage assistantMessageWithText:introText];
openingMsg.isComplete = YES;
openingMsg.needsTypewriterEffect = NO;
2026-01-31 22:40:50 +08:00
if (self.chatView.inverted) {
[self.messages addObject:openingMsg];
} else {
[self.messages insertObject:openingMsg atIndex:0];
}
}
- (nullable KBAiChatMessage *)openingMessageInMessages {
NSInteger index = [self openingMessageIndexInMessages];
if (index == NSNotFound) {
return nil;
}
return self.messages[index];
}
- (NSInteger)openingMessageIndexInMessages {
NSString *introText = self.persona.introText ?: @"";
if (introText.length == 0 || self.messages.count == 0) {
return NSNotFound;
}
if (self.chatView.inverted) {
NSInteger lastIndex = self.messages.count - 1;
KBAiChatMessage *msg = self.messages[lastIndex];
return [self isOpeningMessage:msg] ? lastIndex : NSNotFound;
}
KBAiChatMessage *first = self.messages.firstObject;
return [self isOpeningMessage:first] ? 0 : NSNotFound;
2026-01-28 20:18:18 +08:00
}
2026-01-28 19:31:27 +08:00
#pragma mark -
///
- (void)handleChatSessionReset:(NSNotification *)notification {
NSNumber *companionIdObj = notification.userInfo[@"companionId"];
if (!companionIdObj) {
return;
}
NSInteger companionId = [companionIdObj integerValue];
//
if (self.persona && self.persona.personaId == companionId) {
NSLog(@"[KBPersonaChatCell] 收到聊天重置通知companionId=%ld, 清空聊天记录", (long)companionId);
//
self.messages = [NSMutableArray array];
self.hasLoadedData = NO;
self.currentPage = 1;
self.hasMoreHistory = YES;
//
[self.chatView clearMessages];
2026-01-28 20:18:18 +08:00
//
[self showOpeningMessage];
2026-01-28 19:31:27 +08:00
}
}
2026-01-26 20:36:51 +08:00
#pragma mark - 3
2026-01-26 16:53:41 +08:00
2026-01-26 20:36:51 +08:00
- (void)appendUserMessage:(NSString *)text {
if (text.length == 0) {
return;
2026-01-26 16:53:41 +08:00
}
2026-01-26 20:36:51 +08:00
if (!self.messages) {
self.messages = [NSMutableArray array];
}
2026-01-26 16:53:41 +08:00
2026-01-28 20:18:18 +08:00
[self ensureOpeningMessageAtTop];
2026-01-26 20:36:51 +08:00
KBAiChatMessage *message = [KBAiChatMessage userMessageWithText:text];
2026-01-31 22:40:50 +08:00
if (self.chatView.inverted) {
[self.messages insertObject:message atIndex:0];
} else {
[self.messages addObject:message];
}
2026-01-26 20:36:51 +08:00
[self.chatView addMessage:message autoScroll:YES];
2026-01-26 16:53:41 +08:00
}
2026-01-31 23:17:58 +08:00
- (void)appendUserMessage:(NSString *)text requestId:(NSString *)requestId {
[self appendUserMessage:text];
}
2026-01-29 14:42:49 +08:00
- (void)appendLoadingUserMessage {
if (!self.messages) {
self.messages = [NSMutableArray array];
}
[self ensureOpeningMessageAtTop];
KBAiChatMessage *message = [KBAiChatMessage loadingUserMessage];
2026-01-31 22:40:50 +08:00
if (self.chatView.inverted) {
[self.messages insertObject:message atIndex:0];
} else {
[self.messages addObject:message];
}
2026-01-29 14:42:49 +08:00
[self.chatView addMessage:message autoScroll:YES];
}
- (void)updateLastUserMessage:(NSString *)text {
[self.chatView updateLastUserMessage:text];
//
2026-01-31 22:40:50 +08:00
if (self.chatView.inverted) {
for (NSInteger i = 0; i < self.messages.count; i++) {
KBAiChatMessage *message = self.messages[i];
if (message.type == KBAiChatMessageTypeUser && message.isLoading) {
message.text = text;
message.isLoading = NO;
message.isComplete = YES;
break;
}
}
} else {
for (NSInteger i = self.messages.count - 1; i >= 0; i--) {
KBAiChatMessage *message = self.messages[i];
if (message.type == KBAiChatMessageTypeUser && message.isLoading) {
message.text = text;
message.isLoading = NO;
message.isComplete = YES;
break;
}
2026-01-29 14:42:49 +08:00
}
}
}
2026-01-29 13:44:52 +08:00
- (void)markLastUserMessageLoadingComplete {
2026-01-29 14:42:49 +08:00
[self.chatView markLastUserMessageLoadingComplete];
//
2026-01-31 22:40:50 +08:00
if (self.chatView.inverted) {
for (NSInteger i = 0; i < self.messages.count; i++) {
KBAiChatMessage *message = self.messages[i];
if (message.type == KBAiChatMessageTypeUser && message.isLoading) {
message.isLoading = NO;
break;
}
}
} else {
for (NSInteger i = self.messages.count - 1; i >= 0; i--) {
KBAiChatMessage *message = self.messages[i];
if (message.type == KBAiChatMessageTypeUser && message.isLoading) {
message.isLoading = NO;
break;
}
2026-01-29 14:42:49 +08:00
}
}
2026-01-29 13:44:52 +08:00
}
2026-01-26 20:36:51 +08:00
- (void)appendAssistantMessage:(NSString *)text
2026-01-28 20:18:18 +08:00
audioId:(NSString *)audioId {
2026-01-26 20:36:51 +08:00
if (text.length == 0) {
return;
}
if (!self.messages) {
self.messages = [NSMutableArray array];
}
2026-01-28 20:18:18 +08:00
[self ensureOpeningMessageAtTop];
2026-01-29 20:56:24 +08:00
// loading
[self removeLoadingAssistantMessage];
2026-01-26 20:36:51 +08:00
KBAiChatMessage *message = [KBAiChatMessage assistantMessageWithText:text
audioId:audioId];
message.needsTypewriterEffect = YES;
2026-01-31 22:40:50 +08:00
if (self.chatView.inverted) {
[self.messages insertObject:message atIndex:0];
} else {
[self.messages addObject:message];
}
2026-01-26 20:36:51 +08:00
[self.chatView addMessage:message autoScroll:YES];
2026-01-26 16:53:41 +08:00
}
2026-01-29 20:56:24 +08:00
/// loading AI
- (void)appendLoadingAssistantMessage {
if (!self.messages) {
self.messages = [NSMutableArray array];
}
[self ensureOpeningMessageAtTop];
KBAiChatMessage *message = [KBAiChatMessage loadingAssistantMessage];
2026-01-31 22:40:50 +08:00
if (self.chatView.inverted) {
[self.messages insertObject:message atIndex:0];
} else {
[self.messages addObject:message];
}
2026-01-29 20:56:24 +08:00
[self.chatView addMessage:message autoScroll:YES];
}
2026-01-31 23:17:58 +08:00
- (void)appendLoadingAssistantMessageWithRequestId:(NSString *)requestId {
if (requestId.length == 0) {
[self appendLoadingAssistantMessage];
return;
}
if (!self.pendingAssistantMessages) {
self.pendingAssistantMessages = [NSMutableDictionary dictionary];
}
if (!self.messages) {
self.messages = [NSMutableArray array];
}
[self ensureOpeningMessageAtTop];
KBAiChatMessage *message = [KBAiChatMessage loadingAssistantMessage];
self.pendingAssistantMessages[requestId] = message;
if (self.chatView.inverted) {
[self.messages insertObject:message atIndex:0];
} else {
[self.messages addObject:message];
}
[self.chatView addMessage:message autoScroll:YES];
}
2026-01-29 20:56:24 +08:00
/// loading AI
- (void)removeLoadingAssistantMessage {
//
2026-01-31 22:40:50 +08:00
if (self.chatView.inverted) {
for (NSInteger i = 0; i < self.messages.count; i++) {
KBAiChatMessage *message = self.messages[i];
if (message.type == KBAiChatMessageTypeAssistant && message.isLoading) {
[self.messages removeObjectAtIndex:i];
break;
}
}
} else {
for (NSInteger i = self.messages.count - 1; i >= 0; i--) {
KBAiChatMessage *message = self.messages[i];
if (message.type == KBAiChatMessageTypeAssistant && message.isLoading) {
[self.messages removeObjectAtIndex:i];
break;
}
2026-01-29 20:56:24 +08:00
}
}
// chatView
[self.chatView removeLoadingAssistantMessage];
}
2026-01-31 23:17:58 +08:00
- (void)removeLoadingAssistantMessageWithRequestId:(NSString *)requestId {
if (requestId.length == 0) {
[self removeLoadingAssistantMessage];
return;
}
KBAiChatMessage *target = self.pendingAssistantMessages[requestId];
if (!target) {
return;
}
[self.pendingAssistantMessages removeObjectForKey:requestId];
NSInteger idx = [self.messages indexOfObjectIdenticalTo:target];
if (idx != NSNotFound) {
[self.messages removeObjectAtIndex:idx];
[self.chatView reloadWithMessages:self.messages keepOffset:NO scrollToBottom:NO];
}
}
- (void)updateAssistantMessageWithRequestId:(NSString *)requestId
text:(NSString *)text
audioId:(nullable NSString *)audioId {
if (requestId.length == 0) {
[self appendAssistantMessage:text audioId:audioId];
return;
}
KBAiChatMessage *target = self.pendingAssistantMessages[requestId];
[self.pendingAssistantMessages removeObjectForKey:requestId];
if (!target) {
[self appendAssistantMessage:text audioId:audioId];
return;
}
target.isLoading = NO;
target.text = text ?: @"";
target.audioId = audioId;
target.needsTypewriterEffect = YES;
target.isComplete = NO;
[self.chatView reloadMessage:target];
}
2026-01-27 16:28:17 +08:00
- (void)updateChatViewBottomInset:(CGFloat)bottomInset {
[self.chatView updateContentBottomInset:bottomInset];
}
2026-01-26 20:36:51 +08:00
#pragma mark - KBChatTableViewDelegate
2026-01-26 18:17:02 +08:00
2026-01-26 20:36:51 +08:00
- (void)chatTableViewDidScroll:(KBChatTableView *)chatView
scrollView:(UIScrollView *)scrollView {
2026-01-26 18:17:02 +08:00
CGFloat offsetY = scrollView.contentOffset.y;
2026-01-31 22:40:50 +08:00
if (chatView.inverted) {
CGFloat contentHeight = scrollView.contentSize.height;
CGFloat scrollViewHeight = scrollView.bounds.size.height;
CGFloat maxOffsetY = contentHeight - scrollViewHeight + scrollView.contentInset.bottom;
if (maxOffsetY < 0) {
maxOffsetY = 0;
}
if (offsetY >= maxOffsetY - 50 && !self.isLoading && self.canTriggerLoadMore && self.hasMoreHistory) {
self.canTriggerLoadMore = NO;
[self loadMoreHistory];
} else if (offsetY < maxOffsetY - 100) {
self.canTriggerLoadMore = YES;
}
return;
}
2026-01-30 21:24:17 +08:00
if (offsetY <= 50 && !self.isLoading && self.canTriggerLoadMore && self.hasMoreHistory) {
self.canTriggerLoadMore = NO;
2026-01-26 18:17:02 +08:00
[self loadMoreHistory];
2026-01-30 21:24:17 +08:00
} else if (offsetY > -20) {
self.canTriggerLoadMore = YES;
2026-01-26 18:17:02 +08:00
}
}
2026-01-26 20:36:51 +08:00
- (void)chatTableViewDidTriggerLoadMore:(KBChatTableView *)chatView {
[self loadMoreHistory];
}
2026-01-26 16:53:41 +08:00
#pragma mark - Lazy Load
- (UIImageView *)backgroundImageView {
if (!_backgroundImageView) {
_backgroundImageView = [[UIImageView alloc] init];
_backgroundImageView.contentMode = UIViewContentModeScaleAspectFill;
_backgroundImageView.clipsToBounds = YES;
}
return _backgroundImageView;
}
- (UIImageView *)avatarImageView {
if (!_avatarImageView) {
_avatarImageView = [[UIImageView alloc] init];
_avatarImageView.contentMode = UIViewContentModeScaleAspectFill;
2026-01-27 17:49:45 +08:00
_avatarImageView.layer.cornerRadius = 27;
2026-01-26 16:53:41 +08:00
_avatarImageView.layer.borderWidth = 3;
_avatarImageView.layer.borderColor = [UIColor whiteColor].CGColor;
_avatarImageView.clipsToBounds = YES;
2026-01-29 16:03:21 +08:00
_avatarImageView.userInteractionEnabled = YES;
//
UITapGestureRecognizer *tap = [[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(avatarTapped)];
[_avatarImageView addGestureRecognizer:tap];
2026-01-26 16:53:41 +08:00
}
return _avatarImageView;
}
- (UILabel *)nameLabel {
if (!_nameLabel) {
_nameLabel = [[UILabel alloc] init];
2026-01-27 18:53:19 +08:00
_nameLabel.font = [UIFont boldSystemFontOfSize:12];
2026-01-26 16:53:41 +08:00
_nameLabel.textColor = [UIColor whiteColor];
_nameLabel.textAlignment = NSTextAlignmentCenter;
}
return _nameLabel;
}
- (UILabel *)openingLabel {
if (!_openingLabel) {
_openingLabel = [[UILabel alloc] init];
_openingLabel.font = [UIFont systemFontOfSize:14];
_openingLabel.textColor = [[UIColor whiteColor] colorWithAlphaComponent:0.9];
_openingLabel.textAlignment = NSTextAlignmentCenter;
_openingLabel.numberOfLines = 2;
}
return _openingLabel;
}
2026-01-26 20:36:51 +08:00
- (KBChatTableView *)chatView {
if (!_chatView) {
_chatView = [[KBChatTableView alloc] init];
_chatView.backgroundColor = [UIColor clearColor];
2026-01-31 22:40:50 +08:00
_chatView.inverted = YES;
2026-01-26 20:36:51 +08:00
_chatView.delegate = self;
2026-01-26 16:53:41 +08:00
}
2026-01-26 20:36:51 +08:00
return _chatView;
2026-01-26 16:53:41 +08:00
}
2026-01-27 18:53:19 +08:00
- (KBImagePositionButton *)commentButton {
if (!_commentButton) {
//
_commentButton = [[KBImagePositionButton alloc] initWithImagePosition:KBImagePositionTop spacing:4];
//
_commentButton.titleLabel.font = [UIFont systemFontOfSize:10];
//
[_commentButton setImage:[UIImage imageNamed:@"ai_comment_icon"] forState:UIControlStateNormal];
//
[_commentButton setTitle:@"0" forState:UIControlStateNormal];
[_commentButton setTitleColor:[[UIColor whiteColor] colorWithAlphaComponent:0.8] forState:UIControlStateNormal];
//
[_commentButton addTarget:self action:@selector(commentButtonTapped:) forControlEvents:UIControlEventTouchUpInside];
}
return _commentButton;
}
- (KBImagePositionButton *)likeButton {
if (!_likeButton) {
//
_likeButton = [[KBImagePositionButton alloc] initWithImagePosition:KBImagePositionTop spacing:4];
//
_likeButton.titleLabel.font = [UIFont systemFontOfSize:10];
//
[_likeButton setImage:[UIImage imageNamed:@"ai_live_icon"] forState:UIControlStateNormal];
[_likeButton setImage:[UIImage imageNamed:@"ai_livesel_icon"] forState:UIControlStateSelected];
//
[_likeButton setTitle:@"0" forState:UIControlStateNormal];
[_likeButton setTitleColor:[[UIColor whiteColor] colorWithAlphaComponent:0.8] forState:UIControlStateNormal];
//
[_likeButton addTarget:self action:@selector(likeButtonTapped:) forControlEvents:UIControlEventTouchUpInside];
}
return _likeButton;
}
#pragma mark - Button Actions
2026-01-29 16:03:21 +08:00
- (void)avatarTapped {
NSLog(@"[KBPersonaChatCell] 头像点击,跳转到人设详情页");
if (self.persona.personaId <= 0) {
NSLog(@"[KBPersonaChatCell] personaId 无效,取消跳转");
return;
}
AIPersonInfoVC *vc = [[AIPersonInfoVC alloc] init];
vc.companionId = self.persona.personaId;
[KB_CURRENT_NAV pushViewController:vc animated:YES];
}
2026-01-27 18:53:19 +08:00
- (void)commentButtonTapped:(KBImagePositionButton *)sender {
2026-01-27 21:32:52 +08:00
NSLog(@"[KBPersonaChatCell] 评论按钮点击");
2026-01-27 18:53:19 +08:00
2026-01-27 21:32:52 +08:00
//
[self showComment];
2026-01-27 18:53:19 +08:00
}
- (void)likeButtonTapped:(KBImagePositionButton *)sender {
2026-01-27 21:32:52 +08:00
NSLog(@"[KBPersonaChatCell] 喜欢按钮点击");
NSInteger personaId = self.persona.personaId;
2026-01-27 18:53:19 +08:00
2026-01-27 21:32:52 +08:00
//
sender.enabled = NO;
__weak typeof(self) weakSelf = self;
[self.aiVM likeCompanionWithCompanionId:personaId completion:^(KBCommentLikeResponse * _Nullable response, NSError * _Nullable error) {
__strong typeof(weakSelf) strongSelf = weakSelf;
if (!strongSelf) {
return;
}
dispatch_async(dispatch_get_main_queue(), ^{
//
sender.enabled = YES;
if (error) {
NSLog(@"[KBPersonaChatCell] 点赞失败:%@", error.localizedDescription);
// TODO:
return;
}
if (response && response.code == 0) {
//
NSInteger currentLikeCount = [strongSelf.persona.likeCount integerValue];
// response.data: true false
if (response.data) {
// 1
currentLikeCount += 1;
sender.selected = YES;
NSLog(@"[KBPersonaChatCell] 点赞成功,新喜欢数:%ld", (long)currentLikeCount);
} else {
// 10
currentLikeCount = MAX(0, currentLikeCount - 1);
sender.selected = NO;
NSLog(@"[KBPersonaChatCell] 取消点赞成功,新喜欢数:%ld", (long)currentLikeCount);
}
//
strongSelf.persona.likeCount = [NSString stringWithFormat:@"%ld", (long)currentLikeCount];
strongSelf.persona.liked = sender.selected;
//
[sender setTitle:strongSelf.persona.likeCount forState:UIControlStateNormal];
} else {
NSLog(@"[KBPersonaChatCell] 点赞失败:%@", response.message ?: @"未知错误");
// TODO:
}
});
}];
2026-01-27 18:53:19 +08:00
}
2026-01-27 21:32:52 +08:00
#pragma mark - Comment View
- (void)showComment {
//
if (self.popView) {
[self.popView dismiss];
}
2026-01-29 16:42:43 +08:00
CGFloat customViewHeight = KB_SCREEN_HEIGHT * 0.7;
2026-01-27 21:32:52 +08:00
KBAICommentView *customView = [[KBAICommentView alloc]
initWithFrame:CGRectMake(0, 0, KB_SCREEN_WIDTH, customViewHeight)];
2026-01-29 14:51:42 +08:00
NSString *commentCount = self.persona.commentCount;
NSInteger totalCommentCount = [commentCount integerValue];;
customView.totalCommentCount = totalCommentCount;
2026-01-27 21:32:52 +08:00
// ID
2026-01-28 12:04:31 +08:00
customView.companionId = self.persona.personaId;
//
[customView loadComments];
2026-01-27 21:32:52 +08:00
LSTPopView *popView = [LSTPopView initWithCustomView:customView
parentView:nil
popStyle:LSTPopStyleSmoothFromBottom
dismissStyle:LSTDismissStyleSmoothToBottom];
2026-01-28 12:04:31 +08:00
customView.popView = popView;
2026-01-28 13:43:36 +08:00
popView.bgColor = [UIColor clearColor];
2026-01-27 21:32:52 +08:00
self.popView = popView;
popView.priority = 1000;
popView.isAvoidKeyboard = NO;
popView.hemStyle = LSTHemStyleBottom;
popView.dragStyle = LSTDragStyleY_Positive;
popView.dragDistance = customViewHeight * 0.5;
popView.sweepStyle = LSTSweepStyleY_Positive;
popView.swipeVelocity = 1600;
popView.sweepDismissStyle = LSTSweepDismissStyleSmooth;
2026-01-29 16:42:43 +08:00
popView.bgClickBlock = ^{
[KB_CURRENT_NAV.view endEditing:true];
};
2026-01-27 21:32:52 +08:00
[popView pop];
}
- (AiVM *)aiVM{
if (!_aiVM) {
_aiVM = [[AiVM alloc] init];
}
return _aiVM;
}
2026-01-26 16:53:41 +08:00
@end