(iOS)希望这本《iOS自定义控件剖析》书籍能够让学习中的你有
已经一两个月没有更新过博客了, 不过这两个月并没有闲下来, 只是在完成以前欠下的的任务. 几个月前就开始动手写一本iOS开发方面的书籍, 希望这本书籍是实用性比较强的一本书. 不过在期间因为学习和生活上的琐事耽误了很多, 所以就被搁置了很久. 最近的一两个月, 终于又下定决心将这本未完成的书籍写完. 这两个月就把大多数的课余时间用在了完成这本书籍, 所以博客也很长时间没有更新过了. 不过还好, 目前已经顺利完成了.
书籍中实现了接近20个实际开发中将会遇到的一些效果, 下面是使用的方法和使用的效果. 即使你没有购买书籍, 也是可以获取到他们的源码,并且直接使用的, 因为笔者已经上传到了github上面了.你可以点击相关的链接直接下载. 当然如果购买了书籍, 书籍中将会详细讲解每一个效果的实现的, 你可以从零开始, 跟随着讲解, 一步一步的亲自实现这些效果. 相信你会有收获.然后你的github上面也是可以新增不少的responsibility了.
- 我最初学习iOS开发的时候, 是按照网上讲的iOS开发学习路径进行学习.购买了相关的书籍, 下载了相关的视频, 然后就开始了一段不长不短的学习历程. 当时swift的应用很不广泛, 所以最初是学习Objective-C. 当
学完
了相关的书籍和视频之后,自以为已经学得不错, 可以开发实际的项目了. 当时正好有机会参与到一个外包项目的开发. 不过当一开始的时候, 才真的发现自己太年轻了.书籍上学到的只是一些基本的语法和一些系统基本的API. 拿着设计给的产品原型, 毫无头绪怎么实现一个个看上去很简单的效果. 于是就只能各种的百度, Google, 然后才渐渐的实现了一些效果. 不过当时根本不能自定义系统提供的控件实现设计给的效果. 所以只能在github上面不断的找类似的效果的第三方. 记忆最深刻的是,当时好不容易在github上面找到一个圆形进度条, 但是项目中需要的效果不太一样, 需要在进度条的末端加上一个在圆环上滚动的圆点, 无奈, 当时能力不足, 修改实现这个简单的效果也用了一天多. 所以, 我一直很尊敬和感激
那些第三方的开发者, 在我最初开始iOS开发的时候提供了很多的帮助, 同时在以后的项目中提供了便利.尽管有些是很容易自己实现的简单的效果, 不过对于一些朋友来说, 可能也真的是需要的. 所以, 我也一直希望能够做一些类似的事情.
-
如果你正打算开始学习iOS开发, 那么市面上已经能够买到足够多的介绍Objective-C或者swift语法基础, 并且这些书中大多也会介绍iOS开发中所需要学习的系统库. 比如介绍UIKit中的常用控件来搭建App界面, 介绍绘图, 动画, 网络请求, 数据持久化, 定位... 不过这些书的内容大同小异, 所以你可以选择一本大家推荐的相关的书籍学习即可入门iOS开发, 当然, 本书不是讲解这些内容的.
-
如果你正打算购买阅读本书, 那么首先请确认它是否适合你目前来学习. 本书不是一本讲解iOS开发所需要的Objective-C或者swift语法的书, 也不是一本讲解iOS开发中需要学习的Apple的系统库(比如Foundation, CocoaTouch...)的书. 如果你是没有Objective-C或者swift语言基础的朋友, 那么请在开始阅读本书之前, 先学习Objective-C或者swift. 如果你没有学习过UIKit等iOS开发必须的系统框架, 并且没有用xcode真实构建过iOS项目, 那么请在开始阅读本书之前, 首先系统的学习iOS开发, 当你有能力自己构建项目的时候再来阅读本书.
-
本书是详细讲解iOS实际项目开发中常用到的一些效果的实现, 同时每一节我们都将从零开始实现一个常用的效果, 并且会将它逐步的封装得复用性更高, 可自定义程度更高, 让它也能成为一个可以供他人方便使用的'第三方'.
-
所以, 如果你是刚入门iOS开发不久, 想要学习实际项目中的一些效果的实现; 或者已经参与项目的开发,但是对于项目中需求实现总是没有思路, 需要不断的找第三方来拼凑, 想自己也能动手实现它们; 或者自己的github上面的responsibility很少, 想要在github上面增加一些responsibility... 那么这本书应该比较适合.
-
但是请一定要注意, 本书力求能够帮助需要的朋友, 所以讲解的很细致, 自然有些地方表述的可能显啰嗦. 不过, 希望你能有收获, 然后在阅读完本书之后, 自己的github上面至少能增加十个方便他人使用的responsibility.
-
下面的目录中的每一节都是一个在实际项目中会遇到的开发需求, 当然有特别简单的, 有中等一点的, 不过不用担心, 没有有难度的. 这些所有的源码我都已经上传到了github上面了, 但是
授人以鱼不如授人以渔
, 所以这本书中会带着你一步步的实现他们. -
第一章 广告展示类效果 ..................
-
第二章 提示类效果..................
-
第三章 列表类效果..................
-
第四章 侧滑类效果..................
-
第五章 其他常用效果..................
-
第六章 玩玩swift..................
</br>
- ZJLeadingPageController 最简单的引导页面demo. 使用可以是这样的.
// 如果是第一次安装打开App --- 显示引导页面
ZJLeadingPageController *leadController = [[ZJLeadingPageController alloc] initWithPagesCount:5 setupCellHandler:^(ZJLeadingPageCell *cell, NSIndexPath *indexPath) {
// 设置图片
NSString *imageName = [NSString stringWithFormat:@"wangyiyun%ld",indexPath.row];
cell.imageView.image = [UIImage imageNamed:imageName];
// 设置按钮属性
[cell.finishBtn setTitle:@"立即体验" forState:UIControlStateNormal];
[cell.finishBtn setTitleColor:[UIColor orangeColor] forState:UIControlStateNormal];
} finishHandler:^(UIButton *finishBtn) {
NSLog(@"点击了完成按钮-----");
}];
// 自定义属性
leadController.pageControl.pageIndicatorTintColor = [UIColor yellowColor];
leadController.pageControl.currentPageIndicatorTintColor = [UIColor purpleColor];
- ZJLaunchAd 展示启动广告, 一个简单, 方便的启动广告展示, 可以同时展示logo. 使用可能是下面这样的.
ZJLaunchAdController *launchVc = [[ZJLaunchAdController alloc] initWithLaunchImage:nil setAdImageHandler:^(UIImageView *imageView) {
// 这里可以直接使用SDWebimage等来请求服务器提供的广告图片(SDWebimage会处理好gif图片的显示)
// 不过你需要注意选择SDWebimage的缓存策略
imageView.image = [UIImage imageNamed:@"adImage"];
} finishHandler:^(ZJLaunchAdCallbackType callbackType) {
switch (callbackType) {
case ZJLaunchAdCallbackTypeClickAd:
// 点击了广告, 展示相应的广告即可
NSLog(@"点击了广告, 展示相应的广告即可");
break;
case ZJLaunchAdCallbackTypeShowFinish:
NSLog(@"展示广告图片结束, 可以进入App");
break;
case ZJLaunchAdCallbackTypeClickSkipBtn:
NSLog(@"点击了跳过广告, 可以进入App");
break;
}
}];
- ZJPPTView 可以玩出花样的轮播器, 内部不依赖第三方库, 使用简单, 可自定义轮播任何内容. 图片加载等类似tableView使用代理加载, 可自己选择第三方库来加载图片等.
_defaultPPT = [[ZJPPTViewDefault alloc] initWithDelegate:self];
_defaultPPT.pageControlPositon = ZJPPTViewPageControlPositionBottomCenter;
- (void)pptView:(ZJPPTViewOC *)pptView setUpPageCell:(UICollectionViewCell *)cell withIndex:(NSInteger)index {
if (pptView == _defaultPPT) {
ZJPPTViewDefaultCell *defaultCell = (ZJPPTViewDefaultCell *)cell;
// 可自定义文字属性 ...
// defaultCell.textLabel.backgroundColor = [[UIColor blackColor] colorWithAlphaComponent:0.5];
// defaultCell.textLabel.textAlignment = NSTextAlignmentCenter;
// defaultCell.textLabel.textColor = [UIColor whiteColor];
defaultCell.textLabel.text = [NSString stringWithFormat:@" 这是第: %ld 页", index];
// 设置图片 网络图片, 可自由使用SDWebimage等来加载
if (index%2 == 0) {
UIImage *image = [UIImage imageNamed:@"1"];
defaultCell.imageView.image = image;
}
else {
UIImage *image = [UIImage imageNamed:@"2"];
defaultCell.imageView.image = image;
}
}
}
- ZJProgressHUD 自已写个MBProgressHUD, 经常使用MBProgressHUD? 不妨自己动手来写一个, 反正也不难.
// 显示加载成功的图片和文字提示, 1s后自动隐藏
[ZJProgressHUD showSuccessWithStatus:@"加载成功!!" andAutoHideAfterTime:1.f];
// 显示加载动画, 需要加载完成后调用hideHUD隐藏
[ZJProgressHUD showProgressWithStatus:@"正在努力加载中..."];
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
// 加载完后 移除提示框
[ZJProgressHUD hideHUD];
});
- ZJActionSheet 自定义UIActionSheet, 系统的UIActionSheet用着很不方便自定义, 比如修改文字的颜色, 大小, 以及显示图片等很不方便, 那么我们可以自己写一个, 然后类似下面这样的效果.
ZJActionSheetItem *item8 = [[ZJActionSheetItem alloc] initWithTitle:@"可以自定义所有item的字体颜色 大小等" image:nil handler:^(ZJActionSheetView *actionSheet) {
NSLog(@"点击了收藏");
}];
ZJActionSheetItem *item9 = [[ZJActionSheetItem alloc] initWithTitle:@"可以设置actionSheet居中或者居下显示" image:nil handler:^(ZJActionSheetView *actionSheet) {
NSLog(@"点击了收藏");
}];
ZJActionSheetView *actionSheet = [[ZJActionSheetView alloc] initWithTitle:@"这是提示title" subtitle:@"这是详细说明文字,字体默认14,可修改subtitleLabel" actionSheetItems:@[item1, item2, item3,item4,item5,item6,item7,item8,item9]];
// 显示
[actionSheet show];
circleProgress.gif
self.progressView = [[ZJCircleProgressView alloc] initWithFrame:CGRectMake(100, 100, 100, 100)];
// 背景色
self.progressView.trackBackgroundColor = [UIColor yellowColor];
// 进度颜色
self.progressView.trackColor = [UIColor greenColor];
self.progressView.headerImage = [self drawImage];
// 开始角度位置
// self.progressView.beginAngle =
// 自定义progressLabel的属性...
self.progressView.progressLabel.textColor = [UIColor lightGrayColor];
// self.progressView.progressLabel.hidden = YES;
[self.view addSubview:self.progressView];
- (IBAction)slide:(id)sender {
UISlider *slider = (UISlider *)sender;
// 改变进度
self.progressView.progress = slider.value;
self.pieProgressView.progress = slider.value;
}
contacts.gif
NSArray *testArray = @[@"ZeroJ", @"曾晶", @"你好", @"曾晶", @"曾晶" , @"曾晶" , @"曾晶" , @"曾晶" , @"曾晶" , @"曾晶" , @"曾晶", @"曾好", @"李涵", @"王丹", @"良好", @"124"];
NSMutableArray<ZJContact *> *contacts = [NSMutableArray arrayWithCapacity:testArray.count];
for (NSString *name in testArray) {
ZJContact *test = [ZJContact new];
test.name = name;
test.icon = [UIImage imageNamed:@"icon"];
[contacts addObject:test];
}
[self setupInitialAllDataArrayWithContacts:contacts];
citySelecte.gif
ZJCityViewControllerOne *vc = [[ZJCityViewControllerOne alloc] initWithDataArray:nil];
// __weak typeof(self) weakSelf = self;
[vc setupCityCellClickHandler:^(NSString *title) {
NSLog(@"选中的城市是: %@", title);
[ZJProgressHUD showStatus:[NSString stringWithFormat:@"选中的城市是: %@", title] andAutoHideAfterTime:1.f];
// [weakSelf.navigationController popViewControllerAnimated:YES];
}];
[self.navigationController showViewController:vc sender:nil];
pickerView.gif
[ZJUsefulPickerView showSingleColPickerWithToolBarText:@"单列数据" withData:@[@"objective-C", @"swift", @"iOS", @"iPad", @"iPod", @"mac", @"java", @"php", @"JavaScript"] withDefaultIndex:3 withCancelHandler:^{
NSLog(@"quxiaole -----");
} withDoneHandler:^(NSInteger selectedIndex, NSString *selectedValue) {
NSLog(@"%@---%ld", selectedValue, selectedIndex);
}];
// 省市区选择
[ZJUsefulPickerView showCitiesPickerWithToolBarText:@"省市区选择" withDefaultSelectedValues:@[@"四川", @"成都", @"郫县"] withCancelHandler:^{
NSLog(@"quxiaole -----");
} withDoneHandler:^(NSArray *selectedValues) {
NSLog(@"%@---", selectedValues);
}];
drawer1.gif
drawer2.gif
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
ZJLeftViewController *left = [ZJLeftViewController new];
ZJCenterViewController *center = [ZJCenterViewController new];
UINavigationController *navi = [[UINavigationController alloc] initWithRootViewController:center];
ZJRightViewController *right = [ZJRightViewController new];
ZJDrawerController *drawer = [[ZJDrawerController alloc] initWithLeftController: left centerController:navi rightController:right];
// 背景图片
drawer.backgroundImage = [UIImage imageNamed:@"1"];
// 动画类型
drawer.drawerControllerStyle = ZJDrawerControllerStyleParallaxSlide;
// 任何界面都能打开抽屉
drawer.canOpenDrawerAtAnyPage = YES;
//...
self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];
self.window.backgroundColor = [UIColor whiteColor];
self.window.rootViewController = drawer;
[self.window makeKeyAndVisible];
return YES;
}
swipeTableViewCell.gif
- (NSArray<ZJSwipeButton *> *)tableView:(UITableView *)tableView leftSwipeButtonsAtIndexPath:(NSIndexPath *)indexPath {
ZJSwipeButton *leftBtn = [[ZJSwipeButton alloc] initWithTitle:@"检查1" image:nil onClickHandler:^(UIButton *swipeButton) {
NSLog(@"点击了检查1: --- %ld", indexPath.row);
[ZJProgressHUD showStatus:[NSString stringWithFormat:@"点击了检查1: --- %ld", indexPath.row] andAutoHideAfterTime:1];
}];
ZJSwipeButton *leftBtn1 = [[ZJSwipeButton alloc] initWithTitle:@"测试2" image:nil onClickHandler:^(UIButton *swipeButton) {
NSLog(@"点击了测试2: --- %ld", indexPath.row);
[ZJProgressHUD showStatus:[NSString stringWithFormat:@"点击了测试2: --- %ld", indexPath.row] andAutoHideAfterTime:1];
}];
return @[leftBtn,leftBtn1, leftBtn2,leftBtn3];
}
navigationController.gif
UIViewController *vc = [ViewController new];
ZJNavigationController *navi = [[ZJNavigationController alloc] initWithRootViewController:vc];
// 开启
[navi zj_enableFullScreenPop:YES];
qrScanner.gif
ZJQRScannerView *scanner = [ZJQRScannerView new];
scanner.frame = self.view.bounds;
[self.view addSubview:scanner];
// 开始扫描
[scanner startScanning];
// 扫描完成
[scanner setScannerFinishHandler:^(ZJQRScannerView *scanner, NSString *resultString) {
// 扫描结束
NSLog(@"内容是%@", resultString);
}];
tagView.gif
// 初始化第一个section数据
for (int i=0; i<20; i++) {
ZJTagItem *item = [ZJTagItem new];
item.name = [NSString stringWithFormat:@"选中--- %d",i];
[selectedItems addObject:item];
}
// 初始化第二个section数据
for (int i=0; i<40; i++) {
ZJTagItem *item = [ZJTagItem new];
item.name = [NSString stringWithFormat:@"未选中--- %d",i];
[unselectedItems addObject:item];
}
// 初始化
_tagView = [[ZJTagView alloc] initWithSelectedItems:selectedItems unselectedItems:unselectedItems];
lockView.gif
- (IBAction)deleteBtnOnClick:(id)sender {
if (![ZJLockViewController isAllreadySetPassword]) {
NSLog(@"未曾设置过密码或者密码已经被删除");
return;
}
// 初始化
ZJLockViewController *lock = [[ZJLockViewController alloc] initWithOperationType:ZJLockOperationTypeRemove delegate:self];
// 宽度
lock.lockView.pwdBtnSlideLength = 64.f;
// 线宽
lock.lockView.lineWidth = 8;
// 设置不同状态的图片
[lock.lockView setBtnImage:[UIImage imageNamed:@"normal"] forState:ZJLockButtonStateNormal];
[lock.lockView setBtnImage:[UIImage imageNamed:@"selected"] forState:ZJLockButtonStateSelected];
[lock.lockView setBtnImage:[UIImage imageNamed:@"error"] forState:ZJLockButtonStateError];
[self presentViewController:lock animated:YES completion:nil];
}
scrollPageView.gif
override func viewDidLoad() {
super.viewDidLoad()
// 这个是必要的设置
automaticallyAdjustsScrollViewInsets = false
var style = SegmentStyle()
// 缩放文字
style.scaleTitle = true
// 颜色渐变
style.gradualChangeTitleColor = true
// segment可以滚动
style.scrollTitle = true
style.showExtraButton = true
let childVcs = setChildVcs()
let titles = childVcs.map { $0.title! }
let scrollPageView = ScrollPageView(frame: CGRect(x: 0, y: 64, width: view.bounds.size.width, height: view.bounds.size.height - 64), segmentStyle: style, titles: titles, childVcs: childVcs, parentViewController: self)
view.addSubview(scrollPageView)
}
refreshView.gif
let normalFooter = NormalAnimator.normalAnimator()
normalFooter.lastRefreshTimeKey = "exampleFooter1"
tableView.zj_addRefreshHeader(header, refreshHandler: {[weak self] in
/// 多线程中不要使用 [unowned self]
/// 注意这里的gcd是为了模拟网络加载的过程, 在实际的使用中, 不需要这段gcd代码, 直接在这里进行网络请求, 在请求完毕后, 调用分类方法, 结束刷新
dispatch_async(dispatch_get_global_queue(0, 0), {
for i in 0...50000 {
if i <= 10 {
self?.data.append(i)
}
/// 延时
print("加载数据中")
}
dispatch_async(dispatch_get_main_queue(), {
self?.tableView.reloadData()
/// 刷新完毕, 停止动画
self?.tableView.zj_stopHeaderAnimation()
})
})
})
当你看到这段话的时候, 我心存感激, 感谢您购买正版.这本书是我在电子科技大学就读本科(计算机)的时候完成的. 在开始写作前, 已经做了两年的iOS外包项目开发. 写这本书占据了我大部分的课余时间, 断断续续四个多月. 希望, 它能不辜负你我, 不浪费你我的时间.
一般大家都是习惯了首先找github上面有没有已经开源的效果可以直接使用, 不过不幸的是, 可能我们搜索的时候关键字都不清楚是什么,可能搜索很久找不到我们想要的. 如果你比较幸运的话, 也许能够找到一些开源控件. 那么如果你找不到自己想要的, 或者已经找到的效果不合适, 需要自己去改动源码, 那么这个过程就很艰辛了. 最终可能就不得不自己动手来实现了.
希望, 这本书籍能够让你有所收获. 试读章节一
同时需要说明的:
书中的demo就是上面提到的, 有很简单的, 也有简单的, 但是没有有难度的. 所以源码我已经全部放在了github上了。网上也能找到这些效果,同时也会有人写类似的实现思路,对很多朋友来说完全不用看这本书籍。所以只是希望给需要学习这些东西的人整理了一系列连贯的实现过程。我原本也是打算直接放在网上给大家阅读,不过免费的东西一般不是很受重视,很多朋友只是浏览罢了,所以成书只是个形式,只是希望阅读的人会重视一些.
希望有意购买的朋友, 一定要仔细了解书中的内容, 然后判断是否需要阅读, 虽然书籍不贵, 但是不想坑人.目前书籍《iOS自定义控件剖析》已上线。如果有意购买可以在这个网站上。不过请务必确定您已经阅读过之前的书籍内容说明,和试读章节。毕竟定价35偏高,以免对你造成不必要的损失,谢谢!
如果你购买了书籍, 希望联系到我, 可以通过简书私信给我, 或者联系QQ:854136959 但是请务必备注购买书籍时的账号, 否则可能会被忽略(因为被打扰得很烦了)
如果只是想看看并不打算学习,那么请直接在其他渠道获取免费的。