tip 1 : 给UIImage添加毛玻璃效果
func blurImage(value:NSNumber) -> UIImage {
let context = CIContext(options:[KCIContextUseSoftwareRenderer:true])
let ciImage = CoreImage.CIImage(image:self)
let blurFilter = CIFilter(name:"CIGassianBlur")
blurFilter?.setValue(ciImage, forKey:KCIInputImageKey)
blurFilter?.setValue(value, forKey:"inputRadius")
let imageRef = context.createCGImage((blurFilter?.outputImage)!, fromRect:(ciImage?.extent)!)
let newImage = UIImage(CGImage:imageRef)
return newImage
}
value : value代表毛玻璃效果的程度
核心内容:let blurFilter = CIFilter(name:"CIGassianBlur") 使用滤镜工具
tip 2 : 图片压缩
func imageCompress(targetWidth:CGFloat) -> UIIMage {
let targetHeight = targetWidth/width*height
UIGraphicsBeginImageContext(CGSizeMake(targetWidth,targetHeight))
self.drawInRect(CGRectMake(0,0,targetWidth,targetHeight))
let newImage:UIImage = UIGraphicsGetImageFromCurrentImageContext()
UIGrapicsEndImageContext()
return newImage
}
这里是按原UIImage比例做压缩所以:let targetHeight = targetWidth/width*height
tip 3 : SVN & Git 用法总结
一:SVN
A. 项目经理:1.创建项目—CheckIn
2.设置项目人员
B.工作人员:1.CheckOut 获取项目的完整副本,此工作只需要做"一次"
2. 工作写代码....
3.阶段性工作完成后,Commit(提交) 将自己修改的文件,上传到服务器
每天下班前一定要做一次能够编译的提交!
4.定期Update项目,从服务器将最新的内容更新到本地,每天上班第一件事情一定要做的事情!
二. Git
A.项目经理:1.创建项目push
2.设置项目人员
B. 工作人员:1.Pull从服务器下拉最新的本版
2.Commit是将修改过的代码提交至本地的代码库
3.每天下班前Push自己当前的代码到服务器
4.每天上班前从服务器Pull最新的内容
三. M / A 文件更新
对于文件夹svn支持并不好,需要在工具下选定文件夹commit
对应文件选定文件commits
由于过去在公司多半时间是做独立开发,最多人的时候也是两个人做开发,所以协作工具用的少,但是不断的关注着两种代码仓库管理工具,总结了一些tip,还有待实践验证,吐槽吧......
tip 4: UINavigationController下的坐标系
iOS 9 前:
navigationBar 如果设置了背景图片,那么可视化坐标会从navgationbar下面开始
self.navigationController?.navigationBar.setBackgroundImage(UIImage(named:"nav"), forBarMetrics:UIBarMetrics.Default)
iOS 9 后 : 坐标从状态栏下面开始
tip 5 : C字符串转成NSString & NSString转成C字符串
const char *cString = "CString";
C字符串转成NSString : NSString *str = [NSString stringWithUTF8String:cString];
NSString * str = @"NSString";
NSString转成C字符串 : const char *cString = [str UTF8String];
tip 6 : OC & Swift 代码块(语法糖)
Objective-C :
UILabel *label1 = ({
UILabel *label = [UILabelnew];
[self.view addSubview:label];
label.frame=CGRectMake(100,100,100,45);
label.backgroundColor= [UIColor redColor];
label.text=@"大家好1";
label;
});
UILabel*label2 = ({
UILabel*label = [UILabel new];
[self.view addSubview:label];
label.frame=CGRectMake(100,160,100,45);
label.backgroundColor= [UIColor redColor];
label.text=@"大家好2";
label;
});
UILabel*label3 = ({
UILabel*label = [UILabel new];
label.frame=CGRectMake(100,250,100,45);
label.backgroundColor= [UIColor redColor];
label.text=@"大家好3";
label;
});
[self.viewaddSubview:label3];
({
UILabel *label = [UILabel new];
[self.view addSubview:label];
label.frame=CGRectMake(100,310,100,45);
label.backgroundColor= [UIColor redColor];
label.text=@"大家好4";
});
Swift:
letlabel1:UILabel= {
let label =UILabel()
self.view.addSubview(label)
label.frame=CGRectMake(100,100,100,45)
label.backgroundColor=UIColor.redColor()
label.text="大家好"
return label
}()
let label2:UILabel= {
let label =UILabel()
label.frame=CGRectMake(100,200,100,45)
label.backgroundColor=UIColor.redColor()
label.text="大家好"
return label
}()
self.view.addSubview(label2)
使用语法糖的好处就是拷贝代码时只需做少许的修改就可以达到目的,如上面的栗子,想要创建多个label,只要赋值粘贴,改一处,也就是对象名称就可以轻松完成!
tip 7 : 数据持久化方式归纳总结
数据缓存方式选择:
1: fmdata数据库(增删改查) --数据需要:增删改查
2: coreData --数据需要:增删改查
3:序列化(NSUserDefault) --状态、偏好设置、默认选项
4:单独.plist --列表数据,城市信息等
5:单独.json文件 --页面列表数据
6: realm框架(增删改查) --数据需要:增删改查
7: FastCoder 某“强哥”推荐,哈哈哈!
tip 8 : 清理Profiles证书文件
~/Library/MobileDevice/Provisioning Profiles
由于平时会负责多个项目的上线管理或是开发工作,因此MAC中有很多签名文件,有时候都分不清东西南北,一不做,二不休,前往这个目录下,将文件删个精光,调试的时候用到证书再update一下当前项目的证书即可
tip 9 : 拿到当前屏幕所看到的viewController
Objective-c版本:
- (UIViewController *)getAppRootViewController
{
UIViewController *appRootVC = [UIApplication sharedApplication].keyWindow.rootViewController;
UIViewController *topVC = appRootVC;
while (topVC.presentedViewController) {
topVC = topVC.presentedViewController;
}
return topVC;
Swift版本:
func getAppRootViewController() -> UIViewController? {
var topVC = UIApplication.sharedApplication().keyWindow?.rootViewController
while topVC?.presentedViewController != nil {
topVC = topVC?.presentedViewController
}
return topVC?
}
tip 10 : 制作pch文件
步骤:
1、新建iOS->Other->PCH File
2、targets->BuildSetting->Prefix Header->设置$(SRCROOT)/文件在工程中的路径
3、pch能像以前一样正常使用
如:$(SRCROOT)/FirstProject/PrefixHeader.pch
FirstProject : pch路径中的最后一个拓展名
PrefixHeader.pch: 是pch文件名
简介:/Users/ly/Desktop/FirstProject/FirstProject
tip 11 : 设置UINavigationController title
当 UIViewController作为UINavigationController的根视图控制器的时候,将这个Nav(root)赋给 tabBarController时,
这样写:
root.title = @“title”;
那么 :self.naviagtionItem.title 会显示 title
同时 :nav.tabBarItem.title 也会显示 title
tip 12 : 判断UIScrollView是横向滚动还是竖向滚动
- (BOOL)gestureRecognizerShouldBegin:(UIGestureRecognizer*)gestureRecognizer {
if([gestureRecognizerisKindOfClass:[UIPanGestureRecognizerclass]]) {
CGPointvelocity = [(UIPanGestureRecognizer*)gestureRecognizervelocityInView:gestureRecognizer.view];
BOOLisHorizontalPanning =fabsf(velocity.x) >fabsf(velocity.y);
return isHorizontalPanning;
}
returnYES;
}
tip 13 : 监听 backBarButtonItem 返回事件
github上搜: UIViewController+BackButtonHandler 开源项目
tip 14 : 让一个view透明度失效
self.view.backgroundColor=UIColor(colorLiteralRed:255.0/255, green:255.0/255, blue:255.0/255, alpha:0.3)
let subView = UIView()
subView.tintColor=UIColor.whiteColor()//不透明了
self.view.addSubview(subView)
tip 15 : 从ipod库中读出音乐文件
// 从ipod库中读出音乐文件
MPMediaQuery *everything = [[MPMediaQuery alloc] init];
// 读取条件
MPMediaPropertyPredicate *albumNamePredicate = [MPMediaPropertyPredicate predicateWithValue:[NSNumber numberWithInt:MPMediaTypeMusic] forProperty: MPMediaItemPropertyMediaType];
[everything addFilterPredicate:albumNamePredicate];
NSLog(@"Logging items from a generic query...");
NSArray *itemsFromGenericQuery = [everything items];
for (MPMediaItem *song in itemsFromGenericQuery) {
NSString *songTitle = [song valueForProperty: MPMediaItemPropertyTitle];
NSLog (@"%@", songTitle);
}
tip 16 : 广告标示符(adId) & adfv标示符的那些事
1.如何识别一个应用安装在同一个设备上呢?
2.如何识别一个企业的应用安装在同一个设备上呢?
苹果给我们提供了advertisingIdentifier 来解决问题1;
只要是同一台设备,那么advertisingIdentifier就是一样的
但是如果在设置-隐私-广告那里关掉这个权限或是还原设备的话,就没办法了哭死去吧
苹果给我们提供了identifierForVendor 来作为一个企业的app标示符
比如: com.game.yoyo
com.game.xoxo
只要在同一台设备上,那么identifierForVendor 是一样的
如果:com.game.yoyo
com.buyer.yoyo
不管是不是同一个应用identifierForVendor 都是不一样的
上代码:
广告id:
#import
//每个设备有唯一一个,如果重置广告或设置-隐私-关闭广告就会关闭更换
NSString*adId = [[[ASIdentifierManagersharedManager]advertisingIdentifier]UUIDString];
企业id:
NSString*idfv = [[[UIDevicecurrentDevice]identifierForVendor]UUIDString];
tip 17 : 不使用JPush等的原始推送
由于简书的一些限制,不能粘贴文件,考虑选择一个方式将我自己的实践工程放出来,以后持续更新
(MAC下搭建服务器+证书制作+推送测试)
tip 18 : 设置系统音量(排版有段乱,将就一下(⊙﹏⊙)b)
#import"VolumeSetter.h"
#import
@implementationVolumeSetter
{
MPVolumeView*volumeView;
UISlider* volumeViewSlider;
}
+ (instancetype)shareInstance
{
staticVolumeSetter*vs;
staticdispatch_once_tonce;
dispatch_once(&once, ^{
vs = [[VolumeSetteralloc]init];
});
returnvs;
}
- (void)setVolume:(float)value
{
[self getSlider];
[volumeViewSlidersetValue:valueanimated:NO];// send UI control event to make the change effect right now.
[volumeViewSlidersendActionsForControlEvents:UIControlEventTouchUpInside];
}
- (float)getVolume:(volumeBlock)handel
{
_myHandel= handel;
[selfgetSlider];
returnvolumeViewSlider.value;
}
- (MPVolumeView*)getVolumeView
{
if(volumeView==nil) {
volumeView= [[MPVolumeViewalloc]init];
[[NSNotificationCenterdefaultCenter]addObserver:self
selector:@selector(volumeChanged:)
name:@"AVSystemController_SystemVolumeDidChangeNotification"
object:nil];
}
returnvolumeView;
}
- (UISlider*)getSlider
{
[selfgetVolumeView];
for(UIView*viewin[volumeViewsubviews]){
if([view.class.descriptionisEqualToString:@"MPVolumeSlider"]){
volumeViewSlider= (UISlider*)view;break;
}
}
returnvolumeViewSlider;
}
- (void)volumeChanged:(NSNotification*)notification
{
floatvolume =
[[[notificationuserInfo]
objectForKey:@"AVSystemController_AudioVolumeNotificationParameter"]
floatValue];
NSLog(@"current volume = %f", volume);
_myHandel(volume);
}
@end
tip 19 : 提高开发效率 开发工具(转)
tip 20 : XMPP 开发与服务器搭建
待补充
tip 21 : 设置UINavigationItem 返回按钮 和 back title
1. push 之前
// back title
self.navigationItem.backBarButtonItem=UIBarButtonItem(title:"返回", style:UIBarButtonItemStyle.Plain, target:nil, action:nil)
// 特别注意:在push之前设置
self.navigationController?.pushViewController(vc, animated:true)
2.设置颜色
self.navigationController?.navigationBar.tintColor=UIColor.blackColor()
总结:
self.navigationItem.backBarButtonItem=UIBarButtonItem(title:"返回", style:UIBarButtonItemStyle.Plain, target:nil, action:nil)
self.navigationController?.navigationBar.tintColor=UIColor.blackColor()
self.navigationController?.pushViewController(vc, animated:true)
tip 22 : 语法规范
tip 23 : 谓词
Person类:
@interfacePerson :NSObject
@property(nonatomic) NSUIntager age;
@property(nonatomic,copy)NSString*name;
@end
#import"Person.h"
@implementationPerson
- (NSString*)description
{
return[NSStringstringWithFormat:@"age=%d,name=%@",self.age,self.name];
}
@end
ViewController.m类:
Person*p1 = [[Personalloc]init];
p1.age=15;
p1.name=@"张三";
Person*p2 = [[Personalloc]init];
p2.age=25;
p2.name=@"小米";
Person*p3 = [[Personalloc]init];
p3.age=15;
p3.name=@"李四";
NSArray*arr =@[p1,p2,p3];
//谓词
NSPredicate*pre = [NSPredicatepredicateWithFormat:@"%K==%d || %K==%@ || %K=%@",@"age",15,@"name",@"张三",@"name",@"李四"];
arr = [arrfilteredArrayUsingPredicate:pre];
NSLog(@"%@",arr);
输出:
2015-12-08 11:24:05.862 Predicate[12080:634959] (
"age=15,name=\U5f20\U4e09",
"age=15,name=\U674e\U56db"
)
用于做查询操作非常棒,避免写一大堆的if else操作
tip 24 : Xcode 插件
VVDocumenter 是一个比较好用的注释控件
Alcatraz(插件管理器)
Xcode 6.4找不到packageManager :
tip 25 : 微信登录,分享等 不执行代理方法
方法一:
- (BOOL)application:(UIApplication*)application handleOpenURL:(NSURL*)url
{
return [WXApihandle OpenURL:urldelegate:self];
}
方法二:
- (BOOL)application:(UIApplication*)application openURL:(NSURL*)url sourceApplication:(NSString*)sourceApplication annotation:(id)annotation
{
return[WXApihandle OpenURL:url delegate:self];
}
今天做微信登陆功能,随手写一个demo,没有设置bundle id , display name,结果就不走 方法一,方法二
导致下面两个代理方法不走
- (void) onReq:(BaseReq*)req{
NSLog(@"xxxxxxxx");
}
- (void) onResp:(BaseResp*)resp{
NSLog(@"fffffffff");
}
解决方法:设置Bundle identifier 以及 Bundledisplayname ,注意要与注册获取appid secret key 时所填的保持一致.
当然其他设置也要保证设置上,比如 URL sechme , iOS9 注意适配 ATS,添加白名单
tip 26 : iOS7.0后隐藏状态栏(UIStatusBar)
现象:
升级到iOS7后,UIStatusBar的出现导致现有UI界面乱掉了。
原因:
由于写死了某些控件的绝对位置,原先隐藏UIStatusBar的代码没有在iOS7中起作用
解决方法:
iOS7以下版本隐藏UIStatusBar的方法:
- (BOOL)application:(UIApplication*)applicationdidFinishLaunchingWithOptions:(NSDictionary*)launchOptions
{
[application setStatusBarHidden:YES];
returnYES;
}
升级到iOS7后的方法:
在基类中重载UIViewController.h中的这个方法
- (BOOL)prefersStatusBarHidden NS_AVAILABLE_IOS(7_0);// Defaults to NO
- (BOOL)prefersStatusBarHidden
{
return YES;
// iOS7后以下方法已经不起作用: [[UIApplication sharedApplication] setStatusBarHidden:YES withAnimation:UIStatusBarAnimationFade];
}
tip 27 : UIColor 转 RGB
UIColor*color = [UIColorcolorWithRed:0.0green:0.0blue:1.0alpha:1.0];
constCGFloat*components =CGColorGetComponents(color.CGColor);
NSLog(@"Red: %f", components[0]);
NSLog(@"Green: %f", components[1]);
NSLog(@"Blue: %f", components[2]);
tip 28 : 抠图 & 合成图片 (转)
iOS开发-简单图片背景替换(实现抠图效果)
tip 29 : Mac 生成UUID 命令: uuidgen
在命令行工具输入 $ uuidgen : 8-4-4-4-12
tip 30 : 23种设计模式
tip 31 : 发布APP要求
发布APP要求,不能使用Beta版本的Xcode
如果一个开发团队里面有多个小伙伴,不要为了尝新将所有的Mac OX 或是Xcode都升级到Beta版本,应该由一个小伙伴进行测试感受新版本是否稳定,是否有坑,否定后果很严重,说多都是泪,您想想:重新装系统,重新下载Xcode是一件多么苦逼的事情。。。
tip 32 : UIScrollView 判断是向左滚动还是向右滚动
这是抄自项目里面的一段代码,如果您要用到判断UIScrollView向左还是向右的滚动逻辑,请先定义相关状态的全局变量
-(void)scrollViewWillBeginDragging:(UIScrollView*)scrollView
{
startContentOffsetX= scrollView.contentOffset.x;
}\
- (void)scrollViewWillEndDragging:(UIScrollView*)scrollView withVelocity:(CGPoint)velocity targetContentOffset:(inoutCGPoint*)targetContentOffset{//将要停止前的坐标
willEndContentOffsetX= scrollView.contentOffset.x;
}
- (void)scrollViewDidEndDecelerating:(UIScrollView*)scrollView{
endContentOffsetX= scrollView.contentOffset.x;
if(endContentOffsetX < willEndContentOffsetX && willEndContentOffsetX < startContentOffsetX) {
NSLog(@"画面从右往左移动,前一页");
}else if(endContentOffsetX > willEndContentOffsetX&&
willEndContentOffsetX > startContentOffsetX) {
NSLog(@"画面从左往右移动,后一页");
}
}
tip 33 : 判断是否插入SiM卡
引入:CoreTelephony.framework
extern NSString*constkCTSMSMessageReceivedNotification;
extern NSString*constkCTSMSMessageReplaceReceivedNotification;
extern NSString*constkCTSIMSupportSIMStatusNotInserted;
extern NSString*constkCTSIMSupportSIMStatusReady;
id CTTelephonyCenterGetDefault(void);
void CTTelephonyCenterAddObserver(id,id,CFNotificationCallback,NSString*,void*,int);
void CTTelephonyCenterRemoveObserver(id,id,NSString*,void*);
int CTSMSMessageGetUnreadCount(void);
int CTSMSMessageGetRecordIdentifier(void* msg);
NSString* CTSIMSupportGetSIMStatus();
NSString* CTSIMSupportCopyMobileSubscriberIdentity();
id CTSMSMessageCreate(void* unknow/*always 0*/,NSString* number,NSString* text);
void* CTSMSMessageCreateReply(void* unknow/*always 0*/,void* forwardTo,NSString* text);
void* CTSMSMessageSend(idserver,idmsg);
NSString*CTSMSMessageCopyAddress(void*,void*);
NSString*CTSMSMessageCopyText(void*,void*);
NSLog(@"BOOL:%d", [CTSIMSupportGetSIMStatus()isEqualToString:kCTSIMSupportSIMStatusNotInserted] );
tip 34 : 获取链接的Wifi
#import <SystemConfiguration/CaptiveNetwork.h>
- (id)fetchSSIDInfo {
NSArray*ifs = (__bridge_transfer id)CNCopySupportedInterfaces();
NSLog(@"Supported interfaces: %@", ifs);
id info =nil;
for (NSString*ifnam in ifs) {
info = (__bridge_transfer id)CNCopyCurrentNetworkInfo((__bridge CFStringRef)ifnam);
NSLog(@"%@ => %@", ifnam, info);
if(info && [info count]) {break; }
}
return info;
}
tip 36 : TableView UITableViewStylePlain状态去掉省略多余的cell
_tableView= [[UITableView alloc] initWithFrame:self.view.bounds style:UITableViewStylePlain];
_tableView.delegate=self;
_tableView.dataSource=self;
_tableView.tableFooterView= [UIView new]; // 关键的一行代码
tip 37 : block 类型声明为strong 还是 copy ?
block声明为strong 和copy什么区别
block变量申明为strong或者copy 在ARC 的情况下 是等效的
但苹果推荐的最佳实践是 使用copy作为block的权限控制符
因为 copy能让开发者更明确地知道 block内存操作的隐式行为
及copy 到heap上
ARC下面 strong修饰的block 会自动进行这一过程
MRC下面 使用copy是必须的
tip 38 : 坐标转换
如果我的tableView的每个cell都有输入框,那我怎么样在输入的时候将对应的输入框移到合适的位置?
每个cell可以拿到在当前table的位置然后转换到屏幕上
然后自己算应该偏移多少
self是一个view 一个是把一个坐标从自己的换到其他的View上
一个是把其他的坐标换到自己上 如果self你当做tableView,然后后面的uiview你用vc.view
就能相互转了
tip 39 : 跳转至指定QQ号码
判断是否安装了QQ:
if([[UIApplication shareApplication] canOpenURL:[NSURL:URLWithString:@“mqq://“]]){
NSLog(@“had installed");
} else {
NSLog(@“no installed");
}
UIWebView*webView = [[UIWebView alloc] initWithFrame:CGRectZero];
NSURL*url = [NSURLURLWithString:@"mqq://im/chat?chat_type=wpa&uin=501863587&version=1&src_type=web"];
NSURLRequest*request = [NSURLRequest requestWithURL:url];
webView.delegate=self;
[webView loadRequest:request];
[self.view addSubview:webView];
tip 40 : 企业版app 与 appStore版app之间的问题
问题一 : 一台手机能否装两个名字一样的应用呢.
回答 : 能否装两个一样的应用不是取决于应用的名字,而是取决于boundid,appId(这个本人还没考究。。。)是否一致。
但是苹果应用商店是不允许有两个名字一样的应用程序的,那我们干嘛要纠结于名字一样的问题呢?
因为如果公司有一个企业账号和一个发布账号,企业账号希望发布一个到官网供用户下载,而发布账号希望发布到AppStore,而且名字一样。
结论是:两个应用同名行不行?行,只要你的企业账号发布的应用的boundid跟AppStore上的应用的app的boundid不一致,名字可以一致。
而且苹果不会管你的企业账号发布的应用程序,你爱怎么玩就怎么玩,如果你不希望一台手机装两个名字一样的应用,那么开发者 只要将两个应用的boundid设置成一样就可以了。
注意:AppStore上不能有同名的应用程序,这个是受商标保护法保护的。但是企业账号就可以任意起名字跟设置boundid。但是嘛,你懂的。。。
tip 41 : 打印 NSHomeDirectory为空
情况:打印获取 NSHomeDirectory为空,打印有值,获取到为nil。。
如果版本是release,就会出现上面的情况
解决的办法 : 设置成 debug
安全的做法是:
NSArray *paths = NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES);
NSString *basePath = ([paths count] > 0) ? [paths objectAtIndex:0] : NSTemporaryDirectory();
tip 42 : app跳转到safari
NSURL* url = [NSURL URLWithString:urlStr];
[[UIApplication sharedApplication] openURL:url];
tip 43 : 弹出键盘 & 收键盘
//监听键盘
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyBoardShow:)name:UIKeyboardWillShowNotification object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyBoardHide:)name:UIKeyboardWillHideNotification object:nil];
#pragma mark弹出键盘
-(void)keyBoardShow:(NSNotification*)notification
{
//notification.userInfo获得用户的所有信息,userInfo是一个字典,根据key值 为UIKeyboardFrameEndUserInfoKey拿到键盘的frame的“字符串”,将这个字符串转成Rect
int y = [[notification.userInfoobjectForKey:UIKeyboardFrameBeginUserInfoKey]CGRectValue].size.height;
}
#pragma mark收键盘
-(void)keyBoardHide:(NSNotification*)notification
{
}
最强的收键盘方法 :
[[UIApplicationsharedApplication]sendAction:@selector(resignFirstResponder)to:nilfrom:nilforEvent:nil];
其他的键盘监听 :
[[NSNotificationCenterdefaultCenter]addObserver:selfselector:@selector(keyboardWillChangeFrame:)name:UIKeyboardDidChangeFrameNotificationobject:nil];
tip 44 : 审核加急处理
链接:
备注:到上面的连接填写相关桂东,申请加急的理由:上线的App中有Bug