一、介绍

最近项目有一个需求,需要给朗诵的文字添加对应的拼音,而且要求使用原生的控件实现。一开始听到这个需求挺懵逼的,感觉有点难。后来,静下来想一下,其实还是可以实现的,无非就是自定义了。下面,就来说说思想。

二、思想

server端首先返回汉字和拼音一一对应的字符串,我们根据需要分别截取存入数组。一个汉字要添加拼音,拼音使用一个view显示,汉字也使用一个view显示,那么包含这两个控件的则是一个父view。许多个父view根据横向瀑布流的布局排列,就可以实现需要的汉字添加拼音的效果。结构图如下:

具体步骤如下:

1、创建一个textView,继承自UIScrollView,因为内容可能很多,需要滚动才能显示完;

2、在textView中接收模型时,动态创建很多的contanerView, 拼音View,汉字View,并使用masonry约束;

3、采用横向瀑布流算法对很多的contanerView进行布局;

4、在viewController中创建并添加textView,最后设置它的contentSize即可。

三、特点

优点:采用原生的控件实现,效果很流畅,用户体验好。

缺点:因为是一次性创建并添加很多的控件,如果文字特别多时,view渲染的过程比较耗时,建议放到自动释放池中进行创建。

四、代码

核心算法:

#pragma mark - layout subviews
-(void)setupSubviewsConstraints { //(子视图采用横向瀑布流布局)
CGFloat margin = 2.5;
CGFloat viewHeight = (kFont.pointSize+) * ;
CGFloat sumWidth = ;
CGFloat limitWidth = kScreenWidth-*margin;
CGFloat viewX = margin;
CGFloat viewY = margin;
CGFloat row = ; for (int i= ; i<self.contentArray.count; i++) { //实际宽度
CGFloat chineseWidth = [self layoutListNameLength:self.contentArray[i]];
CGFloat pinYinWidth = [self layoutListNameLength:self.pinYinArray[i]];
CGFloat maxWidth = MAX(chineseWidth, pinYinWidth); //添加容器
UIView *containerView = [[UIView alloc] init];
[self addSubview:containerView]; //添加拼音
UILabel *pinYinLabel = [[UILabel alloc] init];
pinYinLabel.textColor = [UIColor grayColor];
pinYinLabel.font = kFont;
pinYinLabel.text = self.pinYinArray[i];
pinYinLabel.textAlignment = NSTextAlignmentCenter;
[containerView addSubview:pinYinLabel]; //添加汉字
UILabel *chineseLabel = [[UILabel alloc] init];
chineseLabel.textColor = [UIColor blackColor];
chineseLabel.font = kFont;
chineseLabel.text = self.contentArray[i];
chineseLabel.textAlignment = NSTextAlignmentCenter;
[containerView addSubview:chineseLabel]; //设置约束
[containerView mas_makeConstraints:^(MASConstraintMaker *make) {
make.height.equalTo(@(viewHeight));
make.width.equalTo(@(maxWidth));
make.left.equalTo(@(viewX));
make.top.equalTo(@(viewY));
}];
[pinYinLabel mas_makeConstraints:^(MASConstraintMaker *make) {
make.left.top.right.equalTo(containerView);
make.height.equalTo(containerView.mas_height).multipliedBy(0.5);
}];
[chineseLabel mas_makeConstraints:^(MASConstraintMaker *make) {
make.left.bottom.right.equalTo(containerView);
make.height.equalTo(containerView.mas_height).multipliedBy(0.5);
}]; //计算坐标
sumWidth += (maxWidth + margin);
if (i < self.contentArray.count-) {
chineseWidth = [self layoutListNameLength:self.contentArray[i+]];
pinYinWidth = [self layoutListNameLength:self.pinYinArray[i+]];
maxWidth = MAX(chineseWidth, pinYinWidth);
}
if (limitWidth - sumWidth >= maxWidth) { ///不换行
viewX = sumWidth + margin;
}
else{ ///换行
row++;
sumWidth = ;
viewX = margin;
if (i==self.contentArray.count-) {
row--;
}
}
viewY = (viewHeight + *margin)*row + margin;
} self.realHeight = viewY + viewHeight + 1.5*margin;
}

完整代码:(在github上,觉得有用就给个star吧,

iOS:如何实现在文字上添加拼音的更多相关文章

  1. iOS 给NSString文字上添加横线 中间和下划线

    有时候我们需要给文字添加横线,有两种情况: 第一种是贯穿中间的横线: 横线的颜色和文字的颜色保持一致 _oldPriceLabel.text = "; _oldPriceLabel.text ...

  2. IOS 6 和 IOS7 UITableViewCell上添加控件的获取

    假设每个cell上面都有UIButton,怎么判断哪个Cell上的按钮被按下了呢? IOS6上 -(IBAction)btnClick:(id)sender { UIButton *btn = (UI ...

  3. Inno Setup技巧[界面]欢迎页面上添加文字

    原文:Inno Setup技巧[界面]欢迎页面上添加文字 本文介绍在"欢迎页面添加文字"的两种方法. 界面预览: Setup技巧[界面]欢迎页面上添加文字" title= ...

  4. python 图片上添加文字

    import PIL from PIL import ImageFont from PIL import Image from PIL import ImageDraw #设置字体,如果没有,也可以不 ...

  5. iOS: 获取UITableViewCell上添加的子控件对应的cell

    一.简单介绍 UITableViewCell是UITableView的核心部分,我们在开发中因为功能的扩展经常需要自定义,以便在其上面添加子控件,例如button.label等.添加后获取这些子控件的 ...

  6. Office WORD如何在图片上添加文字

    如图所示,在图片格式中选择图片衬于文字下方即可,这样看起来感觉就像在图片上直接加字一样,没有生硬的感觉. 最终效果: Word如何在图片上添加文字Word如何在图片上添加文字Word如何在图片上添加文 ...

  7. python如何在图片上添加文字(中文和英文)

    Python在图片上添加文字的两种方法:OpenCV和PIL 一.OpenCV方法 1.安装cv2 pip install opencv-python 2.利用putText方法来实现在图片的指定位置 ...

  8. iOS 在tableView上添加button导致按钮没有点击效果和不能滑动的 zhuang

    转载请注明出处. 今天在调试代码的时候,在tableviewcell上添加button,发现button快速点击的话,是看不出点击效果的,查找资料发现, ios7上UITableViewCell子层容 ...

  9. C#在图片上添加文字代码

    创建.NET WinForm程序,设置项目的默认命名空间为Keleyi.Com,在窗体上添加一个PictureBox控件pictureBox_keleyi_com和一个Button控件button_A ...

随机推荐

  1. fillder--客户端指定访问IP段

  2. 将程序sublime添加到右键菜单中

    新建wangzhaobo.bat复制一下代码, 粘贴保存,然后打开. @echo Off :START CLS echo *====================================== ...

  3. Java中byte、short、char、int、long运算时自动类型转化问题

    -------------------------------------------------------------------------------------------------- ★ ...

  4. 如何删除github里面的项目

    第一步 :首先登陆github 第二步:如下图选择 第三步:选择如下图 第四步:点击你要删除的项目,点击settings 第五步:把页面向下拉,找到如图按钮并点击 第六步:需要确认输入你要删除的项目名 ...

  5. netty05(netty的一些介绍)

    netty的一些理论 netty是一个异步事件驱动的网络应用框架(NIO框架),所有IO操作都是异步非阻塞的,NIO是对IO的一个补充 用于开发客户端和服务器的通信(TCP/UDP)长短连接 nett ...

  6. git命令详解( 五 )

    此篇只会来介绍rebase和merge的区别 rebase merge 区别 rebase 下面我们进行一个小练习来练习一下rebase 看一下题目要求: 共有三个特性分支 —— side1 side ...

  7. 杭电1024----Max Sum Plus Plus

    /* 这题还没有理解透彻.某个dalao也不写注释.只能自己理解了... 先求为i个元素(1<=i<=M)为一个区间的最大和,保证元素个数大于等于i个,递推到M个即可 借鉴原址:http: ...

  8. C#常用字符串函数

    Compare 比较字符串的内容,考虑文化背景(场所),确定某些字符是否相等 CompareOrdinal 与Compare一样,但不考虑文化背景 Format 格式化包含各种值的字符串和如何格式化每 ...

  9. Java并发编程(十二)-- 阻塞队列

    在介绍Java的阻塞队列之前,我们简单介绍一下队列. 队列 队列是一种数据结构.它有两个基本操作:在队列尾部加人一个元素,和从队列头部移除一个元素就是说,队列以一种先进先出的方式管理数据,如果你试图向 ...

  10. mysql case when then else end 的用法

    case when then end 改语句的执行过程是:将case后面表达式的值与各when子句中的值进行比较,如果两者相等,则返回then后的表达式的值,然后跳出case语 句,否则返回else子 ...