UITextView:定制选中文字的菜单和用代码选择文字
2016-03-15 17:01
288 查看
UITextView:定制选中文字的菜单和用代码选择文字
前言
先看一篇文章中的文字( 原文出处: iOS 7系列译文:认识 TextKit ):iOS 4:iOS 3.2 发布仅仅几个月后就发布了,文本方面没有一丁点新功能。(个人经历:在 WWDC,我走近工程师们,告诉他们我想要一个完善的 iOS 文本布局系统。回答是:“哦…提交个请求。”不出所料…)
iOS 5:文本方面没啥变化。(个人经历:在 WWDC,我和工程师们谈及 iOS 上文本系统。回答是:“我们没有看到太多的请求…” 靠!)
iOS 6:有些动作了:属性文本编辑被加入了UITextView。很不幸的是,它很难定制。默认的UI有粗体、斜体和下划线。用户可以设置字体大小和颜色。粗看起来相当不错,但还是没法控制布局或者提供一个便利的途径来定制文本属性。然而对于(文本编辑)开发者, 有一个大的新功能:可以继承 UITextView 了 ,这样的话,除了以前版本提供的键盘输入外,开发者 可以“免费”获得文本选择功能 。必须实现一个完全自定义的文本选择功能,可能是很多对非纯文本工具开发的尝试半途而废的原因。(个人经历:我,WWDC,工程师们。我想要一个
iOS 的文本系统。回答:“嗯。吖。是的。也许?看,它只是不执行…” 所以毕竟还是有希望,对吧?)
iOS 7:终于来了,TextKit。
很久之前就写了关于本文所说的两个功能的Demo,现在才有空总结下,清下草稿箱里的笔记。
看了以上文字才知道原来定制选中文字菜单和选择文字是iOS 6才能获取的特性,以上文字出处的作者绝对是个元老级iOS开发者了,从iOS 2玩到了iOS 7。
Anyway,定制UITextView绝对是件好玩的事。
继承UITextView
1.定制选中文字的菜单
首先新建一个类,继承自UITextView,假设类名为MyTextView,关键代码如下:/* 选中文字后是否能够呼出菜单 */ - (BOOL)canBecameFirstResponder { return YES; } /* 选中文字后的菜单响应的选项 */ - (BOOL)canPerformAction:(SEL)action withSender:(id)sender { if (action == @selector(copy:)) { // 菜单不能响应copy项 return NO; } else if (action == @selector(selectAll:)) { // 菜单不能响应select all项 return NO; } // 事实上一个return NO就可以将系统的所有菜单项全部关闭了 return NO; }
以上第一个方法用来确保我们选中文字后的菜单可以弹出,第二个方法用来关闭菜单中所有系统的菜单项,如copy, select, select all等。
然后使用UIMenuController定制菜单:
// 自定义text view选中文字后的菜单 UIMenuItem *selectItem = [[UIMenuItem alloc] initWithTitle:@"选择文字" action:@selector(callSelectText:)]; UIMenuItem *cancelItem = [[UIMenuItem alloc] initWithTitle:@"取消选中" action:@selector(cancelSelection:)]; [UIMenuController sharedMenuController].menuItems = @[selectItem, cancelItem];
注意必须实现两个MenuItem的响应方法才能显示出菜单:
#pragma mark - Menu Item Actions - (void)callSelectText:(id)sender { self.currentSelection_ = self.myTextView.selectedRange; self.selectOptionView.hidden = NO; [self.location_inputTextField becomeFirstResponder]; } - (void)cancelSelection:(id)sender { self.myTextView.selectedRange = NSRangeZero; }
最终效果如下:
之前的项目没有要求定制菜单项的图像,直接看SDK的内容的话貌似也没有Image之类的属性或方法,所以深层次定制菜单项的内容不得而知了。
2.通过代码选中一段文字
这个很简单,直接改变UITextView的selectedRange属性的值就可以了:@property(nonatomic) NSRange selectedRange;
例如我们点击选择文字后弹出一个文字选择的输入视图,这个我用一个XIB文件定制:
小心了,将xib中的UI组件和View Controller中的Outlet连接时,在代码中要先从xib文件中加载视图,才能使用其中的UI组件,例如:
NSArray *nibViews = [[NSBundle mainBundle] loadNibNamed:@"SelectOptionView" owner:self options:nil]; self.selectOptionView = nibViews[0]; self.selectOptionView.center = CGPointMake(self.view.center.x, self.view.bounds.size.height / 3); self.selectOptionView.hidden = YES; [self.view addSubview:self.selectOptionView]; // 要先加载了nib,IBOutlet才有意义,然后再设置其属性 self.location_inputTextField.delegate = self; self.length_inputTextField.delegate = self;
如果将
self . location_inputTextField . delegate = self ;
self . length_inputTextField . delegate = self ;
这两行代码置于loadNibNamed方法之前,那么两个文本输入框的delegate将为空(因为他们本身都是空,还没有加载)。
选择文字的Action代码为:
#pragma mark - Select View Actions - (IBAction)selectText:(id)sender { NSInteger loc = self.location_inputTextField.text.integerValue; NSInteger len = self.length_inputTextField.text.integerValue; NSUInteger textLength = self.myTextView.text.length; if (loc < 0 || len < 0 || loc > textLength || len > textLength) { UIAlertView *alerView = [[UIAlertView alloc] initWithTitle:@"错误" message:@"输入出错,输入的数不能小于0和大于文本长度" delegate:nil cancelButtonTitle:@"确定" otherButtonTitles:nil, nil]; [alerView show]; return; } self.currentSelection_ = NSMakeRange(loc, len); [self finishSelectingText]; } - (IBAction)cancelSelectText:(id)sender { [self finishSelectingText]; } - (void)finishSelectingText { [self.location_inputTextField resignFirstResponder]; [self.length_inputTextField resignFirstResponder]; self.selectOptionView.hidden = YES; [self.myTextView becomeFirstResponder]; self.myTextView.selectedRange = self.currentSelection_; }
没错,只要一句 self . myTextView . selectedRange = self . currentSelection_ ;就可以了。
另外,我们可以在UITextView的以下方法中监听到某段文字被选中:
#pragma mark - UITextView Delegate - (void)textViewDidChangeSelection:(UITextView *)textView { NSLog(@"Selection changed"); NSLog(@"loc = %d", self.myTextView.selectedRange.location); NSLog(@"len = %d", self.myTextView.selectedRange.length); }
运行结果:
控制台输出如下:
2014-02-16 23:33:56.197 MyTextView[4890:70b] Selection changed 2014-02-16 23:33:56.198 MyTextView[4890:70b] loc = 507 2014-02-16 23:33:56.198 MyTextView[4890:70b] len = 0 2014-02-16 23:33:56.334 MyTextView[4890:70b] Selection changed 2014-02-16 23:33:56.335 MyTextView[4890:70b] loc = 507 2014-02-16 23:33:56.335 MyTextView[4890:70b] len = 5 2014-02-16 23:34:05.291 MyTextView[4890:70b] Selection changed 2014-02-16 23:34:05.292 MyTextView[4890:70b] loc = 10 2014-02-16 23:34:05.292 MyTextView[4890:70b] len = 100
3.让键盘主动出现
为了让用户更省心,我们可以在一个带输入框的视图出现时就让键盘弹出来,而不用用户再点一下输入框了。方法很简单,就一行代码:[self.location_inputTextField becomeFirstResponder];
4.两个输入框按return时仿回车功能
有多个输入框,在一个输入框中按了return,然后好像在网站输入框中按了回车,直接跳到下一个输入框,这个也非常简单,就是resignFirstResponder和becomeFirstResponder方法结合使用而已,在UITextField的委托方法中实现:#pragma mark - UITextField Delegate - (BOOL)textFieldShouldReturn:(UITextField *)textField { if ([self.location_inputTextField isFirstResponder]) { [self.location_inputTextField resignFirstResponder]; [self.length_inputTextField becomeFirstResponder]; } else if ([self.length_inputTextField isFirstResponder]) { [self.length_inputTextField resignFirstResponder]; } return YES; }
Demo已经上传,有兴趣的可以下载看看:点此进入下载页
相关文章推荐
- Assertion failure in -[UITableView _configureCellForDisplay:forIndexPath:]
- 重装win7+uefi+gpt
- String、StringBuffer和StringBuilder的区别
- uitextfield
- EasyUI DataGrid动态加载数据
- iOS UIView动画实践(五):Keyframe Animation
- iOS UIView动画实践(四):过渡与伪3D动画
- UIPickerView控件中自定义显示的字体大小及样式
- UI进阶——数据请求
- iOS UIView动画实践(三):Transition Animation
- iOS UIView动画实践(一)揭开Animation的神秘面纱
- poj2299Ultra-QuickSort【线段树求逆序数】离散化
- finish with non-zero exit value 42
- jmeter 性能测试 JDBC Request (查询数据库获取数据库数据) 的使用
- 教你如何用PS制作多款按钮UI设计教程
- OCiOS开发:UIKit 力学(UIDynamics)(二)
- iOS用xib文件创建UITableView
- iOS Controller 瘦身运动之 UItableView datasource 与 delegate的分离
- leetcode:Unique Paths II
- EasyUI加载树控件自动展开所有目录