您的位置:首页 > 移动开发 > IOS开发

[置顶] iOS捕捉键盘移动(根据键盘移动view跟随上下)

2016-06-29 20:32 477 查看
在QQ上我们经常会看到进入聊天页面时,输入框会跟着键盘的弹起而上下移动,本demo就是仿照此功能而设计

代码如下:

#pragma mark- vc
- (void)viewDidLoad {
[super viewDidLoad];
self.title = @"与xxx聊天中";

//从IOS7开始,导航条和状态栏合为一体,而且呈半透明状,viewController.view的坐标0,0点在屏幕左上方,如果子视图位置在view的顶部,就会被导航条遮盖,所以,viewController会对自身的第一个子视图(scrollView)顶部添加64像素的contentInset,避免scrollView内容被查盖。
self.automaticallyAdjustsScrollViewInsets = NO;

_table.contentInset = UIEdgeInsetsMake(64, 0, 0, 0);
_table.delegate = self;
_table.dataSource = self;

//设置分割线的样式。
_table.separatorStyle = UITableViewCellSeparatorStyleNone;

//scrollView滑动时键盘消失;
//_table.keyboardDismissMode = UIScrollViewKeyboardDismissModeOnDrag;

[self loadSubView];

//监听键盘frame变化的通知,键盘位置发生变化时,让输入框位置也随之变化。
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(receiveKeyboardFrameChangeNotification:) name:UIKeyboardWillChangeFrameNotification object:nil];

_array = [[NSMutableArray alloc] initWithCapacity:0];

[_table registerNib:[UINib nibWithNibName:@"SelfCell" bundle:nil] forCellReuseIdentifier:@"self"];

[_table registerNib:[UINib nibWithNibName:@"OtherCell" bundle:nil] forCellReuseIdentifier:@"other"];
}

- (void)loadSubView{
_inputView = [[UIView alloc] initWithFrame:CGRectMake(0, 627, 375, 40)];
[self.view addSubview:_inputView];

UIImageView *imgView = [[UIImageView alloc] initWithFrame:_inputView.bounds];
imgView.image = [UIImage imageNamed:@"chatinputbg.png"];
[_inputView addSubview:imgView];
[imgView release];

_textField = [[UITextField alloc] initWithFrame:CGRectMake(3, 3, 369, 34)];
_textField.borderStyle = UITextBorderStyleRoundedRect;
//设置return键的样式
_textField.returnKeyType = UIReturnKeySend;
//自动可用return键。
_textField.enablesReturnKeyAutomatically = YES;
//英文首字母是否大写。
_textField.autocapitalizationType = UITextAutocapitalizationTypeNone;
//清除键的显示方式。
_textField.clearButtonMode = UITextFieldViewModeWhileEditing;
[_inputView addSubview:_textField];

_textField.delegate = self;
}

#pragma mark- notificationMethod
- (void)receiveKeyboardFrameChangeNotification:(NSNotification *)noti{
//NSLog(@"%@",noti.userInfo);

//NSValue值类,基本类型和结构体不能直接放入数组和字典中,如果需要把结构体放入数组或字典,那么必须把结构体转换为NSValue,然后把NSValue放入字典或数组。
NSValue *value = [noti.userInfo objectForKey:UIKeyboardFrameEndUserInfoKey];

//CGRect rect = [value CGRectValue];
CGRect rect;
[value getValue:&rect];

[UIView beginAnimations:nil context:nil];
[UIView setAnimationDuration:[[noti.userInfo objectForKey:UIKeyboardAnimationDurationUserInfoKey] floatValue]];
//7是键盘动画专用速率
[UIView setAnimationCurve:[[noti.userInfo objectForKey:UIKeyboardAnimationCurveUserInfoKey] integerValue]];
_inputView.frame = CGRectMake(0, rect.origin.y-40, 375, 40);
_table.frame = CGRectMake(0, 0, 375, _inputView.frame.origin.y);
if (_array.count>1) {
[_table scrollToRowAtIndexPath:[NSIndexPath indexPathForRow:_array.count-1 inSection:0] atScrollPosition:UITableViewScrollPositionNone animated:NO];
}
[UIView commitAnimations];
}

#pragma mark- scrollViewDelegate
//scrollView将要开始拖拽时的回调方法。
- (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView{
[self.view endEditing:YES];
}

#pragma mark- textField
//return键点击时调用。
- (BOOL)textFieldShouldReturn:(UITextField *)textField{

[_array addObject:textField.text];
textField.text = @"";
[_table reloadData];

//scrollToRowAtIndexPath让tableView滚动到某一行,atScrollPosition参数是让这一行滚动出来之后显示在tableView上的位置(如果有可能)
[_table scrollToRowAtIndexPath:[NSIndexPath indexPathForRow:_array.count-1 inSection:0] atScrollPosition:UITableViewScrollPositionNone animated:YES];
return YES;
}

#pragma mark- table
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section{
return _array.count;
}

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{

NSString *text = [_array objectAtIndex:indexPath.row];
//boundingRectWithSize计算出文本显示的边框大小。第一个参数是限定的宽度和高度。第二个参数是计算选项,第三个参数是计算的文本的具体属性。
NSMutableParagraphStyle *style = [[NSMutableParagraphStyle alloc] init];
style.lineBreakMode = NSLineBreakByCharWrapping;
CGRect rect = [text boundingRectWithSize:CGSizeMake(220, 0) options:NSStringDrawingUsesLineFragmentOrigin|NSStringDrawingUsesFontLeading attributes:@{NSFontAttributeName:[UIFont systemFontOfSize:17],NSParagraphStyleAttributeName:style} context:nil];
[style release];

if (indexPath.row%2 == 0) {
SelfCell *cell = [tableView dequeueReusableCellWithIdentifier:@"self"];
cell.messageLabel.frame = CGRectMake(30, 10, rect.size.width, rect.size.height);
cell.bubbleImageView.frame = CGRectMake(20, 5, rect.size.width+20, rect.size.height+10);
cell.messageLabel.text = text;
return cell;
}else{
OtherCell *cell = [tableView dequeueReusableCellWithIdentifier:@"other"];
cell.messageLabel.frame = CGRectMake(375-rect.size.width-30, 10, rect.size.width, rect.size.height);
cell.bubbleImageView.frame = CGRectMake(375-rect.size.width-40, 5, rect.size.width+20, rect.size.height+10);
cell.messageLabel.text = text;
return cell;
}
}

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath{
NSString *text = [_array objectAtIndex:indexPath.row];
CGRect rect = [text boundingRectWithSize:CGSizeMake(220, 0) options:NSStringDrawingUsesLineFragmentOrigin|NSStringDrawingUsesFontLeading attributes:@{NSFontAttributeName:[UIFont systemFontOfSize:17]} context:nil];
return rect.size.height+20;
}
cell中

- (void)awakeFromNib {
// Initialization code
UIImage *img = [UIImage imageNamed:@"bubble.png"];
//stretchableImageWithLeftCapWidth设置边帽,能够使图片在拉伸或压缩时只按照某个像素点的颜色进行填充。(保证图片拉伸时不变形)
self.bubbleImageView.image = [img stretchableImageWithLeftCapWidth:25 topCapHeight:20];

self.messageLabel.numberOfLines = 0;
//折行方式
self.messageLabel.lineBreakMode = NSLineBreakByCharWrapping;
}
效果如下:

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: