一、介绍

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

二、思想

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. OSPF协议之详细图解

    OSPF是一种基于SPF算法的链路状态路由协议. 上图是在一个OSPF区域里面添入一台新的路由器的时候,OSPF协议的工作过程,如果你能非常详细的叙述出这张图的话,基本上OSPF协议的工作过程你就掌握 ...

  2. NOI2018Day2T1 屠龙勇士 set 扩展欧几里德 中国剩余定理

    原文链接https://www.cnblogs.com/zhouzhendong/p/NOI2018Day2T1.html 题目传送门 - 洛谷P4774 题意 题解 首先我们仔细看一看样例可以发现如 ...

  3. 3186Treats for the Cows(区间dp)

    题意:给一个数组v,每次可以取前面的或者后面的,第k次取的v[i]价值为v[i]*k,问总价值最大是多少. 区间dp. 区间dp可以不枚举len  直接枚举i和j即可  见代码 #include &l ...

  4. Lua + win 10 + vs2017的运行环境和创建cocos2dx 3.17的lua项目(亲测)

    转:https://blog.csdn.net/maoye198602102339/article/details/82047920   不管用什么引擎写游戏,脚本语言是少不了要接触的! 首先,我说的 ...

  5. HDU 3749 Financial Crisis (点双连通+并查集)

    <题目连接> 题目大意: 给你一个(保证输入无重边,无自环)无向图,然后有下面Q条询问,每条询问为:问你u点与v点之间有几条(除了首尾两点外,其他点不重复)的路径.如果有0条或1条输出0或 ...

  6. Python库: PrettyTable 模块

    一,PrettyTable简介 PrettyTable是python中的一个第三方库,可用来生成美观的ASCII格式的表格: 二,PrettyTable安装 使用PIP即可十分方便的安装PrettyT ...

  7. hdu3038

    hdu3038带权并查集这种问题不仅仅要处理不同的点的是否在同一个集合里之类的问题,点与点之间存在连线,其带有权值,在路径压缩的时候也要对权值进行操作这道题就是带权并查集+向量去做 #include& ...

  8. [ 严重 ] my网SQL注入

    RANK  80 金币    100 数据包 POST maoyan.com/sendapp HTTP/1.1Host: xxx.maoyan.comUser-Agent: Mozilla/5.0 ( ...

  9. [dotnet core]使用Peach简化Socket网络通讯协议开发

    Peach是基于DotNetty的Socket网络通讯帮助类库,可以帮助开发者简化使用DotNetty,关于DotNetty可参考我之前的这篇文章. Peach内置实现了一个基于文本协议的Comman ...

  10. ASP.Net Core中使用jquery-ajax-unobtrusive替换Ajax.BeginForm

    在大潮流下,大家都在研究MVVM框架,但是做面向搜索引擎的外网项目还是得用服务器渲染. 在.Net中肯定就是用Razor模板引擎了. .Net Core断臂式重构后,很多在老得Mvc中使用得好好的一些 ...