【iOS】知乎日报第三周总结
文章目录
前言
本周笔者因为金工实习整个项目进展比较慢,因为金工实习耽搁了挺长一段时间的,这周项目进展比较缓慢。下面简单介绍一下有关于这周学到的一些内容和遇到的一些问题。
评论区文字
笔者采用了一个textView实现了一个cell自适应文字的行高的效果,这里的实现主要是通过以下几个步骤来实现的,首先我们先定义uitableView的两个属性
objc
self.tableView.estimatedRowHeight = 100;
self.tableView.rowHeight = UITableViewAutomaticDimension;
一个是设置一个自适应行高的一个效果,另一个则是一个设置每一个cell的一个预估的高度,这样可以实现一个减小计算量的效果,从而提高效率。然后便是通过我们的一个cell中的控价实现一个撑开cell高度的效果,这里我们只要设置textView的时候需要注意一下我们一定要和cell的contentView来进行一个关联:
objc
[_headLabel mas_makeConstraints:^(MASConstraintMaker *make) {
make.top.equalTo(self.contentView).offset(10);
make.left.equalTo(self.contentView).offset(10);
make.width.equalTo(@100);
}];
[_image mas_makeConstraints:^(MASConstraintMaker *make) {
make.left.equalTo(self.contentView).offset(10);
make.top.equalTo(self.headLabel.mas_bottom).offset(20);
make.height.equalTo(@40);
make.width.equalTo(@40);
}];
[_nameLabel mas_makeConstraints:^(MASConstraintMaker *make) {
make.left.equalTo(self.image.mas_right).offset(10);
make.top.equalTo(self.headLabel.mas_bottom).offset(30);
make.width.equalTo(@200);
make.height.equalTo(@20);
}];
[_textView mas_makeConstraints:^(MASConstraintMaker *make) {
make.top.equalTo(self.image.mas_bottom);
//make.bottom.equalTo(self.contentView.mas_bottom).offset(-10);
//make.bottom.equalTo(self.contentView.mas_bottom).offset(-10);
make.left.equalTo(self.nameLabel);
//make.top.equalTo(self.contentView.mas_top).offset(5);
make.right.equalTo(self.contentView);
}];
[_commentText mas_makeConstraints:^(MASConstraintMaker *make) {
make.top.equalTo(_textView.mas_bottom);
make.left.equalTo(self.textView.mas_left);
make.right.equalTo(self.contentView.mas_right);
//make.bottom.equalTo(self.contentView.mas_bottom);
}];
[_timeLabel mas_makeConstraints:^(MASConstraintMaker *make) {
make.top.equalTo(_commentText.mas_bottom).offset(5);
make.bottom.equalTo(self.contentView.mas_bottom).offset(-10);
make.left.equalTo(self.textView).offset(5);
make.width.equalTo(@120);
}];
[_commentButton mas_makeConstraints:^(MASConstraintMaker *make) {
make.height.equalTo(@20);
make.bottom.equalTo(self.timeLabel.mas_bottom);
make.right.equalTo(self.contentView).offset(-20);
make.width.equalTo(@20);
}];
[_likeButton mas_makeConstraints:^(MASConstraintMaker *make) {
make.height.equalTo(@20);
make.bottom.equalTo(self.timeLabel.mas_bottom);
make.right.equalTo(_commentButton.mas_left).offset(-30);
make.width.equalTo(@30);
}];
这里我们需要注意下这里我们之所以不设置高度是为了让我们的文字来撑开textView,让textView来适应文字的高度来决定它控件的一个高度。再让我们的textView和一系列相关空间来撑开我们的一个cell。之后笔者会再写一篇博客总结所有的评论区的内容。
评论区的一个展开效果
笔者这里的展开效果存在一些问题,但是我大致可以实现一个展开的效果,这里先简单介绍一下如何实现,后面会在博客中重新总结一下。
这里笔者采用了一个NSMutableAttributedString
这个类来实现的,这个类可以给给某一句话设置点击事件,(虽然那个似乎是一个链接的效果)笔者也仅仅只是一个初步的学习,没有很理解这个类的一个具体实现,后面会在补充。下面笔者先给出代码:
objc
- (void)addTitleButton:(NSString*) suffixStr andContent:(NSString*) contentStr andAtrutbeString:(NSMutableAttributedString*) attStr{
NSLog(@"111");
if(suffixStr){
// NSLog(@"响应事件");
NSLog(@"添加响应事件:%@, %@, %@", suffixStr, attStr.string, contentStr);
NSRange range3 = [contentStr rangeOfString:suffixStr];
NSLog(@"%ld %ld", range3.location, range3.length);
NSLog(@"%@", [contentStr substringWithRange:range3]);
[attStr addAttribute:NSForegroundColorAttributeName value:[UIColor lightGrayColor] range:[contentStr rangeOfString:contentStr]];
[attStr addAttribute:NSForegroundColorAttributeName value:[UIColor grayColor] range:range3];
NSString *valueString3 = [[NSString stringWithFormat:@"didOpenClose://%@", suffixStr] stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet URLFragmentAllowedCharacterSet]];
NSLog(@"%@", valueString3);
[attStr addAttribute:NSLinkAttributeName value:valueString3 range:range3];
}
}
- (BOOL)textView:(UITextView *)textView shouldInteractWithURL:(NSURL *)URL inRange:(NSRange)characterRange {//设置textView的一个代理
if ([[URL scheme] isEqualToString:@"didOpenClose"]) {
CommentsTableViewCell* cell = (CommentsTableViewCell*)textView.superview.superview;
NSIndexPath* indexpath = [self.iView.tableView indexPathForCell:cell];
self.shortCommentsModel.comments[indexpath.row].isOpen = !self.shortCommentsModel.comments[indexpath.row].isOpen;
[self.iView.tableView reloadRowsAtIndexPaths:@[indexpath] withRowAnimation:UITableViewRowAnimationNone];//刷新某一行的cell
NSLog(@"响应事件");
return NO;
}
return YES;
}
这里笔者还没有彻底实现一个让展开二字紧紧跟在第二行的末尾的一个实现,所以笔者之后还要修改一下这部分代码,这部分代码的逻辑还存在着一些问题。
评论区数据的一个请求
评论区主要分成了两个部分,一个是长评论,一部分是短评论,我们需要让两个评论区的内容全部加载完在开始布局我们的tableView所以这里笔者采用了一个方法来封装我们获取长短评论的内容。
objc
- (void)allCommentsContentNsstring:(nonnull NSString *)string {
dispatch_group_t group = dispatch_group_create();
NSMutableArray* ary = [NSMutableArray array];
dispatch_group_enter(group);
[[Manger sharedManger] shortCommentContent:^(CommentsModel * _Nonnull model) {
self.shortCommentsModel = model;
dispatch_group_leave(group);
} andNsstring:string];
dispatch_group_enter(group);
[[Manger sharedManger] longCommentContent:^(CommentsModel * _Nonnull model) {
self.LongCommentsModel = model;
dispatch_group_leave(group);
} andNsstring:string];
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
self.iView.tableView.delegate = self;
self.iView.tableView.dataSource = self;
[self.iView.tableView registerClass:[CommentsTableViewCell class] forCellReuseIdentifier:@"comments"];
[self.iView.tableView reloadData];
});
}
但笔者对于GCD和AFN的原理还不是很清楚,对于这种实现是否正确还不太清楚,但笔者采用了这种方式来保证两个评论内容获取完毕之后才会进行一个加载。
修改了主页获取数据的逻辑
听从学长的一个建议将一次加载一个数据修改成了一次加载三个数据,让滑动更加流畅和自然。
主页无限轮播图图片主色调的一个获取
笔者找到了获取图片主色调的一个代码内容,但是还没有实现一个下方轮播图的一个模糊效果,这里也不太了解代码的原理实现,需要之后学习GCD之后再进行一个补充学习。
objc
//根据图片获取图片的主色调
-(void)getSubjectColor:(void(^)(UIColor*))callBack {
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
// 第一步 先把图片缩小 加快计算速度. 但越小结果误差可能越大
int bitmapInfo = kCGBitmapByteOrderDefault | kCGImageAlphaPremultipliedLast;
CGSize thumbSize = CGSizeMake(40, 40);
CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();
CGContextRef context = CGBitmapContextCreate(NULL,thumbSize.width,thumbSize.height, 8, thumbSize.width*4, colorSpace,bitmapInfo);
CGRect drawRect = CGRectMake(0, 0, thumbSize.width, thumbSize.height);
CGContextDrawImage(context, drawRect, self.CGImage);
CGColorSpaceRelease(colorSpace);
// 第二步 取每个点的像素值
unsigned char* data = CGBitmapContextGetData (context);
if (data == NULL) {
dispatch_async(dispatch_get_main_queue(), ^{
callBack(nil);
});
};
NSCountedSet* cls = [NSCountedSet setWithCapacity: thumbSize.width * thumbSize.height];
for (int x = 0; x < thumbSize.width; x++) {
for (int y = 0; y < thumbSize.height; y++) {
int offset = 4 * (x * y);
int red = data[offset];
int green = data[offset + 1];
int blue = data[offset + 2];
int alpha = data[offset + 3];
// 过滤透明的、基本白色、基本黑色
if (alpha > 0 && (red < 250 && green < 250 && blue < 250) && (red > 5 && green > 5 && blue > 5)) {
NSArray *clr = @[@(red),@(green),@(blue),@(alpha)];
[cls addObject:clr];
}
}
}
CGContextRelease(context);
//第三步 找到出现次数最多的那个颜色
NSEnumerator *enumerator = [cls objectEnumerator];
NSArray *curColor = nil;
NSArray *MaxColor = nil;
NSUInteger MaxCount = 0;
while ((curColor = [enumerator nextObject]) != nil){
NSUInteger tmpCount = [cls countForObject:curColor];
if ( tmpCount < MaxCount ) continue;
MaxCount = tmpCount;
MaxColor = curColor;
}
UIColor * subjectColor = [UIColor colorWithRed:([MaxColor[0] intValue]/255.0f) green:([MaxColor[1] intValue]/255.0f) blue:([MaxColor[2] intValue]/255.0f) alpha:([MaxColor[3] intValue]/255.0f)];
dispatch_async(dispatch_get_main_queue(), ^{
callBack(subjectColor);
});
});
}
将一些拓展部分的内容写在分类里
为了代码的一个高内聚的实现,我们会把一些相关实现的代码写在分类里面,这样让我们的代码更利于一个维护
小结
笔者这周因为金工实习浪费了很多时间,导致了这个项目的进展比较缓慢。笔者下周尽量完成这个项目的大致内容和以及完成相应的一个bug。