iOS开发(小小总结-)

1.1 UINavigationController

导航栏控制器基本是项目必备啊,所以我们UI大姐们也是想着花样搞这个东西。我们做这方面开发的都是到NVC(码字 有点烦,后面都这样简称了),基本构造做开发的都是知道,是一个容器类 下面的VCs都是放在 stack中的。所以说,在stack里面那么多VC都是公用一个navigationBar啊。UI大姐就喜欢一个VC 拥有一个不同的bar....欲哭无泪。

现在按按修改程度,一点一点写。

1.1.1bar 上的返回按钮太丑了 

~~第一种方法就是原本backItem不动,将文字移到看不见的地方,并且更换返回图片。个人感觉不是很好

~第二个方法就是整个替换掉leftItem,当然随之而来的问题就是要实现点击返回功能和左滑返回功能。

因为要实现左滑 最好还是重写一个NVC子类 而不是用appearce

//遵循手势的协议@interfaceIGONavigationController()- (void)viewDidLoad{    [superviewDidLoad];self.interactivePopGestureRecognizer.delegate=self;}//替换掉各个页面的返回按钮- (void)pushViewController:(UIViewController*)viewController animated:(BOOL)animated{//设置返回按钮viewController.navigationItem.leftBarButtonItem= [[UIBarButtonItemalloc] initWithImage:[UIImageimageNamed:@"fanhui"] style:UIBarButtonItemStyleDonetarget:selfaction:@selector(popViewControllerAnimated:)];    [superpushViewController:viewController animated:animated];}//右滑返回的处理- (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer*)gestureRecognizer{BOOLok =YES;// 默认为支持右滑反回if([self.topViewControllerisKindOfClass:[RootViewController class]] ||        [self.topViewControllerisKindOfClass:[Root1ViewController class]] ||        [self.topViewControllerisKindOfClass:[Root2ViewController class]] ||        [self.topViewControllerisKindOfClass:[Root3ViewController class]]) {returnNO;    }returnok;}

要注意的是 开起左滑 返回 移动去去掉 NVC 的rootVC 不然就会出现空stack的问题,从而导致各种问题。

1.1.2bar  有的页面需要,有的页面不需要

这个也是挺烦的,我看到很多同鞋。在处理这个问题上,都会这么写。加入A页面需要,B页面不需要的话

A- (void)viewWillAppear:(BOOL)animated {self.navigationController.navigationBar.hidden=NO;}B- (void)viewWillAppear:(BOOL)animated {self.navigationController.navigationBar.hidden=YES;}- (void)viewWillDisappear:(BOOL)animated {self.navigationController.navigationBar.hidden=YES;}

这种做法,也就效果勉强出来了,不过会带来一系列的问题

·····1 左滑 返回的时候NVC 无法分清那个VC 有Bar .出现显示问题

·····2当2个VC对应 statusBar 的字体颜色不一样的时候。上面这种做法,虽然隐藏了Bar 但在VC中重写preferredStatusBarStyle依旧没有作用的。NVC很难分清其中的界限。

正确的做法应该是

这种修改Bar 的工作还得交给代理去做,毕竟出现的意外情况,苹果的大牛们都考虑到了,代码要写在改写的地方。

因为我子类化得NVC用的地方很多,所以对于Bar的代理 我基本会放在rootVC中

下面的代码 就是在 SearchViewController和SearchResultController中隐藏了Bar

在ProductDetailViewController中对Bar的样式 进行了修改。

//在rootVC 中 遵循NVC代理 UINavigationControllerDelegate//隐藏导航栏- (void)navigationController:(UINavigationController*)navigationController willShowViewController:(UIViewController*)viewController animated:(BOOL)animated{// 隐藏BarBOOLisShowHomePage = [viewController isKindOfClass:[SearchViewController class]] || [viewController isKindOfClass:[SearchResultController class]];    [self.navigationControllersetNavigationBarHidden:isShowHomePage animated:YES];UIImage*shadowImage =self.navigationController.navigationBar.shadowImage;UINavigationBar*bar =self.navigationController.navigationBar;if([viewController isKindOfClass:[ProductDetailViewController class]]) {        bar.shadowImage= [UIImagenew];        [bar lt_setBackgroundColor:[UIColorclearColor]];    }else{        bar.shadowImage= shadowImage;        [bar setBackgroundImage:[UIImageimageNamed:@"导航栏"] forBarMetrics:UIBarMetricsDefault];    }}

这样 在左滑的时候消除了显示问题,以及在在隐藏了Bar的VC中无法控制statusBar的状态颜色问题。

1.1.3 每个bar完全不同

童鞋 全隐藏 自定义吧QAQ

1.2UINavigationBar

大家现在都是最低支持7以上了,太老的东西就不涉及了。

bar 有2个烦人的地方就是偏移量和透明度了

1.2.1偏移量

我们知道默认translucent = YES。就是Bar 是有透明度的。

在有透明度的情况下,系统默认automaticallyAdjustsScrollViewInsets属性是YES就是对于scrollerview的子类会默认content偏移64个单位。这样做的目的,既然Bar都是透明的了,系统就觉得你的scrollerView一定在会在(0,0)点,content偏移一个64个单位。在你滑动的时候,隐藏在Bar下面的Content会有一个模糊显示的效果。QAQ

如果你不想要这个偏移量

1 设置translucent = NO,既然Bar不透明。自然不需要偏移量喽

2 关闭这个偏移量,automaticallyAdjustsScrollViewInsets = NO

提一下像tableView,在storyboard 设置上下左右的约束。结果cell上面多了一块空白。就是这个问题。

1.2.2透明度

这种需求现在太多了,尤其都要做成淘宝那样的详情页,烦的一腿。现在我就在分析一下。

首先如果想滑动控制Bar的透明度,translucent 属性一定要设置成YES(默认)

当Bar 被设置成透明状态的时候

ED5A2936-7199-47B4-9E14-73E44F49BE74.png

还是这张图 _UIBackdropView 这个view是设置半透明毛玻璃效果的view(当然只在透明属性为YES 或者 没有背景图片的时候才会出现)

首先我们要去掉层特殊的视觉图层 利用设置BarBackground图片

- (void)setBackgroundImage:(nullableUIImage*)backgroundImage forBarMetrics:(UIBarMetrics)barMetricsNS_AVAILABLE_IOS(5_0)UI_APPEARANCE_SELECTOR;

千万不要用translucent = NO 这种方法来去掉视觉图层,都设置不透明了,那我们还玩按什么是吧。。。

这里注意的是我们设置的barTintColor 以及 background 都是对_UINavigationBarground这个Imageview来进行操作的。

这时候 我们就剩下Bar 和 BarBackground两层了

首先我们来说下Bar上面设置颜色的3个方法

//背景颜色 继承父类的 不过这个是设置bar 的背景颜色@property(nullable,nonatomic,copy)UIColor*backgroundColorUI_APPEARANCE_SELECTOR;// default is nil. Can be useful with the appearance proxy on custom UIView subclasses.//这个是设置bar 上面的那层 barBackground imaeView的颜色@property(nullable,nonatomic,strong)UIColor*barTintColorNS_AVAILABLE_IOS(7_0)UI_APPEARANCE_SELECTOR;// default is nil//这个是item的颜色@property(null_resettable,nonatomic,strong)UIColor*tintColor;

再来看下各个控件的高度 bar 本身是44 而barBackgoundImageView 确实64.。

这就是 为什么你设置了bar你颜色 或者 背景图片 。为什么status Bar 的背景也会跟着改变哦。

其实Bar 本身的颜色几乎很少设置,因为现在都要2个bar 一致才符合主流嘛

怎么改变barBackgroundImageView的背景颜色,又成了一个问题。因为刚才为了去掉视觉图层,我们给barBackgroundImageView 赋值了一个空图片。问题出来了,有了图片属性之后,给barBackgroundImageView设置背景颜色就没有用了,现在主流的做法是

查一个view进去  去调节这个 view的颜色 来控制bar 的背景颜色控制

写个类目 给 bar 加一个 - (void)iGo_setBackgroundColor:(UIColor *)backgroundColor

- (UIView*)backView{returnobjc_getAssociatedObject(self, &backView);}- (void)backView:(UIView*)backView{    objc_setAssociatedObject(self, &backView, backView, OBJC_ASSOCIATION_RETAIN_NONATOMIC);}- (void)iGo_setBackgroundColor:(UIColor*)backgroundColor{if(!self.backView) {        [selfsetBackgroundImage:[UIImagenew] forBarMetrics:UIBarMetricsDefault];self.backView= [[UIViewalloc] initWithFrame:CGRectMake(0, -20,CGRectGetWidth(self.bounds),CGRectGetHeight(self.bounds) +20)];            [selfinsertSubview:self.backViewatIndex:0];    }self.backView.backgroundColor= backgroundColor;}

最后 我们需要调节透明度的页面

在滑动过程中 调节颜色的透明度 就OK拉

- (void)scrollViewDidScroll:(UIScrollView*)scrollView{UIColor* color = [UIColorcolorWithHexString:@"#fa2245"];CGFloatoffsetY = scrollView.contentOffset.y;if(offsetY >50) {CGFloatalpha = MIN(1,1- ((50+64- offsetY) /64));        [self.navigationController.navigationBarlt_setBackgroundColor:[color colorWithAlphaComponent:alpha]];    }else{        [self.navigationController.navigationBarlt_setBackgroundColor:[color colorWithAlphaComponent:0]];    }}

2旋转处理

一半我我们的app不支持旋转的话 就会关掉下面这两个向选项

来禁止横屏。

但如果我们的app部分页面支持横屏,部分不支持的话就要做一些处理了

我们不需要对每一个VC都进行处理,只要对最外层的容器类VC重写一些方法。

例如我的所有VC都是由tabbar管理的

我会在tabbar的子类中

#pragma mark -- Orientation//如果目前是竖屏 则不支持旋转,如果现在是横屏 则支持旋转- (BOOL)shouldAutorotate{return!([UIApplicationsharedApplication].statusBarOrientation==UIInterfaceOrientationPortrait);}- (UIInterfaceOrientation)preferredInterfaceOrientationForPresentation{returnUIInterfaceOrientationPortrait;}- (UIInterfaceOrientationMask)supportedInterfaceOrientations {returnUIInterfaceOrientationMaskPortrait;}

在某些需要旋转的VC继续重写这几个方法

//因为是个别页面是横屏 其他页面都是竖屏,这里可以直接返回YES//如果有特殊情况 也可以判断返回#pragma mark -- orientation- (BOOL)shouldAutorotate {returnYES;}- (UIInterfaceOrientationMask)supportedInterfaceOrientations {returnUIInterfaceOrientationMaskLandscapeRight;}

这样即使是横屏状态打开app 也会自动切换成竖屏。不过如果你在tabbar 在加一些东西的话可能会出现问题。

中间这个 取 的按钮 ,按照系统的是无法实现的。又不想重新自定义tabbar。这时候我就在tabbar中中间加上一个Button.来模拟tabbar item 的功能。

这时候 从横屏 进来之后,就会出现问题,因为bar 上是无法添加约束的,所以只能用frame 布局。 但是横竖屏的宽高大小 是相反的0.0

这时候就需要判断了

//可以用宽高大小判断 也可以用上面的设备方向判断CGFloatwidth = kScreenHeight>kScreenWidth? kScreenWidth: kScreenHeight;UIButton*getGoodsButton = [[UIButtonalloc] initWithFrame:CGRectMake(width/2- kButtonWidth/2,0, kButtonWidth, kButtonHeight)];

这只是一盒小例子,总之 处理屏幕旋转的时候要注意布局。尤其是没有约束控制的布局

3扫描

3.1淘宝的扫描效果

这个看起来很炫酷,其实就是一个网格的图片上下移动,把多余的部分Mask 掉就可以了。

//放个可视面积差不多大小的view,将imageview 加上去UIView*tempView = [[UIViewalloc] initWithFrame:VISIBLERECT];//重点tempView.layer.masksToBounds=YES;    [self.viewaddSubview:tempView];    _scanningImageView = [[UIImageViewalloc] init];    [tempView addSubview:_scanningImageView];    _scanningImageView.image= [UIImageimageNamed:@"扫描网格"];

上下的移动的代码就不贴,思路知道了。做起来很简单

3.2 unsupported type found

在设置AVCaptureMetadataOutput的metadataObjectTypes时候,一定要注意设置在载入session 之后不然就会出现 特别迷惑人的错误

E9F4D45F-0FA8-426D-AA5C-11A7FEB11253.png

我纳闷了好久 啊 。。。。。要记得 前后 顺序

[_session addInput:_input];    [_session addOutput:_output];    _output.metadataObjectTypes= @[AVMetadataObjectTypeQRCode,AVMetadataObjectTypeEAN13Code,AVMetadataObjectTypeEAN8Code,AVMetadataObjectTypeCode128Code];

3.3 坐标转换

我们知道扫描的可视范围rectOfInterest坐标系统 和 我们 UIKit用的坐标系统不一样,我们在设置这个坐标的时候有两种做法

1,根据 两种坐标系 自己换算去 其实就是将屏幕左翻转 (x,y轴对调)

CGRectMake(y的起点/屏幕的高,x的起点/屏幕的宽,扫描的区域的高/屏幕的高,扫描的区域的宽/屏幕的宽)

2,通过系统的方法换算

在通过系统的方法换算的时候,出现了小问题

-(CGRect)metadataOutputRectOfInterestForRect:(CGRect)rectInLayerCoordinatesNS_AVAILABLE_IOS(7_0);

在使用这个方法换算的时候始终是没有作用的,

其实方法是没错,但是调用的时机很重要

当我们收到AVCaptureInputPortFormatDescriptionDidChangeNotification通知的时候在进行左边转换就没有问题了

具体原因,我也没有找到。希望大神指点

[[NSNotificationCenterdefaultCenter] addObserverForName:AVCaptureInputPortFormatDescriptionDidChangeNotificationobject:nilqueue:[NSOperationQueuecurrentQueue] usingBlock:^(NSNotification* _Nonnull note) {        _output.rectOfInterest= [_scanView metadataOutputRectOfInterestForRect:VISIBLERECT];    }];

4UIButton

我们都知道button中有2个子控件 一个imageView 一个label。默认imageview 在左label 在右。我们控制好这几个控件 能满足好多需求

首先 button 是UIControl的子类

UIControl有控制content位置的属性

typedefNS_ENUM(NSInteger,UIControlContentVerticalAlignment) {UIControlContentVerticalAlignmentCenter=0,UIControlContentVerticalAlignmentTop=1,UIControlContentVerticalAlignmentBottom=2,UIControlContentVerticalAlignmentFill=3,};typedefNS_ENUM(NSInteger,UIControlContentHorizontalAlignment) {UIControlContentHorizontalAlignmentCenter=0,UIControlContentHorizontalAlignmentLeft=1,UIControlContentHorizontalAlignmentRight=2,UIControlContentHorizontalAlignmentFill=3,};@property(nonatomic)UIControlContentVerticalAlignmentcontentVerticalAlignment;// how to position content vertically inside control. default is center@property(nonatomic)UIControlContentHorizontalAlignmentcontentHorizontalAlignment;// how to position content hozontally inside control. default is center

只要我们合理的运用 content(imageview 和 label) 可以在button 内部的16个位置上。

但有时这个东西满足不了我们的需求

看个例子

FE0CDCAA-451F-4ACF-97BA-74694BA7548B.png

这种图片在右边 文字在左边,就令我们很蛋疼。看到这个需求有的童鞋可能就会自定义一个视图类了,其实不需要button的属性就可以帮助我们解决

@property(nonatomic)UIEdgeInsetscontentEdgeInsetsUI_APPEARANCE_SELECTOR;// default is UIEdgeInsetsZero@property(nonatomic)UIEdgeInsetstitleEdgeInsets;// default is UIEdgeInsetsZero

可以设置图片文字的偏移量,不过此处有坑

#import"UIButton+Common.h"@implementationUIButton(Common)- (void)reverseImageAndTitle {CGRectimageFrame =self.imageView.frame;CGRecttitleFrame =self.titleLabel.frame;self.titleEdgeInsets=UIEdgeInsetsMake(0, -imageFrame.size.width*2,0,0);self.imageEdgeInsets=UIEdgeInsetsMake(0,0,0, -titleFrame.size.width*2);}@end

这边单位是px 不是pt 整个人都蒙了有木有。

不过还好发现了

5UITextField

iOS开发(小小总结-)_第1张图片

C04C5E25-F123-4372-A707-B14ACD5D7EAD.png

设置左边图片 sb 上面没有对应的属性。。。庆幸的在textfield中还是有对应的属性 不过切图的时候记得要来拿左边的空白一起切下来

@property(nullable,nonatomic,strong)UIView*leftView;// e.g. magnifying glass@property(nonatomic)UITextFieldViewModeleftViewMode;// sets when the left view shows up. default is UITextFieldViewModeNever

6.tableView collectionView

1.圆角 始终是个争论的话题了。

各种花式的解决办法。

其实我认为,如果固定的页面 如果只有个别原角,无需优化

如果想tableview 这样大量重复的圆角,其实只要做设置一次缓存即可。开启光栅化,可解决性能问题。

什么 离屏渲染 CPU GPU线程切换消耗大量性能,我只做一次还不行meQAQ

//rasterizecell.layer.shouldRasterize=YES;    cell.layer.rasterizationScale= [UIScreenmainScreen].scale;

不要各种贝塞尔 各种mask 简单点好。

2.对于selected

tableview 和 collectionview 都是支持单选的,只要重写  selected方法就行,系统在你创建cell类的时候也预留了这个方法。

不过要选择默认的selected 设置cell.seleted 是不行的

//collectionView- (void)selectItemAtIndexPath:(nullableNSIndexPath*)indexPath animated:(BOOL)animated scrollPosition:(UICollectionViewScrollPosition)scrollPosition;

seleted 可以完美的实现默认选中 单选的效果。

对于对选 智能自定义了

3section 展开

只要用数组记录每个section 需要展示的cell 的数量 刷新view 就可以了

7VC设置背景图片

没有必要再放在一个imageview上面 直接在VC的view.layer上画就可以了

self.view.layer.contents= (__bridgeid_Nullable)([UIImageimageNamed:@"beijing"].CGImage.

你可能感兴趣的:(iOS开发(小小总结-))