Files
keyboard/CustomKeyboard/View/KBFunctionView.m

1214 lines
44 KiB
Mathematica
Raw Normal View History

2025-10-28 14:30:03 +08:00
//
// KBFunctionView.m
// CustomKeyboard
//
// Created by Mac on 2025/10/28.
//
#import "KBFunctionView.h"
2026-01-09 19:35:36 +08:00
#import "KBAuthManager.h" //
#import "KBBackspaceLongPressHandler.h"
#import "KBBackspaceUndoManager.h"
#import "KBBizCode.h"
#import "KBFullAccessGuideView.h"
#import "KBFullAccessManager.h"
2025-10-28 14:30:03 +08:00
#import "KBFunctionBarView.h"
#import "KBFunctionPasteView.h"
#import "KBFunctionTagCell.h"
2026-01-09 19:35:36 +08:00
#import "KBFunctionTagListView.h"
#import "KBHostAppLauncher.h"
2026-01-09 19:35:36 +08:00
#import "KBInputBufferManager.h"
#import "KBResponderUtils.h" // UIInputViewController
#import "KBSkinManager.h"
2025-11-12 16:03:30 +08:00
#import "KBStreamOverlayView.h" //
2026-01-09 19:35:36 +08:00
#import "KBStreamTextView.h" //
2025-12-08 16:39:47 +08:00
#import "KBTagItemModel.h"
2026-01-09 19:35:36 +08:00
#import "KBULBridgeNotification.h" // Darwin UL
#import "Masonry.h"
#import "WJXEventSource.h"
#import <MBProgressHUD.h>
2025-12-08 16:39:47 +08:00
#import <MJExtension/MJExtension.h>
2025-10-28 14:30:03 +08:00
2026-01-09 19:35:36 +08:00
@interface KBFunctionView () <KBFunctionBarViewDelegate,
KBStreamOverlayViewDelegate,
KBFunctionTagListViewDelegate>
2025-10-28 14:30:03 +08:00
// UI
2026-01-09 19:35:36 +08:00
@property(nonatomic, strong) KBFunctionBarView *barViewInternal;
@property(nonatomic, strong) KBFunctionPasteView *pasteViewInternal;
@property(nonatomic, strong) KBFunctionTagListView *tagListView;
@property(nonatomic, strong) UIView *rightButtonContainer; //
@property(nonatomic, strong) UIButton *pasteButtonInternal;
@property(nonatomic, strong) UIButton *deleteButtonInternal;
@property(nonatomic, strong) UIButton *clearButtonInternal;
@property(nonatomic, strong) UIButton *sendButtonInternal;
2025-10-28 14:30:03 +08:00
2025-11-12 16:03:30 +08:00
// +
2026-01-09 19:35:36 +08:00
@property(nonatomic, strong, nullable) KBStreamOverlayView *streamOverlay;
2025-11-11 20:24:13 +08:00
2025-11-12 14:18:56 +08:00
//
2026-01-09 19:35:36 +08:00
@property(nonatomic, strong, nullable) WJXEventSource *eventSource;
@property(nonatomic, assign)
BOOL streamHasOutput; // \t
@property(nonatomic, strong, nullable)
NSNumber *loadingTagIndex; // loadingindex
@property(nonatomic, copy, nullable) NSString *loadingTagTitle;
@property(nonatomic, assign) BOOL eventSourceDidReceiveDone;
@property(nonatomic, copy, nullable) NSString *eventSourceSplitPrefix;
2025-11-12 13:43:48 +08:00
2025-10-28 14:30:03 +08:00
// Data
2025-12-08 16:39:47 +08:00
//@property (nonatomic, strong) NSArray<NSString *> *itemsInternal;
2026-01-09 19:35:36 +08:00
@property(nonatomic, strong) NSMutableArray<KBTagItemModel *> *modelArray;
2025-10-29 15:49:43 +08:00
//
2026-01-09 19:35:36 +08:00
@property(nonatomic, strong)
NSTimer *pasteboardTimer; // 线
@property(nonatomic, assign)
NSInteger lastHandledPBCount; // changeCount
2025-11-12 21:23:31 +08:00
// UL
2026-01-09 19:35:36 +08:00
@property(nonatomic, assign) NSUInteger kb_ulSeq; // UL
@property(nonatomic, assign) BOOL kb_ulHandledFlag; // App UL
@property(nonatomic, strong) KBBackspaceLongPressHandler *backspaceHandler;
2025-10-28 14:30:03 +08:00
@end
@implementation KBFunctionView
- (instancetype)initWithFrame:(CGRect)frame {
2026-01-09 19:35:36 +08:00
if (self = [super initWithFrame:frame]) {
// 使
[self kb_applyTheme];
self.backspaceHandler =
[[KBBackspaceLongPressHandler alloc] initWithContainerView:self];
[self setupUI];
// [self reloadDemoData];
[self kb_reloadTagsFromSharedDefaults];
//
_lastHandledPBCount = [UIPasteboard generalPasteboard].changeCount;
// 访访
// TCC/XPC
[[NSNotificationCenter defaultCenter]
addObserver:self
selector:@selector(kb_fullAccessChanged)
name:KBFullAccessChangedNotification
object:nil];
// App Darwin UL
CFNotificationCenterAddObserver(
CFNotificationCenterGetDarwinNotifyCenter(),
(__bridge const void *)(self), KBULDarwinCallback,
(__bridge CFStringRef)KBDarwinULHandled, NULL,
CFNotificationSuspensionBehaviorDeliverImmediately);
}
return self;
2025-10-28 14:30:03 +08:00
}
2025-12-08 16:39:47 +08:00
#pragma mark - Data
/// App Group NSUserDefaults JSON model +
- (void)kb_reloadTagsFromSharedDefaults {
2026-01-09 19:35:36 +08:00
NSUserDefaults *sharedDefaults =
[[NSUserDefaults alloc] initWithSuiteName:AppGroup];
NSDictionary *jsonDict = [sharedDefaults objectForKey:AppGroup_MyKbJson];
if (jsonDict != nil) {
id dataObj = jsonDict[@"data"];
NSArray<KBTagItemModel *> *modelList =
[KBTagItemModel mj_objectArrayWithKeyValuesArray:(NSArray *)dataObj];
if (modelList.count > 0) {
self.modelArray = [NSMutableArray array];
[self.modelArray addObjectsFromArray:modelList];
// [self.collectionView reloadData];
[self.tagListView setItems:self.modelArray];
2025-12-08 16:39:47 +08:00
}
2026-01-09 19:35:36 +08:00
} else {
NSLog(@"json❎");
}
2025-12-08 16:39:47 +08:00
}
2025-11-04 21:01:46 +08:00
#pragma mark - Theme
2026-01-09 19:35:36 +08:00
///
- (BOOL)kb_isDarkMode {
if (@available(iOS 13.0, *)) {
return self.traitCollection.userInterfaceStyle == UIUserInterfaceStyleDark;
}
return NO;
2025-11-04 21:01:46 +08:00
}
2026-01-09 19:35:36 +08:00
#pragma mark - Theme Colors
/// #323232 #D0D3DA
+ (UIColor *)kb_backgroundColor {
if (@available(iOS 13.0, *)) {
return [UIColor colorWithDynamicProvider:^UIColor *_Nonnull(
UITraitCollection *_Nonnull traitCollection) {
if (traitCollection.userInterfaceStyle == UIUserInterfaceStyleDark) {
return [UIColor colorWithHex:0x2B2B2B];
} else {
return [UIColor colorWithHex:0xD0D3DA];
}
}];
}
return [UIColor colorWithHex:0xD0D3DA];
2025-10-29 15:49:43 +08:00
}
2026-01-09 19:35:36 +08:00
/// Cell #707070 90%
+ (UIColor *)kb_cellBackgroundColor {
if (@available(iOS 13.0, *)) {
return [UIColor colorWithDynamicProvider:^UIColor *_Nonnull(
UITraitCollection *_Nonnull traitCollection) {
if (traitCollection.userInterfaceStyle == UIUserInterfaceStyleDark) {
return [UIColor colorWithHex:0x707070];
} else {
return [UIColor colorWithWhite:1 alpha:0.9];
}
2025-10-28 14:30:03 +08:00
}];
2026-01-09 19:35:36 +08:00
}
return [UIColor colorWithWhite:1 alpha:0.9];
}
2025-10-28 14:30:03 +08:00
2026-01-09 19:35:36 +08:00
/// Cell #FFFFFF #1B1F1A
+ (UIColor *)kb_cellTextColor {
if (@available(iOS 13.0, *)) {
return [UIColor colorWithDynamicProvider:^UIColor *_Nonnull(
UITraitCollection *_Nonnull traitCollection) {
if (traitCollection.userInterfaceStyle == UIUserInterfaceStyleDark) {
return [UIColor whiteColor];
} else {
return [UIColor colorWithHex:0x1B1F1A];
}
2025-10-28 14:30:03 +08:00
}];
2026-01-09 19:35:36 +08:00
}
return [UIColor colorWithHex:0x1B1F1A];
}
2025-10-28 14:30:03 +08:00
2026-01-09 19:35:36 +08:00
/// Clear
+ (UIColor *)kb_clearButtonTextColor {
if (@available(iOS 13.0, *)) {
return [UIColor colorWithDynamicProvider:^UIColor *_Nonnull(
UITraitCollection *_Nonnull traitCollection) {
if (traitCollection.userInterfaceStyle == UIUserInterfaceStyleDark) {
return [UIColor whiteColor];
} else {
return [UIColor blackColor];
}
2025-10-28 14:30:03 +08:00
}];
2026-01-09 19:35:36 +08:00
}
return [UIColor blackColor];
}
2025-10-28 14:30:03 +08:00
2026-01-09 19:35:36 +08:00
/// #707070 #B9BDC8
+ (UIColor *)kb_deleteButtonBackgroundColor {
if (@available(iOS 13.0, *)) {
return [UIColor colorWithDynamicProvider:^UIColor *_Nonnull(
UITraitCollection *_Nonnull traitCollection) {
if (traitCollection.userInterfaceStyle == UIUserInterfaceStyleDark) {
return [UIColor colorWithHex:0x707070];
} else {
return [UIColor colorWithHex:0xB9BDC8];
}
2025-10-28 14:30:03 +08:00
}];
2026-01-09 19:35:36 +08:00
}
return [UIColor colorWithHex:0xB9BDC8];
}
- (void)kb_applyTheme {
// 使
self.backgroundColor = [KBFunctionView kb_backgroundColor];
//
self.clearButtonInternal.backgroundColor =
[KBFunctionView kb_deleteButtonBackgroundColor];
[self.clearButtonInternal
setTitleColor:[KBFunctionView kb_clearButtonTextColor]
forState:UIControlStateNormal];
self.deleteButtonInternal.backgroundColor =
[KBFunctionView kb_deleteButtonBackgroundColor];
// TagListView cell
[self.tagListView.collectionView reloadData];
}
- (void)dealloc {
[self stopPasteboardMonitor];
[self kb_stopNetworkStreaming];
[[NSNotificationCenter defaultCenter] removeObserver:self];
CFNotificationCenterRemoveObserver(
CFNotificationCenterGetDarwinNotifyCenter(),
(__bridge const void *)(self), (__bridge CFStringRef)KBDarwinULHandled,
NULL);
}
#pragma mark - UI
- (void)setupUI {
// 1. Bar
[self addSubview:self.barViewInternal];
CGFloat barTopInset = KBFit(6.0f);
CGFloat barHeight = KBFit(52.0f);
[self.barViewInternal mas_makeConstraints:^(MASConstraintMaker *make) {
make.left.right.equalTo(self);
make.top.equalTo(self.mas_top).offset(barTopInset);
make.height.mas_equalTo(barHeight);
}];
//
[self addSubview:self.rightButtonContainer];
CGFloat rightInset = KBFit(4.0f);
CGFloat containerBottomInset = KBFit(10.0f);
CGFloat containerWidth = KBFit(60.0f);
[self.rightButtonContainer mas_makeConstraints:^(MASConstraintMaker *make) {
make.right.equalTo(self.mas_right).offset(-rightInset);
make.top.equalTo(self.barViewInternal.mas_bottom).offset(0);
make.bottom.equalTo(self.mas_bottom).offset(0);
make.width.mas_equalTo(containerWidth);
}];
//
[self.rightButtonContainer addSubview:self.pasteButtonInternal];
[self.rightButtonContainer addSubview:self.deleteButtonInternal];
[self.rightButtonContainer addSubview:self.clearButtonInternal];
[self.rightButtonContainer addSubview:self.sendButtonInternal];
// 8px稿
CGFloat smallH = KBFit(41.0f);
CGFloat vSpace = KBFit(8.0f);
[self.pasteButtonInternal mas_makeConstraints:^(MASConstraintMaker *make) {
make.top.equalTo(self.rightButtonContainer.mas_top);
make.left.right.equalTo(self.rightButtonContainer);
}];
[self.deleteButtonInternal mas_makeConstraints:^(MASConstraintMaker *make) {
make.top.equalTo(self.pasteButtonInternal.mas_bottom).offset(vSpace);
make.left.right.equalTo(self.rightButtonContainer);
make.height.equalTo(self.pasteButtonInternal);
}];
[self.clearButtonInternal mas_makeConstraints:^(MASConstraintMaker *make) {
make.top.equalTo(self.deleteButtonInternal.mas_bottom).offset(vSpace);
make.left.right.equalTo(self.rightButtonContainer);
make.height.equalTo(self.pasteButtonInternal);
}];
[self.sendButtonInternal mas_makeConstraints:^(MASConstraintMaker *make) {
make.top.equalTo(self.clearButtonInternal.mas_bottom).offset(vSpace);
make.left.right.equalTo(self.rightButtonContainer);
make.height.equalTo(self.pasteButtonInternal);
make.bottom.equalTo(self.rightButtonContainer.mas_bottom);
}];
// 2.
[self addSubview:self.pasteViewInternal];
[self.pasteViewInternal mas_makeConstraints:^(MASConstraintMaker *make) {
make.left.equalTo(self.mas_left).offset(vSpace);
make.right.equalTo(self.rightButtonContainer.mas_left).offset(-vSpace);
make.top.equalTo(self.barViewInternal.mas_bottom).offset(0);
make.height.mas_equalTo(smallH);
}];
// Paste
[self.pasteViewInternal.pasBtn addTarget:self
action:@selector(onTapPaste)
forControlEvents:UIControlEventTouchUpInside];
// 3. Tag List View
[self addSubview:self.tagListView];
[self.tagListView mas_makeConstraints:^(MASConstraintMaker *make) {
make.left.equalTo(self.pasteViewInternal);
make.right.equalTo(self.rightButtonContainer.mas_left).offset(-vSpace);
make.top.equalTo(self.pasteViewInternal.mas_bottom).offset(vSpace);
make.bottom.equalTo(self.mas_bottom).offset(0);
}];
2025-10-28 14:30:03 +08:00
}
#pragma mark - Data
2025-12-08 16:39:47 +08:00
//- (void)reloadDemoData {
// //
// self.itemsInternal = @[KBLocalized(@"Warm hearted man"),
// KBLocalized(@"Warm2 hearted man"),
// KBLocalized(@"Warm3 hearted man"),
// KBLocalized(@"撩女生啊u发顺丰大师傅"),
// KBLocalized(@"Warm = man"),
// KBLocalized(@"Warm hearted man"),
// KBLocalized(@"一枚暖男发放"),
// KBLocalized(@"聊天搭子"),
// KBLocalized(@"表达爱意"),
// KBLocalized(@"更多话术")];
// [self.tagListView setItems:self.itemsInternal];
//}
2025-10-28 14:30:03 +08:00
2025-11-12 19:46:07 +08:00
// UICollectionView KBFunctionTagListView
2025-10-28 14:30:03 +08:00
2025-11-11 20:24:13 +08:00
- (void)kb_showStreamTextViewIfNeededWithTitle:(NSString *)title {
2026-01-09 19:35:36 +08:00
//
if (self.streamOverlay.superview) {
return;
}
// 使
self.tagListView.hidden = YES;
KBStreamOverlayView *overlay = [[KBStreamOverlayView alloc] init];
overlay.delegate = (id)self;
[self addSubview:overlay];
[overlay mas_makeConstraints:^(MASConstraintMaker *make) {
//
CGFloat vSpace = KBFit(4.0f);
CGFloat overlayTopInset = KBFit(10.0f);
CGFloat overlayBottomInset = KBFit(10.0f);
make.left.equalTo(self.pasteViewInternal);
make.right.equalTo(self).offset(-vSpace);
make.top.equalTo(self.pasteViewInternal.mas_bottom).offset(overlayTopInset);
make.bottom.equalTo(self.mas_bottom).offset(-overlayBottomInset);
}];
// //Paste
self.pasteButtonInternal.hidden = NO;
self.deleteButtonInternal.hidden = YES;
self.clearButtonInternal.hidden = YES;
self.sendButtonInternal.hidden = YES;
//
overlay.textView.contentHorizontalPadding = 8.0;
self.streamOverlay = overlay;
// UI cell start
2025-11-11 20:24:13 +08:00
}
- (void)kb_onTapStreamDelete {
2026-01-09 19:35:36 +08:00
//
[self kb_stopNetworkStreaming];
[self.streamOverlay removeFromSuperview];
self.streamOverlay = nil;
self.tagListView.hidden = NO;
//
self.pasteButtonInternal.hidden = NO;
self.deleteButtonInternal.hidden = NO;
self.clearButtonInternal.hidden = NO;
self.sendButtonInternal.hidden = NO;
2025-11-12 16:03:30 +08:00
}
//
- (void)streamOverlayDidTapClose:(KBStreamOverlayView *)overlay {
2026-01-09 19:35:36 +08:00
[self kb_onTapStreamDelete];
2025-11-11 20:24:13 +08:00
}
2025-12-09 14:32:21 +08:00
#pragma mark - Network Streaming (WJXEventSource)
2025-11-12 13:43:48 +08:00
2025-11-12 15:40:30 +08:00
- (void)kb_startNetworkStreamingWithSeed:(NSString *)seedTitle {
2026-01-09 19:35:36 +08:00
[self kb_stopNetworkStreaming];
if (![[KBFullAccessManager shared] hasFullAccess]) {
return;
}
2025-12-08 19:48:13 +08:00
2026-01-09 19:35:36 +08:00
NSString *apiUrl =
[NSString stringWithFormat:@"%@%@", KB_BASE_URL, API_AI_TALK];
NSURL *url = [NSURL URLWithString:apiUrl];
if (!url) {
return;
}
NSInteger characterId = 0;
if (self.loadingTagIndex != nil) {
NSInteger idx = self.loadingTagIndex.integerValue;
if (idx >= 0 && idx < self.modelArray.count) {
KBTagItemModel *model = self.modelArray[idx];
characterId = model.characterId;
2025-12-08 19:48:13 +08:00
}
2026-01-09 19:35:36 +08:00
}
NSInteger resolvedCharacterId = (characterId > 0) ? characterId : 75;
NSString *message =
seedTitle.length > 0 ? seedTitle : @"aliqua non cupidatat";
// message = [NSString stringWithFormat:@"%@%d",message,arc4random() %
// 10000];
NSDictionary *payload =
@{@"characterId" : @(resolvedCharacterId), @"message" : message};
NSLog(@"[KBFunction] request payload: %@", payload);
NSError *bodyError = nil;
NSData *bodyData = [NSJSONSerialization dataWithJSONObject:payload
options:0
error:&bodyError];
if (bodyError || bodyData.length == 0) {
NSLog(@"[KBFunction] build body failed: %@", bodyError);
return;
}
NSMutableURLRequest *request =
[NSMutableURLRequest requestWithURL:url
cachePolicy:NSURLRequestReloadIgnoringCacheData
timeoutInterval:60];
request.HTTPMethod = @"POST";
[request setValue:@"application/json" forHTTPHeaderField:@"Content-Type"];
NSString *token = KBAuthManager.shared.current.accessToken ?: @"";
if (token.length > 0) {
[request setValue:token forHTTPHeaderField:@"auth-token"];
}
request.HTTPBody = bodyData;
self.streamHasOutput = NO;
self.eventSourceSplitPrefix = nil;
self.eventSourceDidReceiveDone = NO;
__weak typeof(self) weakSelf = self;
WJXEventSource *source = [[WJXEventSource alloc] initWithRquest:request];
source.ignoreRetryAction = YES;
[source
addListener:^(WJXEvent *_Nonnull event) {
__strong typeof(weakSelf) self = weakSelf;
if (!self)
return;
2025-12-09 14:32:21 +08:00
[self kb_handleEventSourceMessage:event];
2026-01-09 19:35:36 +08:00
}
forEvent:WJXEventNameMessage
queue:NSOperationQueue.mainQueue];
[source
addListener:^(WJXEvent *_Nonnull event) {
__strong typeof(weakSelf) self = weakSelf;
if (!self)
return;
2025-12-09 14:32:21 +08:00
[self kb_handleEventSourceError:event.error];
2026-01-09 19:35:36 +08:00
}
forEvent:WJXEventNameError
queue:NSOperationQueue.mainQueue];
self.eventSource = source;
[self.eventSource open];
2025-11-12 13:43:48 +08:00
}
2025-11-12 14:18:56 +08:00
- (void)kb_stopNetworkStreaming {
2026-01-09 19:35:36 +08:00
[self.eventSource close];
self.eventSource = nil;
self.eventSourceSplitPrefix = nil;
self.eventSourceDidReceiveDone = NO;
self.streamHasOutput = NO;
2025-12-09 14:32:21 +08:00
}
- (void)kb_handleEventSourceMessage:(WJXEvent *)event {
2026-01-09 19:35:36 +08:00
if (event.data.length == 0) {
return;
}
NSLog(@"[KBStream] SSE raw payload: %@", event.data);
NSData *jsonData = [event.data dataUsingEncoding:NSUTF8StringEncoding];
if (!jsonData) {
return;
}
NSError *error = nil;
NSDictionary *payload = [NSJSONSerialization JSONObjectWithData:jsonData
options:0
error:&error];
if (error || ![payload isKindOfClass:[NSDictionary class]]) {
return;
}
if ([self kb_handleBizErrorIfNeeded:payload]) {
return;
}
NSString *type = payload[@"type"];
if (![type isKindOfClass:[NSString class]]) {
return;
}
if ([type isEqualToString:@"llm_chunk"]) {
NSString *chunk = [self kb_normalizedLLMChunkString:payload[@"data"]];
if (chunk.length > 0) {
[self kb_appendChunkToStreamView:chunk];
2025-12-09 14:32:21 +08:00
}
2026-01-09 19:35:36 +08:00
return;
}
if ([type isEqualToString:@"search_result"]) {
NSString *text = [self kb_formattedSearchResultString:payload[@"data"]];
if (text.length > 0) {
[self kb_appendChunkToStreamView:text];
2025-12-09 14:32:21 +08:00
}
2026-01-09 19:35:36 +08:00
return;
}
if ([type isEqualToString:@"done"]) {
self.eventSourceDidReceiveDone = YES;
[self kb_finishEventSourceWithError:nil];
return;
}
2025-12-09 14:32:21 +08:00
}
2026-01-09 19:35:36 +08:00
- (void)kb_handleEventSourceError:(NSError *_Nullable)error {
if (self.eventSourceDidReceiveDone) {
return;
}
[self kb_finishEventSourceWithError:error];
2025-12-09 14:32:21 +08:00
}
2026-01-09 19:35:36 +08:00
- (void)kb_finishEventSourceWithError:(NSError *_Nullable)error {
[self.eventSource close];
self.eventSource = nil;
if (!self.streamHasOutput && self.loadingTagIndex) {
[self.tagListView setLoading:NO atIndex:self.loadingTagIndex.integerValue];
self.loadingTagIndex = nil;
self.loadingTagTitle = nil;
}
BOOL shouldShowError = (error != nil);
if (shouldShowError) {
[KBHUD showInfo:error.localizedDescription ?: KBLocalized(@"拉取失败")];
}
if (self.streamOverlay) {
[self.streamOverlay finish];
}
self.eventSourceSplitPrefix = nil;
self.eventSourceDidReceiveDone = NO;
2025-12-09 14:32:21 +08:00
}
2025-12-17 17:01:49 +08:00
- (BOOL)kb_handleBizErrorIfNeeded:(NSDictionary *)payload {
2026-01-09 19:35:36 +08:00
NSInteger code = KBBizCodeFromJSONObject(payload);
if (code == NSNotFound || code == KBBizCodeSuccess) {
return NO;
}
BOOL needSubscriptionGuide = (code == KBBizCodeQuotaExhausted);
NSString *msg = KBBizMessageFromJSONObject(payload);
if (msg.length == 0) {
msg = KBLocalized(@"拉取失败");
}
NSError *bizError =
[NSError errorWithDomain:@"KBStreamBizError"
code:code
userInfo:@{NSLocalizedDescriptionKey : msg}];
[self kb_finishEventSourceWithError:bizError];
if (needSubscriptionGuide) {
[self kb_requestSubscriptionGuide];
}
return YES;
2025-12-17 17:01:49 +08:00
}
- (void)kb_requestSubscriptionGuide {
2026-01-09 19:35:36 +08:00
if ([self.delegate
respondsToSelector:@selector(functionViewDidRequestSubscription:)]) {
[self.delegate functionViewDidRequestSubscription:self];
}
2025-12-17 17:01:49 +08:00
}
2025-12-09 14:32:21 +08:00
#pragma mark - Event Parsing
- (NSString *)kb_normalizedLLMChunkString:(id)dataValue {
2026-01-09 19:35:36 +08:00
if (![dataValue isKindOfClass:[NSString class]]) {
return @"";
}
NSString *text = (NSString *)dataValue;
2025-12-09 16:12:54 +08:00
2026-01-09 19:35:36 +08:00
// 1. <SPLIT> "<SP" + "LIT>"
if (self.eventSourceSplitPrefix.length > 0) {
text = [self.eventSourceSplitPrefix stringByAppendingString:text ?: @""];
self.eventSourceSplitPrefix = nil;
}
if (text.length == 0) {
return @"";
}
// 2.
while (text.length > 0) {
unichar c0 = [text characterAtIndex:0];
if (c0 == '\n' || c0 == '\r') {
text = [text substringFromIndex:1];
continue;
2025-12-09 14:32:21 +08:00
}
2026-01-09 19:35:36 +08:00
break;
}
if (text.length == 0) {
return @"";
}
// 3. "<SPLIT"
NSString *suffix = [self kb_pendingSplitSuffixForString:text];
if (suffix.length > 0) {
self.eventSourceSplitPrefix = suffix;
text = [text substringToIndex:text.length - suffix.length];
}
if (text.length == 0) {
return @"";
}
2025-12-09 16:12:54 +08:00
2026-01-09 19:35:36 +08:00
// 4. <SPLIT> \t
text = [text stringByReplacingOccurrencesOfString:@"<SPLIT>"
withString:@"\t"];
2025-12-09 16:12:54 +08:00
2026-01-09 19:35:36 +08:00
// /t UI
return text;
2025-12-09 14:32:21 +08:00
}
- (NSString *)kb_formattedSearchResultString:(id)dataValue {
2026-01-09 19:35:36 +08:00
// data
if (![dataValue isKindOfClass:[NSArray class]]) {
return @"";
}
NSArray *list = (NSArray *)dataValue;
NSMutableArray<NSString *> *segments = [NSMutableArray array];
[list enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx,
BOOL *_Nonnull stop) {
NSString *payload = nil;
if ([obj isKindOfClass:[NSDictionary class]]) {
id val = obj[@"payload"];
if ([val isKindOfClass:[NSString class]]) {
payload = (NSString *)val;
}
} else if ([obj isKindOfClass:[NSString class]]) {
//
payload = (NSString *)obj;
}
2025-12-09 15:19:10 +08:00
2026-01-09 19:35:36 +08:00
payload = [payload
stringByTrimmingCharactersInSet:[NSCharacterSet
whitespaceAndNewlineCharacterSet]];
if (payload.length > 0) {
// payload
[segments addObject:payload];
}
}];
2025-12-09 15:19:10 +08:00
2026-01-09 19:35:36 +08:00
if (segments.count == 0) {
return @"";
}
2025-12-09 15:19:10 +08:00
2026-01-09 19:35:36 +08:00
// \t KBStreamTextView \t label
NSMutableString *result = [NSMutableString string];
2025-12-09 15:19:10 +08:00
2026-01-09 19:35:36 +08:00
[segments enumerateObjectsUsingBlock:^(NSString *_Nonnull obj, NSUInteger idx,
BOOL *_Nonnull stop) {
// \t
[result appendFormat:@"\t%@", obj];
}];
2025-12-09 14:32:21 +08:00
2026-01-09 19:35:36 +08:00
return result;
}
2025-12-09 15:19:10 +08:00
2025-12-09 14:32:21 +08:00
- (NSString *)kb_pendingSplitSuffixForString:(NSString *)text {
2026-01-09 19:35:36 +08:00
static NSString *const token = @"<SPLIT>";
if (text.length == 0) {
2025-12-09 14:32:21 +08:00
return @"";
2026-01-09 19:35:36 +08:00
}
NSUInteger tokenLen = token.length;
if (tokenLen <= 1) {
return @"";
}
NSUInteger maxLen = MIN(tokenLen - 1, text.length);
for (NSUInteger len = maxLen; len > 0; len--) {
NSString *suffix = [text substringFromIndex:text.length - len];
NSString *prefix = [token substringToIndex:len];
if ([suffix isEqualToString:prefix]) {
return suffix;
}
}
return @"";
2025-11-12 13:43:48 +08:00
}
2025-11-12 14:18:56 +08:00
#pragma mark - Helpers
/// KBStreamTextView
2025-12-09 14:32:21 +08:00
/// - `<SPLIT>` `\t`
2025-11-12 15:31:22 +08:00
/// - UI
2025-11-12 14:18:56 +08:00
- (void)kb_appendChunkToStreamView:(NSString *)chunk {
2026-01-09 19:35:36 +08:00
if (chunk.length == 0)
return;
// overlay cell
if (!self.streamOverlay) {
[self kb_showStreamTextViewIfNeededWithTitle:self.loadingTagTitle ?: @""];
if (self.loadingTagIndex) {
[self.tagListView setLoading:NO
atIndex:self.loadingTagIndex.integerValue];
self.loadingTagIndex = nil;
self.loadingTagTitle = nil;
2025-11-12 16:49:19 +08:00
}
2026-01-09 19:35:36 +08:00
}
if (!self.streamOverlay)
return;
[self.streamOverlay appendChunk:chunk];
self.streamHasOutput = YES;
2025-11-12 13:43:48 +08:00
}
2025-11-28 16:55:26 +08:00
///
/// -
/// - +
2026-01-09 19:35:36 +08:00
- (void)kb_updatePasteButtonWithDisplayText:(NSString *_Nullable)text {
if (text.length > 0) {
NSString *displayText = text;
if (displayText.length > 30) {
displayText =
[[displayText substringToIndex:30] stringByAppendingString:@"…"];
2025-11-28 16:55:26 +08:00
}
2026-01-09 19:35:36 +08:00
[self.pasteView.pasBtn setImage:nil forState:UIControlStateNormal];
[self.pasteView.pasBtn setTitle:displayText forState:UIControlStateNormal];
} else {
UIImage *img = [UIImage imageNamed:@"kb_zt_icon"];
[self.pasteView.pasBtn setImage:img forState:UIControlStateNormal];
[self.pasteView.pasBtn setTitle:KBLocalized(@" Paste Ta's Words")
forState:UIControlStateNormal];
}
2025-11-28 16:55:26 +08:00
}
2025-11-12 16:03:30 +08:00
#pragma mark - KBFunctionTagListViewDelegate
2026-01-09 19:35:36 +08:00
- (void)tagListView:(KBFunctionTagListView *)view
didSelectIndex:(NSInteger)index
title:(NSString *)title {
// 1) 访
if (![[KBFullAccessManager shared] hasFullAccess]) {
// 访
[KBHUD showInfo:KBLocalized(@"处理中…")];
[[KBFullAccessManager shared] ensureFullAccessOrGuideInView:self];
return;
}
2025-11-12 21:23:31 +08:00
2026-01-09 19:35:36 +08:00
// 2) -> App App
//
if (!KBAuthManager.shared.isLoggedIn) {
UIInputViewController *ivc = KBFindInputViewController(self);
NSString *schemeStr =
[NSString stringWithFormat:@"%@://login?src=keyboard", KB_APP_SCHEME];
NSURL *scheme = [NSURL URLWithString:schemeStr];
// UIApplication App
BOOL ok = [KBHostAppLauncher openHostAppURL:scheme fromResponder:ivc.view];
2025-11-12 21:23:31 +08:00
return;
2026-01-09 19:35:36 +08:00
// if (!ivc) return;
// NSString *encodedTitle = [title
// stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet
// URLQueryAllowedCharacterSet]] ?: @""; NSURL *ul = [NSURL
// URLWithString:[NSString
// stringWithFormat:@"%@?src=functionView&index=%ld&title=%@",
// KB_UL_LOGIN, (long)index, encodedTitle]]; if (!ul) return;
// // UL ok
// dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.05 *
// NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
// [ivc.extensionContext openURL:ul completionHandler:^(__unused
// BOOL ok) {}];
// });
// // 500ms App 退
// Scheme宿 UIApplication self.kb_ulHandledFlag = NO;
// NSUInteger token = ++self.kb_ulSeq;
// dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.5 *
// NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
// if (token != self.kb_ulSeq) return; //
// if (self.kb_ulHandledFlag) return; // App
// NSURL *scheme = [NSURL URLWithString:[NSString
// stringWithFormat:@"%@://login?src=functionView&index=%ld&title=%@",
// KB_APP_SCHEME, (long)index, encodedTitle]]; if (!scheme)
// return; UIResponder *start = ivc.view ?: (UIResponder *)self;
// //
// [ivc dismissKeyboard];
// BOOL ok = [KBHostAppLauncher openHostAppURL:scheme
// fromResponder:start]; if (!ok) {
// [KBHUD showInfo:KBLocalized(@"请切换到主App完成登录")];
// }else{
//
// }
// });
// return;
}
BOOL hasPasteText = ![self.pasteView.pasBtn.currentTitle
isEqualToString:KBLocalized(@" Paste Ta's Words")];
// BOOL hasPasteText = (self.pasteView.pasBtn.imageView.image == nil);
if (!hasPasteText) {
[KBHUD showInfo:KBLocalized(@"Please copy the text first")];
return;
}
NSString *copyTitle = self.pasteView.pasBtn.currentTitle;
// 3)
[self.tagListView setLoading:YES atIndex:index];
self.loadingTagIndex = @(index);
self.loadingTagTitle = title ?: @"";
[self kb_startNetworkStreamingWithSeed:copyTitle];
return;
2025-11-12 21:23:31 +08:00
}
// Darwin App UL
2026-01-09 19:35:36 +08:00
static void KBULDarwinCallback(CFNotificationCenterRef center, void *observer,
CFStringRef name, const void *object,
CFDictionaryRef userInfo) {
KBFunctionView *self_ = (__bridge KBFunctionView *)observer;
if (!self_)
return;
dispatch_async(dispatch_get_main_queue(), ^{
self_.kb_ulHandledFlag = YES;
});
}
2025-11-11 20:24:13 +08:00
2026-01-09 19:35:36 +08:00
// UL App
// Scheme访 访
// KBStreamTextView
- (void)collectionView:(UICollectionView *)collectionView
didSelectItemAtIndexPath:(NSIndexPath *)indexPath {
// KBFunctionTagListView id/name
// + 访访
if ([[KBFullAccessManager shared] hasFullAccess]) {
KBTagItemModel *selModel = self.modelArray[indexPath.item];
[self kb_showStreamTextViewIfNeededWithTitle:selModel.characterName];
return;
}
2025-11-11 20:24:13 +08:00
2026-01-09 19:35:36 +08:00
[KBHUD showInfo:KBLocalized(@"处理中…")];
2025-10-30 20:23:34 +08:00
2026-01-09 19:35:36 +08:00
UIInputViewController *ivc = KBFindInputViewController(self);
if (!ivc)
return;
2025-10-30 20:46:54 +08:00
2026-01-09 19:35:36 +08:00
NSString *title = self.modelArray[indexPath.item].characterName;
NSString *encodedTitle =
[title stringByAddingPercentEncodingWithAllowedCharacters:
[NSCharacterSet URLQueryAllowedCharacterSet]]
?: @"";
NSURL *ul = [NSURL
URLWithString:
[NSString stringWithFormat:@"%@?src=functionView&index=%ld&title=%@",
KB_UL_LOGIN, (long)indexPath.item,
encodedTitle]];
if (!ul)
return;
2025-10-30 20:23:34 +08:00
2026-01-09 19:35:36 +08:00
dispatch_after(
dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.05 * NSEC_PER_SEC)),
dispatch_get_main_queue(), ^{
// extensionContext UL
2026-01-09 19:35:36 +08:00
[ivc.extensionContext
openURL:ul
completionHandler:^(BOOL ok) {
if (ok) {
return;
2026-01-09 19:35:36 +08:00
}
// UL 宿 UIApplication + Scheme
NSURL *scheme = [NSURL
URLWithString:
[NSString
stringWithFormat:
@"%@@//login?src=functionView&index=%ld&title=%@",
KB_APP_SCHEME, (long)indexPath.item,
encodedTitle]];
UIResponder *start = ivc.view ?: (UIResponder *)self;
BOOL ok2 = [KBHostAppLauncher openHostAppURL:scheme
fromResponder:start];
if (!ok2) {
// 访宿 Manager
//
2025-11-21 18:26:02 +08:00
dispatch_async(dispatch_get_main_queue(), ^{
2026-01-09 19:35:36 +08:00
[[KBFullAccessManager shared]
ensureFullAccessOrGuideInView:self];
2025-11-21 18:26:02 +08:00
});
2026-01-09 19:35:36 +08:00
}
}];
});
2025-10-30 20:23:34 +08:00
}
2025-10-28 14:30:03 +08:00
#pragma mark - Button Actions
2025-10-29 15:49:43 +08:00
- (void)onTapPaste {
2026-01-09 19:35:36 +08:00
[[KBMaiPointReporter sharedReporter]
reportClickWithEventName:@"click_keyboard_function_paste_btn"
pageId:@"keyboard_function_panel"
elementId:@"paste_btn"
extra:nil
completion:nil];
//
// - iOS16+ App
// - iOS15
//
// viewDidLoad +
// 访访
if (![[KBFullAccessManager shared] hasFullAccess]) {
// 访
[[KBFullAccessManager shared] ensureFullAccessOrGuideInView:self];
return;
}
UIPasteboard *pb = [UIPasteboard generalPasteboard];
NSString *text = pb.string; //
if (text.length <= 0) {
//
NSLog(@"粘贴板无可用文本或未授权粘贴");
[KBHUD showInfo:KBLocalized(@"Clipboard is empty")];
return;
}
2025-11-28 16:55:26 +08:00
2026-01-09 19:35:36 +08:00
// 1
// UIInputViewController *ivc = KBFindInputViewController(self);
// if (ivc) {
// id<UITextDocumentProxy> proxy = ivc.textDocumentProxy;
// [proxy insertText:text];
// }
2025-11-28 16:55:26 +08:00
2026-01-09 19:35:36 +08:00
// 2便便
[self kb_updatePasteButtonWithDisplayText:text];
2025-10-29 15:49:43 +08:00
}
#pragma mark -
//
// -
// - changeCount pasteboard.string
// * iOS16+
// * iOS15
// - / changeCount
- (void)startPasteboardMonitor {
2026-01-09 19:35:36 +08:00
//
return;
// 访宿/
if (![[KBFullAccessManager shared] hasFullAccess])
return;
2026-01-09 19:35:36 +08:00
if (self.pasteboardTimer)
return;
KBWeakSelf self.pasteboardTimer = [NSTimer
scheduledTimerWithTimeInterval:0.5
repeats:YES
block:^(NSTimer *_Nonnull timer) {
__strong typeof(weakSelf) self = weakSelf;
if (!self)
return;
UIPasteboard *pb =
[UIPasteboard generalPasteboard];
NSInteger cc = pb.changeCount;
if (cc <= self.lastHandledPBCount)
return; //
self.lastHandledPBCount =
cc; //
// iOS16+
NSString *text = pb.string;
// -> / ->
// +
[self
kb_updatePasteButtonWithDisplayText:text];
}];
2025-10-29 15:49:43 +08:00
}
- (void)stopPasteboardMonitor {
2026-01-09 19:35:36 +08:00
[self.pasteboardTimer invalidate];
self.pasteboardTimer = nil;
2025-10-29 15:49:43 +08:00
}
- (void)didMoveToWindow {
2026-01-09 19:35:36 +08:00
[super didMoveToWindow];
[self kb_refreshPasteboardMonitor];
2025-10-29 15:49:43 +08:00
}
- (void)setHidden:(BOOL)hidden {
2026-01-09 19:35:36 +08:00
BOOL wasHidden = self.isHidden;
[super setHidden:hidden];
if (wasHidden != hidden) {
[self kb_refreshPasteboardMonitor];
}
}
// 访
- (void)kb_refreshPasteboardMonitor {
2026-01-09 19:35:36 +08:00
BOOL visible = (self.window && !self.isHidden);
if (visible && [[KBFullAccessManager shared] hasFullAccess]) {
[self startPasteboardMonitor];
} else {
[self stopPasteboardMonitor];
}
2025-10-29 15:49:43 +08:00
}
- (void)kb_fullAccessChanged {
2026-01-09 19:35:36 +08:00
dispatch_async(dispatch_get_main_queue(), ^{
[self kb_refreshPasteboardMonitor];
});
}
- (void)onTapDelete {
2026-01-09 19:35:36 +08:00
NSLog(@"点击:删除");
[[KBMaiPointReporter sharedReporter]
reportClickWithEventName:@"click_keyboard_function_delete_btn"
pageId:@"keyboard_function_panel"
elementId:@"delete_btn"
extra:nil
completion:nil];
UIInputViewController *ivc = KBFindInputViewController(self);
id<UITextDocumentProxy> proxy = ivc.textDocumentProxy;
[[KBInputBufferManager shared] refreshFromProxyIfPossible:proxy];
[[KBInputBufferManager shared]
prepareSnapshotForDeleteWithContextBefore:proxy.documentContextBeforeInput
after:proxy
.documentContextAfterInput];
[[KBBackspaceUndoManager shared] captureAndDeleteBackwardFromProxy:proxy
count:1];
[[KBInputBufferManager shared] applyHoldDeleteCount:1];
2025-10-30 13:27:09 +08:00
}
2025-12-19 18:45:14 +08:00
- (void)onTapClear {
2026-01-09 19:35:36 +08:00
NSLog(@"点击:清空");
[[KBMaiPointReporter sharedReporter]
reportClickWithEventName:@"click_keyboard_function_clear_btn"
pageId:@"keyboard_function_panel"
elementId:@"clear_btn"
extra:nil
completion:nil];
[self.backspaceHandler performClearAction];
2025-10-30 13:27:09 +08:00
}
- (void)onTapSend {
2026-01-09 19:35:36 +08:00
NSLog(@"点击:发送");
[[KBMaiPointReporter sharedReporter]
reportClickWithEventName:@"click_keyboard_function_send_btn"
pageId:@"keyboard_function_panel"
elementId:@"send_btn"
extra:nil
completion:nil];
[[KBBackspaceUndoManager shared] registerNonClearAction];
// App
UIInputViewController *ivc = KBFindInputViewController(self);
id<UITextDocumentProxy> proxy = ivc.textDocumentProxy;
[proxy insertText:@"\n"];
[[KBInputBufferManager shared] appendText:@"\n"];
2025-10-30 13:27:09 +08:00
}
2025-10-28 14:30:03 +08:00
#pragma mark - Lazy
- (KBFunctionBarView *)barViewInternal {
2026-01-09 19:35:36 +08:00
if (!_barViewInternal) {
_barViewInternal = [[KBFunctionBarView alloc] init];
_barViewInternal.delegate = self; // BarView
}
return _barViewInternal;
2025-10-28 14:30:03 +08:00
}
2025-10-28 15:18:12 +08:00
#pragma mark - KBFunctionBarViewDelegate
2026-01-09 19:35:36 +08:00
- (void)functionBarView:(KBFunctionBarView *)bar
didTapLeftAtIndex:(NSInteger)index {
//
if ([self.delegate respondsToSelector:@selector(functionView:
didTapToolActionAtIndex:)]) {
[self.delegate functionView:self didTapToolActionAtIndex:index];
}
2025-10-28 15:18:12 +08:00
}
2026-01-09 19:35:36 +08:00
- (void)functionBarView:(KBFunctionBarView *)bar
didTapRightAtIndex:(NSInteger)index {
// /
if ([self.delegate respondsToSelector:@selector(functionView:
didRightTapToolActionAtIndex:)]) {
[self.delegate functionView:self didRightTapToolActionAtIndex:index];
}
2025-10-28 15:18:12 +08:00
}
2025-10-28 14:30:03 +08:00
- (KBFunctionPasteView *)pasteViewInternal {
2026-01-09 19:35:36 +08:00
if (!_pasteViewInternal) {
_pasteViewInternal = [[KBFunctionPasteView alloc] init];
}
return _pasteViewInternal;
2025-10-28 14:30:03 +08:00
}
2025-11-12 16:03:30 +08:00
- (KBFunctionTagListView *)tagListView {
2026-01-09 19:35:36 +08:00
if (!_tagListView) {
_tagListView = [[KBFunctionTagListView alloc] init];
_tagListView.delegate = (id)self;
}
return _tagListView;
2025-10-28 14:30:03 +08:00
}
- (UIView *)rightButtonContainer {
2026-01-09 19:35:36 +08:00
if (!_rightButtonContainer) {
_rightButtonContainer = [[UIView alloc] init];
_rightButtonContainer.backgroundColor = [UIColor clearColor];
}
return _rightButtonContainer;
2025-10-28 14:30:03 +08:00
}
2026-01-09 19:35:36 +08:00
- (UIButton *)buildRightButtonWithTitle:(NSString *)title
color:(UIColor *)color {
UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
btn.backgroundColor = color;
btn.layer.cornerRadius = 8.0;
btn.layer.masksToBounds = YES;
btn.titleLabel.font = [KBFont medium:13];
[btn setTitle:title forState:UIControlStateNormal];
[btn setTitleColor:[UIColor whiteColor] forState:UIControlStateNormal];
return btn;
2025-10-28 14:30:03 +08:00
}
- (UIButton *)pasteButtonInternal {
2026-01-09 19:35:36 +08:00
if (!_pasteButtonInternal) {
_pasteButtonInternal =
[self buildRightButtonWithTitle:KBLocalized(@"Paste")
color:[UIColor colorWithRed:0.13
green:0.73
blue:0.60
alpha:1.0]];
[_pasteButtonInternal addTarget:self
action:@selector(onTapPaste)
forControlEvents:UIControlEventTouchUpInside];
}
return _pasteButtonInternal;
2025-10-28 14:30:03 +08:00
}
- (UIButton *)deleteButtonInternal {
2026-01-09 19:35:36 +08:00
if (!_deleteButtonInternal) {
_deleteButtonInternal = [UIButton buttonWithType:UIButtonTypeCustom];
_deleteButtonInternal.backgroundColor = [UIColor colorWithHex:0xB9BDC8];
_deleteButtonInternal.layer.cornerRadius = 8.0;
_deleteButtonInternal.layer.masksToBounds = YES;
[_deleteButtonInternal setImage:[UIImage imageNamed:@"kb_del_icon"]
forState:UIControlStateNormal];
[_deleteButtonInternal addTarget:self
action:@selector(onTapDelete)
forControlEvents:UIControlEventTouchUpInside];
[self.backspaceHandler bindDeleteButton:_deleteButtonInternal
showClearLabel:NO];
}
return _deleteButtonInternal;
2025-10-28 14:30:03 +08:00
}
- (UIButton *)clearButtonInternal {
2026-01-09 19:35:36 +08:00
if (!_clearButtonInternal) {
_clearButtonInternal = [UIButton buttonWithType:UIButtonTypeCustom];
_clearButtonInternal.backgroundColor = [UIColor colorWithHex:0xB9BDC8];
_clearButtonInternal.layer.cornerRadius = 8.0;
_clearButtonInternal.layer.masksToBounds = YES;
_clearButtonInternal.titleLabel.font = [KBFont medium:13];
[_clearButtonInternal setTitle:KBLocalized(@"Clear")
forState:UIControlStateNormal];
[_clearButtonInternal setTitleColor:[UIColor blackColor]
forState:UIControlStateNormal];
[_clearButtonInternal addTarget:self
action:@selector(onTapClear)
forControlEvents:UIControlEventTouchUpInside];
}
return _clearButtonInternal;
2025-10-28 14:30:03 +08:00
}
2025-11-26 21:16:56 +08:00
- (UIButton *)sendButtonInternal {
2026-01-09 19:35:36 +08:00
if (!_sendButtonInternal) {
_sendButtonInternal =
[self buildRightButtonWithTitle:KBLocalized(@"Send")
color:[UIColor colorWithHex:0x02BEAC]];
[_sendButtonInternal addTarget:self
action:@selector(onTapSend)
forControlEvents:UIControlEventTouchUpInside];
}
return _sendButtonInternal;
2025-10-28 14:30:03 +08:00
}
#pragma mark - Expose
2026-01-09 19:35:36 +08:00
- (UICollectionView *)collectionView {
return self.tagListView.collectionView;
}
2025-12-08 16:39:47 +08:00
//- (NSArray<NSString *> *)items { return self.itemsInternal; }
2026-01-09 19:35:36 +08:00
- (KBFunctionBarView *)barView {
return self.barViewInternal;
}
- (KBFunctionPasteView *)pasteView {
return self.pasteViewInternal;
}
- (UIButton *)pasteButton {
return self.pasteButtonInternal;
}
- (UIButton *)deleteButton {
return self.deleteButtonInternal;
}
- (UIButton *)clearButton {
return self.clearButtonInternal;
}
- (UIButton *)sendButton {
return self.sendButtonInternal;
}
2025-10-28 14:30:03 +08:00
2025-10-30 13:27:09 +08:00
#pragma mark - Find Owner Controller
2025-11-04 16:37:24 +08:00
// KBResponderUtils.h
2025-10-30 13:27:09 +08:00
2025-10-28 14:30:03 +08:00
@end