iOS开发之自定义表情键盘(组件封装与自动布局)

简介:

  下面的东西是编写自定义的表情键盘,话不多说,开门见山吧!下面主要用到的知识有MVC, iOS开发中的自动布局,自定义组件的封装与使用,Block回调,CoreData的使用。有的小伙伴可能会问写一个自定义表情键盘肿么这么麻烦?下面将会介绍我们如何用上面提到的东西来定义我们的表情键盘。下面的内容会比较多,这篇博文还是比较有料的。

  还是那句话写技术博客是少不了代码的,下面会结合代码来回顾一下iOS的知识,本篇博文中用到的知识点在前面的博客中都能找到相应的内容,本篇算是一个小小的功能整合。先来张图看一下本app的目录结构。我是根据自己对MVC的理解来构建的目录结构,希望起到抛砖引玉的作用,有好的解决方案欢迎评论或者留言指出。Face文件中存放的时我们的表情图片,Model文件封装的是从sqlite中读取历史头像的组件,View文件中封装的时我们自定义的组件,也就是自定义键盘相关的视图,Controller负责将我们的各个组件组装到一起完成我们想要的功能。下面会一一介绍。

 

  上面是文件的组织结构,下面为了更为直观的了解我们想要的效果,下面先看几张截图,来直观的感受一下运行效果,上面是竖屏的显示效果,下面是横屏的显示效果。因为在封装自定义键盘中用到了自动布局所以横屏显示或者在更大的屏幕上显示是没问题的,常用表情是用户用过的表情,然后存在Sqlite中,显示时并按时间降序排列。more是用来扩展功能用的接口。话不多说,来的代码才是实在的。

 

  一.View(自定义视图)

    View文件夹下存放的时我们自定义的视图组件,因为是自定义的组件所以storyboard我们就用不了啦,所有的代码都必须手写,这样才能保证组件使用的灵活性和减少各个组件之间的耦合性,更利于团队之间的合作。在封装组件时要预留好外界可能使用到的接口,和返回该返回的数据。好啦,废话少说,来点干货吧!

    1、FaceView组件的封装:FaceView即负责显示一个个的头像。在使用该组件时要传入要显示的图片和图片对应的文字(如【哈哈】),当点击图片的时候,会通过block回调的形式把该图片的image以及图片文字返回到使用的组件中去,下面是关键代码:

      FaceView.h中的代码如下(下面代码是定义啦相应的Block类型和对外的接口):

#import <UIKit/UIKit.h>

//声明表情对应的block,用于把点击的表情的图片和图片信息传到上层视图
typedef void (^FaceBlock) (UIImage *image, NSString *imageText);

@interface FaceView : UIView

//图片对应的文字
@property (nonatomic, strong) NSString *imageText;
//表情图片
@property (nonatomic, strong) UIImage *headerImage;

//设置block回调
-(void)setFaceBlock:(FaceBlock)block;

//设置图片,文字
-(void)setImage:(UIImage *) image ImageText:(NSString *) text;

@end

      FaceView.m中的代码如下
//
//  FaceView.m
//  MyKeyBoard
//
//  Created by 青玉伏案 on 14-9-16.
//  Copyright (c) 2014年 Mrli. All rights reserved.
//

#import "FaceView.h"

@interface FaceView ()
@property(strong, nonatomic) FaceBlock block;
@property (strong, nonatomic) UIImageView *imageView;
@end

@implementation FaceView

//初始化图片
- (id)initWithFrame:(CGRect)frame
{
    //face的大小
    frame.size.height = 30;
    frame.size.width = 30;
    self = [super initWithFrame:frame];
    if (self) {
        self.imageView = [[UIImageView alloc] initWithFrame:CGRectMake(0, 0, 30, 30)];
        [self addSubview:self.imageView];
    }
    return self;
}

-(void) setFaceBlock:(FaceBlock)block
{
    self.block = block;
}


-(void) setImage:(UIImage *)image ImageText:(NSString *)text
{
    //显示图片
    [self.imageView setImage:image];
    
    //把图片存储起来
    self.headerImage = image;
    
    self.imageText = text;
}

//点击时回调
-(void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event
{
    UITouch *touch = [touches anyObject];
    CGPoint point = [touch locationInView:self];
    //判断触摸的结束点是否在图片中
    if (CGRectContainsPoint(self.bounds, point))
    {
        //回调,把该头像的信息传到相应的controller中
        self.block(self.headerImage, self.imageText);
    }
}

@end

    代码说明:

      主要就是block回调的使用,就是封装了一个自定义的button,具体内容请参考之前的博客“IOS开发之自定义Button(集成三种回调模式)

 

    2、FunctionView组件的封装,FunctionView就是使用FaceView组件和ScrollView组件把表情加载进来,在实例化FunctionView组件时,我们用到了自动布局来设置ScrollView和下面的Button

      FunctionView.h的代码如下,在.h中留有组件的接口和回调用的Block, plistFileName用于加载我们的资源文件时使用,至于如何使用plist文件,请参考之前的博客:IOS开发之显示微博表情

//
//  FunctionView.h
//  MyKeyBoard
//
//  Created by 青玉伏案 on 14-9-16.
//  Copyright (c) 2014年 Mrli. All rights reserved.
//

#import <UIKit/UIKit.h>

//定义对应的block类型,用于数据的交互
typedef void (^FunctionBlock) (UIImage *image, NSString *imageText);

@interface FunctionView : UIView
//资源文件名
@property (nonatomic, strong) NSString *plistFileName;
//接受block块
-(void)setFunctionBlock:(FunctionBlock) block;

@end

      FunctionView.m中的代码如下,常用表情是在sqlite中获取的,而全部表情是通过plist文件的信息在Face文件中加载的:
//
//  FunctionView.m
//  MyKeyBoard
//
//  Created by 青玉伏案 on 14-9-16.
//  Copyright (c) 2014年 Mrli. All rights reserved.
//

#import "FunctionView.h"
#import "FaceView.h"
#import "ImageModelClass.h"
#import "HistoryImage.h"

@interface FunctionView()

@property (strong, nonatomic) FunctionBlock block;
//暂存表情组件回调的表情和表情文字
@property (strong, nonatomic) UIImage *headerImage;
@property (strong, nonatomic) NSString *imageText;

//display我们的表情图片
@property (strong, nonatomic) UIScrollView *headerScrollView;

//定义数据模型用于获取历史表情
@property (strong, nonatomic) ImageModelClass *imageModel;

@end


@implementation FunctionView
- (id)initWithFrame:(CGRect)frame
{
    self = [super initWithFrame:frame];
    if (self) {
        
        //实例化数据模型
        self.imageModel =[[ImageModelClass alloc] init];
       
        //实例化下面的button
        UIButton *faceButton = [[UIButton alloc] initWithFrame:CGRectZero];
        faceButton.backgroundColor = [UIColor grayColor];
        
        [faceButton setTitle:@"全部表情" forState:UIControlStateNormal];
        [faceButton setShowsTouchWhenHighlighted:YES];
        [faceButton addTarget:self action:@selector(tapButton1:) forControlEvents:UIControlEventTouchUpInside];
        [self addSubview:faceButton];
        
        
        //实例化常用表情按钮
        UIButton *moreButton = [[UIButton alloc] initWithFrame:CGRectZero];
        moreButton.backgroundColor = [UIColor orangeColor];
        [moreButton setTitle:@"常用表情" forState:UIControlStateNormal];
        [moreButton setShowsTouchWhenHighlighted:YES];
        [moreButton addTarget:self action:@selector(tapButton2:) forControlEvents:UIControlEventTouchUpInside];
        [self addSubview:moreButton];
        
        //给按钮添加约束
        faceButton.translatesAutoresizingMaskIntoConstraints = NO;
        moreButton.translatesAutoresizingMaskIntoConstraints = NO;
        //水平约束
        NSArray *buttonH = [NSLayoutConstraint constraintsWithVisualFormat:@"H:|[faceButton][moreButton(==faceButton)]|" options:0 metrics:0 views:NSDictionaryOfVariableBindings(faceButton,moreButton)];
        [self addConstraints:buttonH];
        
        //垂直约束
        NSArray *button1V = [NSLayoutConstraint constraintsWithVisualFormat:@"V:[faceButton(44)]|" options:0 metrics:0 views:NSDictionaryOfVariableBindings(faceButton)];
        [self addConstraints:button1V];
        
        NSArray *button2V = [NSLayoutConstraint constraintsWithVisualFormat:@"V:[moreButton(44)]|" options:0 metrics:0 views:NSDictionaryOfVariableBindings(moreButton)];
        [self addConstraints:button2V];
        
        //默认显示表情图片
        [self tapButton1:nil];
        
    }
    return self;
}

//接受回调
-(void)setFunctionBlock:(FunctionBlock)block
{
    self.block = block;
}

//点击全部表情按钮回调方法
-(void)tapButton1: (id) sender
{
    // 从plist文件载入资源
    NSBundle *bundle = [NSBundle mainBundle];
    NSString *path = [bundle pathForResource:self.plistFileName ofType:@"plist"];
    NSArray *headers = [NSArray arrayWithContentsOfFile:path];
    
    if (headers.count == 0) {
        NSLog(@"访问的plist文件不存在");
    }
    else
    {
        //调用headers方法显示表情
        [self header:headers];
    }
}

//点击历史表情的回调方法
-(void) tapButton2: (id) sender
{
    //从数据库中查询所有的图片
    NSArray *imageData = [self.imageModel queryAll];
    //解析请求到的数据
    NSMutableArray *headers = [NSMutableArray arrayWithCapacity:imageData.count];
    
    //数据实体,相当于javaBean的东西
    HistoryImage *tempData;
    
    for (int i = 0; i < imageData.count; i ++) {
        tempData = imageData[i];
        
        //解析数据,转换成函数headers要用的数据格式
        NSMutableDictionary *dic = [NSMutableDictionary dictionaryWithCapacity:2];
        [dic setObject:tempData.imageText forKey:@"chs"];
        UIImage *image = [UIImage imageWithData:tempData.headerImage];
        [dic setObject:image forKey:@"png"];
        
        [headers addObject:dic];
    }
    
    [self header:headers];
    
}


//负责把查出来的图片显示
-(void) header:(NSArray *)headers
{
    [self.headerScrollView removeFromSuperview];
    self.headerScrollView = [[UIScrollView alloc] initWithFrame:CGRectZero];
    [self addSubview:self.headerScrollView];
    
    //给scrollView添加约束
    self.headerScrollView.translatesAutoresizingMaskIntoConstraints = NO;
    //水平约束
    NSArray *scrollH = [NSLayoutConstraint constraintsWithVisualFormat:@"H:|-10-[_headerScrollView]-10-|" options:0   metrics:0 views:NSDictionaryOfVariableBindings(_headerScrollView)];
    [self addConstraints:scrollH];
    
    //垂直约束
    NSArray *scrolV = [NSLayoutConstraint constraintsWithVisualFormat:@"V:|-10-[_headerScrollView]-50-|" options:0 metrics:0 views:NSDictionaryOfVariableBindings(_headerScrollView)];
    [self addConstraints:scrolV];
    
    
    
    CGFloat scrollHeight = (self.frame).size.height-60;
    
    //根据图片量来计算scrollView的Contain的宽度
    CGFloat width = (headers.count/(scrollHeight/30))*30;
    self.headerScrollView.contentSize = CGSizeMake(width, scrollHeight);
    self.headerScrollView.pagingEnabled = YES;
    
    
    //图片坐标
    CGFloat x = 0;
    CGFloat y = 0;
    
    //往scroll上贴图片
    for (int i = 0; i < headers.count; i ++) {
        //获取图片信息
        UIImage *image;
        if ([headers[i][@"png"] isKindOfClass:[NSString class]])
        {
             image = [UIImage imageNamed:headers[i][@"png"]];
        }
        else
        {
            image = headers[i][@"png"];
        }
        
        NSString *imageText = headers[i][@"chs"];
        
        //计算图片位置
        y = (i%(int)(scrollHeight/30)) * 30;
        x = (i/(int)(scrollHeight/30)) * 30;
        
        FaceView *face = [[FaceView alloc] initWithFrame:CGRectMake(x, y, 0, 0)];
        [face setImage:image ImageText:imageText];
        
        //face的回调,当face点击时获取face的图片
        __weak __block FunctionView *copy_self = self;
        [face setFaceBlock:^(UIImage *image, NSString *imageText)
         {
             copy_self.block(image, imageText);
         }];
        
        [self.headerScrollView addSubview:face];
    }
    
    [self.headerScrollView setNeedsDisplay];

}

@end

      代码说明:

        1、主要是通过对资源文件或者对从数据库中查询的资源进行遍历然后添加到ScrollView中

        2.为了适应不同的屏幕给相应的组件添加了约束

 

    3.ToolView组件的封装: ToolView就是在主屏幕上下面的类似于TabBar的东西,当键盘出来的时候,ToolView会运动到键盘上面的位置。为了使用不同的屏幕,也需要用自动布局来实现。

      ToolView.h的代码如下:预留组件接口和声明block类型

//
//  ToolView.h
//  MyKeyBoard
//
//  Created by 青玉伏案 on 14-9-16.
//  Copyright (c) 2014年 Mrli. All rights reserved.
//

/*****************
 封装下面的工具条组件
 *****************/
#import <UIKit/UIKit.h>

//定义block块变量类型,用于回调,把本View上的按钮的index传到Controller中
typedef void (^ToolIndex) (NSInteger index);

@interface ToolView : UIView

//块变量类型的setter方法
-(void)setToolIndex:(ToolIndex) toolBlock;

@end

      ToolView.m的代码实现:
//
//  ToolView.m
//  MyKeyBoard
//
//  Created by 青玉伏案 on 14-9-16.
//  Copyright (c) 2014年 Mrli. All rights reserved.
//

#import "ToolView.h"

@interface ToolView ()

//定义ToolIndex类型的block,用于接受外界传过来的block
@property (nonatomic, strong) ToolIndex myBlock;

@end


@implementation ToolView

- (id)initWithFrame:(CGRect)frame
{
    self = [super initWithFrame:frame];
    if (self) {
       
        //1初始化表情按钮
        UIButton *faceButton = [[UIButton alloc] initWithFrame:CGRectZero];
        faceButton.backgroundColor = [UIColor orangeColor];
        [faceButton setTitle:@"表情" forState:UIControlStateNormal];
        [faceButton setShowsTouchWhenHighlighted:YES];
        [faceButton addTarget:self action:@selector(tapFaceButton:) forControlEvents:UIControlEventTouchUpInside];
        [self addSubview:faceButton];
        
        
        //初始化更多按钮
        UIButton *moreButton = [[UIButton alloc] initWithFrame:CGRectZero];
        moreButton.backgroundColor = [UIColor grayColor];
        [moreButton setTitle:@"More" forState:UIControlStateNormal];
        [moreButton setShowsTouchWhenHighlighted:YES];
        [moreButton addTarget:self action:@selector(tapMoreButton:) forControlEvents:UIControlEventTouchUpInside];
        [self addSubview:moreButton];
        
        
        //给我们的按钮添加约束来让按钮来占满toolView;
        faceButton.translatesAutoresizingMaskIntoConstraints = NO;
        moreButton.translatesAutoresizingMaskIntoConstraints = NO;
        
        //添加水平约束
        NSArray *buttonH = [NSLayoutConstraint constraintsWithVisualFormat:@"H:|[faceButton][moreButton(==faceButton)]|" options:0 metrics:0 views:NSDictionaryOfVariableBindings(faceButton,moreButton)];
        [self addConstraints:buttonH];
        
        //添加垂直约束
        NSArray *button1V = [NSLayoutConstraint constraintsWithVisualFormat:@"V:|[faceButton]|" options:0 metrics:0 views:NSDictionaryOfVariableBindings(faceButton)];
        [self addConstraints:button1V];
        
        NSArray *button2V = [NSLayoutConstraint constraintsWithVisualFormat:@"V:|[moreButton]|" options:0 metrics:0 views:NSDictionaryOfVariableBindings(moreButton)];
        [self addConstraints:button2V];

    }
    return self;
}

//接受传入的回调
-(void) setToolIndex:(ToolIndex)toolBlock
{
    self.myBlock = toolBlock;
}

//点击表情按钮要回调的方法
-(void) tapFaceButton: (id) sender
{
    self.myBlock(1);
}

//点击more要回调的方法
-(void) tapMoreButton: (id) sender
{
    self.myBlock(2);
}

@end

      代码说明:

        主要是对block回调的应用和给相应的组件添加相应的约束

    4.MoreView组件的封装代码就不往上贴啦,和上面的类似,下面是调用MoreView组件的运行效果,有兴趣的读者请自行编写,以上就是视图部分的代码了

     

  二. Mode部分的内容:

    1.先定义我们要使用的数据模型,数据模型如下,time是使用表情的时间,用于排序。

    2.下面编写我们的ImageModelClass类,里面封装了我们操作数据要用的方法

      ImageModelClass.h的代码如下,主要是预留的对外的接口:

//
//  ImageModelClass.h
//  MyKeyBoard
//
//  Created by 青玉伏案 on 14-9-16.
//  Copyright (c) 2014年 Mrli. All rights reserved.
//

#import <Foundation/Foundation.h>
#import <CoreData/CoreData.h>
#import "HistoryImage.h"

@interface ImageModelClass : NSObject
//保存数据
-(void)save:(NSData *) image ImageText:(NSString *) imageText;
//查询所有的图片
-(NSArray *) queryAll;
@end

      ImageModelClass.m的代码如下,主要是用CoreData对sqlite的操作:
//
//  ImageModelClass.m
//  MyKeyBoard
//
//  Created by 青玉伏案 on 14-9-16.
//  Copyright (c) 2014年 Mrli. All rights reserved.
//

#import "ImageModelClass.h"

@interface ImageModelClass ()

@property (nonatomic, strong) NSManagedObjectContext *manager;

@end

@implementation ImageModelClass
- (instancetype)init
{
    self = [super init];
    if (self) {
        //通过上下文获取manager
        UIApplication *application = [UIApplication sharedApplication];
        id delegate = application.delegate;
        self.manager = [delegate managedObjectContext];
    }
    return self;
}

-(void)save:(NSData *)image ImageText:(NSString *)imageText
{
    if (image != nil) {
        NSArray *result = [self search:imageText];
        
        HistoryImage *myImage;
        
        if (result.count == 0)
        {
            myImage = [NSEntityDescription insertNewObjectForEntityForName:NSStringFromClass([HistoryImage class]) inManagedObjectContext:self.manager];
            myImage.imageText = imageText;
            myImage.headerImage = image;
            myImage.time = [NSDate date];
        }
        else
        {
            myImage = result[0];
            myImage.time = [NSDate date];
        }
        
        //存储实体
        NSError *error = nil;
        if (![self.manager save:&error]) {
            NSLog(@"保存出错%@", [error localizedDescription]);
        }

    }

}


//查找
-(NSArray *)search:(NSString *) image
{
    NSArray *result;
    
        //新建查询条件
        NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] initWithEntityName:NSStringFromClass([HistoryImage class])];
        
        //添加谓词
        NSPredicate *predicate = [NSPredicate predicateWithFormat:@"imageText=%@",image];
        
        //把谓词给request
        [fetchRequest setPredicate:predicate];
        
        //执行查询
        NSError *error = nil;
        result = [self.manager executeFetchRequest:fetchRequest error:&error];
        if (error) {
            NSLog(@"查询错误:%@", [error localizedDescription]);
        }
    return result;
}



//查询所有的
-(NSArray *) queryAll
{
    //新建查询条件
    NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] initWithEntityName:NSStringFromClass([HistoryImage class])];
    
    //添加排序规则
    //定义排序规则
    NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"time" ascending:NO];
    
    //添加排序规则
    [fetchRequest setSortDescriptors:@[sortDescriptor]];

    
    //执行查询
    NSError *error = nil;
    NSArray *result = [self.manager executeFetchRequest:fetchRequest error:&error];
    if (error) {
        NSLog(@"查询错误:%@", [error localizedDescription]);
    }
    
    return result;
}

@end

      代码说明:

        1.保存图片时先查找图片是否存在,如果存在则更新时间,如果不存在则插入数据(写到这感觉想在用Hibernate写东西)。

 

  三.Controller部分,把上面的组件进行组装

    1.MainViewController.m中的延展部分的代码如下:

@interface MainViewController ()

//自定义组件
@property (nonatomic, strong) ToolView *toolView;

@property (nonatomic, strong) FunctionView *functionView;

@property (nonatomic, strong) MoreView *moreView;

//系统组件
@property (strong, nonatomic) IBOutlet UITextView *myTextView;

@property (strong, nonatomic) NSDictionary *keyBoardDic;

@property (strong, nonatomic) IBOutlet UIImageView *imageView;

@property (strong, nonatomic) NSString *sendString;

//数据model
@property (strong, nonatomic) ImageModelClass  *imageMode;

@property (strong, nonatomic)HistoryImage *tempImage;

@end

    2.在viewDidLoad中进行组件的初始化和实现组件的Block回调,代码如下
- (void)viewDidLoad
{
    [super viewDidLoad];
    
    //从sqlite中读取数据
    self.imageMode = [[ImageModelClass alloc] init];

    
    //实例化FunctionView
    self.functionView = [[FunctionView alloc] initWithFrame:CGRectMake(0, 0, 320, 216)];
    self.functionView.backgroundColor = [UIColor blackColor];
    
    //设置资源加载的文件名
    self.functionView.plistFileName = @"emoticons";
    
    __weak __block MainViewController *copy_self = self;
    //获取图片并显示
    [self.functionView setFunctionBlock:^(UIImage *image, NSString *imageText)
     {
         NSString *str = [NSString stringWithFormat:@"%@%@",copy_self.myTextView.text, imageText];
         
        copy_self.myTextView.text = str;
        copy_self.imageView.image = image;
         
        //把使用过的图片存入sqlite
         NSData *imageData = UIImagePNGRepresentation(image);
         [copy_self.imageMode save:imageData ImageText:imageText];
    }];
    
    
    //实例化MoreView
    self.moreView = [[MoreView alloc] initWithFrame:CGRectMake(0, 0, 0, 0)];
    self.moreView.backgroundColor = [UIColor blackColor];
    [self.moreView setMoreBlock:^(NSInteger index) {
        NSLog(@"MoreIndex = %d",index);
    }];
    

    
    //进行ToolView的实例化
    self.toolView = [[ToolView alloc] initWithFrame:CGRectZero];
    self.toolView.backgroundColor = [UIColor blackColor];
    [self.view addSubview:self.toolView];
    
    //给ToolView添加约束
    //开启自动布局
    self.toolView.translatesAutoresizingMaskIntoConstraints = NO;
    
    //水平约束
    NSArray *toolHConstraint = [NSLayoutConstraint constraintsWithVisualFormat:@"H:|[_toolView]|" options:0 metrics:0 views:NSDictionaryOfVariableBindings(_toolView)];
    [self.view addConstraints:toolHConstraint];
    
    //垂直约束
    NSArray *toolVConstraint = [NSLayoutConstraint constraintsWithVisualFormat:@"V:[_toolView(44)]|" options:0 metrics:0 views:NSDictionaryOfVariableBindings(_toolView)];
    [self.view addConstraints:toolVConstraint];
    
    

    
    //回调toolView中的方法
    [self.toolView setToolIndex:^(NSInteger index)
    {
        NSLog(@"%d", index);
        
        switch (index) {
            case 1:
                [copy_self changeKeyboardToFunction];
                break;
                
            case 2:
                [copy_self changeKeyboardToMore];
                break;
                
            default:
                break;
        }
        
    }];
    
    
    
    //当键盘出来的时候通过通知来获取键盘的信息
    //注册为键盘的监听着
    NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
    [center addObserver:self selector:@selector(keyNotification:) name:UIKeyboardWillChangeFrameNotification object:nil];
    
    
    //给键盘添加dan
    //TextView的键盘定制回收按钮
    UIToolbar * toolBar = [[UIToolbar alloc]initWithFrame:CGRectMake(0, 0, 320, 30)];
    
    UIBarButtonItem * item1 = [[UIBarButtonItem alloc]initWithBarButtonSystemItem:UIBarButtonSystemItemDone target:self action:@selector(tapDone:)];
    UIBarButtonItem * item2 = [[UIBarButtonItem alloc]initWithBarButtonSystemItem:UIBarButtonSystemItemFlexibleSpace target:nil action:nil];
    UIBarButtonItem * item3 = [[UIBarButtonItem alloc]initWithBarButtonSystemItem:UIBarButtonSystemItemFlexibleSpace target:nil action:nil];
    toolBar.items = @[item2,item1,item3];
    
    self.myTextView.inputAccessoryView =toolBar;
    
}

    3.当横竖屏幕切换时设置自定义键盘的高度
-(void)willAnimateRotationToInterfaceOrientation:(UIInterfaceOrientation)toInterfaceOrientation duration:(NSTimeInterval)duration
{
    //纵屏
    if (UIInterfaceOrientationIsPortrait(toInterfaceOrientation)) {
        CGRect frame = self.functionView.frame;
        frame.size.height = 216;
        self.functionView.frame = frame;
        self.moreView.frame = frame;

    }
    //横屏
    if (UIInterfaceOrientationIsLandscape(toInterfaceOrientation)) {
        CGRect frame = self.functionView.frame;
        frame.size.height = 150;
        self.functionView.frame = frame;
        self.moreView.frame = frame;
    }
}

    4.当键盘出来的时候,改变toolView的位置,通过键盘的通知来实现。当横屏的时候键盘的坐标系和我们当前的Frame的坐标系不一样所以当横屏时得做一坐标系的转换,代码如下;
//当键盘出来的时候改变toolView的位置(接到键盘出来的通知要做的方法)
-(void) keyNotification : (NSNotification *) notification
{
    NSLog(@"%@", notification.userInfo);
    
    self.keyBoardDic = notification.userInfo;
    //获取键盘移动后的坐标点的坐标点
    CGRect rect = [self.keyBoardDic[@"UIKeyboardFrameEndUserInfoKey"] CGRectValue];
    
    //把键盘的坐标系改成当前我们window的坐标系
    CGRect r1 = [self.view convertRect:rect fromView:self.view.window];
    
    [UIView animateWithDuration:[self.keyBoardDic[UIKeyboardAnimationDurationUserInfoKey] floatValue] animations:^{
        
        //动画曲线
        [UIView setAnimationCurve:[self.keyBoardDic[UIKeyboardAnimationCurveUserInfoKey] doubleValue]];
        
        CGRect frame = self.toolView.frame;
        
        frame.origin.y = r1.origin.y - frame.size.height;

        //根据键盘的高度来改变toolView的高度
        self.toolView.frame = frame;
    }];
}

    5.系统键盘和自定义键盘切换的代码如下:
//切换键盘的方法
-(void) changeKeyboardToFunction
{
    if ([self.myTextView.inputView isEqual:self.functionView])
    {
        self.myTextView.inputView = nil;
        [self.myTextView reloadInputViews];
    }
    else
    {
        self.myTextView.inputView = self.functionView;
        [self.myTextView reloadInputViews];
    }
    
    if (![self.myTextView isFirstResponder])
    {
        [self.myTextView becomeFirstResponder];
    }
}

   

  以上就是上面展示效果的核心代码了,在做的时候感觉难点在于如何进行屏幕适配,尤其是当屏幕横过来的时候键盘的坐标系和我们frame的坐标系不同,得做一个转换。发表博客的目的是想起到抛砖引玉的作用,有好的东西希望大家相互交流一下。

相关文章
|
14天前
|
iOS开发 开发者
uniapp开发ios打包Error code = -5000 Error message: Error: certificate file(p12) import failed!报错问题如何解决
uniapp开发ios打包Error code = -5000 Error message: Error: certificate file(p12) import failed!报错问题如何解决
109 67
uniapp开发ios打包Error code = -5000 Error message: Error: certificate file(p12) import failed!报错问题如何解决
|
2月前
|
开发框架 前端开发 Android开发
安卓与iOS开发中的跨平台策略
在移动应用开发的战场上,安卓和iOS两大阵营各据一方。随着技术的演进,跨平台开发框架成为开发者的新宠,旨在实现一次编码、多平台部署的梦想。本文将探讨跨平台开发的优势与挑战,并分享实用的开发技巧,帮助开发者在安卓和iOS的世界中游刃有余。
|
1月前
|
iOS开发 开发者 MacOS
深入探索iOS开发中的SwiftUI框架
【10月更文挑战第21天】 本文将带领读者深入了解Apple最新推出的SwiftUI框架,这一革命性的用户界面构建工具为iOS开发者提供了一种声明式、高效且直观的方式来创建复杂的用户界面。通过分析SwiftUI的核心概念、主要特性以及在实际项目中的应用示例,我们将展示如何利用SwiftUI简化UI代码,提高开发效率,并保持应用程序的高性能和响应性。无论你是iOS开发的新手还是有经验的开发者,本文都将为你提供宝贵的见解和实用的指导。
127 66
|
24天前
|
存储 监控 API
app开发之安卓Android+苹果ios打包所有权限对应解释列表【长期更新】-以及默认打包自动添加权限列表和简化后的基本打包权限列表以uniapp为例-优雅草央千澈
app开发之安卓Android+苹果ios打包所有权限对应解释列表【长期更新】-以及默认打包自动添加权限列表和简化后的基本打包权限列表以uniapp为例-优雅草央千澈
|
1月前
|
开发框架 Android开发 iOS开发
安卓与iOS开发中的跨平台策略:一次编码,多平台部署
在移动应用开发的广阔天地中,安卓和iOS两大阵营各占一方。随着技术的发展,跨平台开发框架应运而生,它们承诺着“一次编码,到处运行”的便捷。本文将深入探讨跨平台开发的现状、挑战以及未来趋势,同时通过代码示例揭示跨平台工具的实际运用。
152 3
|
1月前
|
Java 调度 Android开发
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。
|
2月前
|
存储 前端开发 Swift
探索iOS开发:从新手到专家的旅程
本文将带您领略iOS开发的奇妙之旅,从基础概念的理解到高级技巧的掌握,逐步深入iOS的世界。文章不仅分享技术知识,还鼓励读者在编程之路上保持好奇心和创新精神,实现个人成长与技术突破。
|
2月前
|
安全 IDE Swift
探索iOS开发之旅:从初学者到专家
在这篇文章中,我们将一起踏上iOS开发的旅程,从基础概念的理解到深入掌握核心技术。无论你是编程新手还是希望提升技能的开发者,这里都有你需要的指南和启示。我们将通过实际案例和代码示例,展示如何构建一个功能齐全的iOS应用。准备好了吗?让我们一起开始吧!
|
2月前
|
安全 Swift iOS开发
Swift 与 UIKit 在 iOS 应用界面开发中的关键技术和实践方法
本文深入探讨了 Swift 与 UIKit 在 iOS 应用界面开发中的关键技术和实践方法。Swift 以其简洁、高效和类型安全的特点,结合 UIKit 丰富的组件和功能,为开发者提供了强大的工具。文章从 Swift 的语法优势、类型安全、编程模型以及与 UIKit 的集成,到 UIKit 的主要组件和功能,再到构建界面的实践技巧和实际案例分析,全面介绍了如何利用这些技术创建高质量的用户界面。
52 2
|
2月前
|
vr&ar Android开发 iOS开发
安卓与iOS开发中的用户界面设计原则
【10月更文挑战第41天】探索移动应用开发的精髓,本文将深入分析安卓和iOS平台上用户界面设计的核心原则。通过比较两大操作系统的设计哲学,我们将揭示如何打造直观、易用且美观的应用程序界面。无论你是初学者还是资深开发者,这篇文章都将为你提供宝贵的见解和实用的技巧,帮助你在竞争激烈的应用市场中脱颖而出。

热门文章

最新文章