Swift实现糗事百科Demo(实战项目)
在这里,你将会学习到解析JSON数据,网络请求功能,动态调整cell内容等功能!!!
最终的结果 是这样的,项目相对简单,很适合入门!下面让我们一起开始教程之旅吧!
1、先看下项目工程结构:
第一步:创建Utitlities文件夹,先完成基础通用的辅助功能
1、网络请求类:HttpRequest.swift
- import Foundation
- ///
- /// @brief 网络请求相关类
- /// @date 2014-10-09
- /// @author huangyibiao
- ///
- class HttpRequest: NSObject {
- override init() {
- super.init()
- }
- ///
- /// @brief 把二进制数据转换成JSON格式的数据
- /// @param data NSDictionary?类型,网络请求返回来的二进制数据
- /// @return 如果解析成功,返回NSDictionary?类型的JSON数据;如果解析失败,会返回nil
- ///
- class func parseJSONData(data: AnyObject?) ->NSDictionary? {
- if let downloadData: NSData = data as? NSData {
- var jsonData: AnyObject? = NSJSONSerialization.JSONObjectWithData(downloadData,
- options: NSJSONReadingOptions.MutableContainers,
- error: nil) as? NSDictionary
- return jsonData as? NSDictionary
- }
- // 当解析失败时,会返回nil
- return nil
- }
- ///
- /// @brief 异步网络请求方法,需要请求地址参数及回调
- /// @param urlString 请求地址
- /// @param completion 请求完成或者请求失败的回调
- /// @return 请求成功时,会返回NSDictionary?的字典格式的数据,如果请求失败,会返回nil
- ///
- class func request(#urlString: String?, completion: (data: NSDictionary?) ->Void) {
- if urlString == nil {
- dispatch_async(dispatch_get_main_queue(), { () -> Void in
- println("urlstring 为空")
- // 请求出现,则返回nil对象表示
- completion(data: nil)
- })
- return
- }
- let url = NSURL.URLWithString(urlString!)
- let request = NSURLRequest(URL: url)
- let queue = NSOperationQueue()
- NSURLConnection.sendAsynchronousRequest(request, queue: queue) {
- (response, data, error) -> Void in
- if error != nil {
- dispatch_async(dispatch_get_main_queue(), { () -> Void in
- println(error)
- // 请求出现,则返回nil对象表示
- completion(data: nil)
- })
- } else { // 请求成功,则返回正确的数据
- let parseData = self.parseJSONData(data)
- dispatch_async(dispatch_get_main_queue(), { () -> Void in
- completion(data: parseData)
- })
- }
- }
- }
- }
2、对String的扩展:
- import Foundation
- import UIKit
- ///
- /// @brief String的通用扩展方法
- /// @date 2014-10-09
- /// @author huangyibiao
- ///
- extension String {
- ///
- /// @brief 获取字符串的高度
- /// @param fontSize 字体大小
- /// @param width 限制一行显示的宽度
- /// @return 返回文本在width宽度的条件下的总高度
- ///
- func height(let fontSize: CGFloat, let width: CGFloat) ->CGFloat {
- let font = UIFont.systemFontOfSize(fontSize)
- let size = CGSizeMake(width, CGFloat.max)
- var style = NSMutableParagraphStyle()
- style.lineBreakMode = NSLineBreakMode.ByCharWrapping
- var attributes = [NSFontAttributeName: font, NSParagraphStyleAttributeName: style.copy()];
- // 强转成NSString
- var text = self as NSString
- var rect = text.boundingRectWithSize(size,
- options: NSStringDrawingOptions.UsesLineFragmentOrigin,
- attributes: attributes,
- context: nil)
- return rect.size.height
- }
- ///
- /// @brief 把时间戳转换成“2014年12月12日 8:20:20”格式的日期字符串
- /// @param timeStamp 时间戳
- /// @return “2014年12月12日 8:20:20”格式的日期字符串
- ///
- func dateStringFromTimeStamp(let timeStamp: NSString) ->String {
- var formatter = NSDateFormatter()
- formatter.dateFormat = "yyyy年MM月dd日 HH:MM:ss"
- let date = NSDate(timeIntervalSince1970: timeStamp.doubleValue)
- return formatter.stringFromDate(date)
- }
- }
3、UIView扩展:
- import Foundation
- import UIKit
- ///
- /// @brief UIView的扩展方法,方便工程全局使用扩展方法来创建或者使用所有继承于UIView的控件
- /// @date 2014-10-09
- /// @author huangyibiao
- ///
- extension UIView {
- ///
- /// 获取或设置origin.x
- ///
- func originX() ->CGFloat {
- return self.frame.origin.x
- }
- func originX(let originX: CGFloat) {
- var rect = self.frame
- rect.origin.x = originX
- self.frame = rect
- }
- ///
- /// 获取或设置origin.y
- ///
- func originY() ->CGFloat {
- return self.frame.origin.y
- }
- func originY(let originY: CGFloat) {
- var rect = self.frame
- rect.origin.y = originY
- self.frame = rect
- }
- ///
- /// 获取或设置origin
- ///
- func origin() ->CGPoint {
- return self.frame.origin
- }
- func origin(let origin: CGPoint) {
- var rect = self.frame
- rect.origin = origin
- self.frame = rect
- }
- ///
- /// 获取或设置width
- ///
- func width() ->CGFloat {
- return self.frame.size.width
- }
- func width(let width: CGFloat) {
- var rect = self.frame
- rect.size.width = width
- self.frame = rect
- }
- ///
- /// 获取或设置height
- ///
- func height() ->CGFloat {
- return self.frame.size.height
- }
- func height(let height: CGFloat) {
- var rect = self.frame
- rect.size.height = height
- self.frame = rect
- }
- ///
- /// 获取rightX
- ///
- func rightX() ->CGFloat {
- return originX() + width()
- }
- ///
- /// 获取或设置bottomY
- ///
- func bottomY() ->CGFloat {
- return originY() + height()
- }
- func bottomY(let bottomY: CGFloat) {
- var rect = self.frame
- rect.origin.y = bottomY - height()
- self.frame = rect
- }
- }
第二步:分析好项目的特性,这几个显示的内容格式很像,因此这里使用了同一个基类,这也需要我们先封装一个基类:
BaseRefreshController
- ///
- /// @brief 由HotController、LatestController、TruthController继承,用于
- /// 统一管理数据显示
- /// @data 2014-10-09
- /// @author huangyibiao
- class BaseRefreshController: UIViewController,
- UITableViewDataSource,
- UITableViewDelegate,
- RefreshViewDelegate,
- JokerCellDelegate {
- var refreshView: RefreshView?
- var dataSource = NSMutableArray()
- var tableView: UITableView?
- var currentPage: Int = 1
- var cellIdentifier = "JokerCellIdentifier"
- override func viewDidLoad() {
- super.viewDidLoad()
- self.automaticallyAdjustsScrollViewInsets = false;
- self.view.backgroundColor = UIColor.whiteColor()
- // table view
- self.tableView = UITableView(frame: CGRectMake(0, 64, self.view.width(), kScreenHeight - 49 - 64))
- self.tableView?.dataSource = self
- self.tableView?.delegate = self
- self.tableView?.separatorStyle = UITableViewCellSeparatorStyle.None
- self.view.addSubview(self.tableView!)
- var nib = UINib(nibName: "JokerCell", bundle: nil)
- self.tableView!.registerNib(nib, forCellReuseIdentifier: cellIdentifier)
- // refresh view
- var array = NSBundle.mainBundle().loadNibNamed("RefreshView", owner: self, options: nil) as Array
- self.refreshView = array[0] as? RefreshView
- self.tableView!.tableFooterView = self.refreshView
- self.refreshView!.delegate = self
- }
- ///
- /// @brief 加载更多数据,此方法由子类调用
- /// @param urlString 请求地址,其中不指定page值
- func downloadData(#urlString: String) {
- let url = "\(urlString)\(self.currentPage)"
- self.refreshView!.startLoadingMore()
- HttpRequest.request(urlString: url) { (data) -> Void in
- if data == nil {
- UIAlertView.show(title: "温馨提示", message: "加载失败!")
- } else {
- var itemArray = data?["items"] as NSArray
- for item: AnyObject in itemArray {
- self.dataSource.addObject(item)
- }
- self.tableView!.reloadData()
- self.refreshView!.stopLoadingMore()
- self.currentPage++
- }
- }
- }
- ///
- /// UITableViewDataSource
- ///
- func numberOfSectionsInTableView(tableView: UITableView) -> Int {
- return 1
- }
- func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
- return self.dataSource.count
- }
- func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
- var cell = tableView.dequeueReusableCellWithIdentifier(self.cellIdentifier, forIndexPath: indexPath) as? JokerCell
- cell?.delegate = self
- if indexPath.row < self.dataSource.count {
- var dataDict = self.dataSource[indexPath.row] as NSDictionary
- cell?.data = dataDict
- }
- return cell!
- }
- ///
- /// UITableViewDelegate
- ///
- func tableView(tableView: UITableView!, heightForRowAtIndexPath indexPath: NSIndexPath!) -> CGFloat {
- var index = indexPath.row
- var data = self.dataSource[index] as NSDictionary
- return JokerCell.cellHeight(data)
- }
- func tableView(tableView: UITableView, didSelectRowAtIndexPath indexPath: NSIndexPath) {
- var index = indexPath.row
- var data = self.dataSource[index] as NSDictionary
- let comment = CommentController()
- comment.jokeId = data["id"] as? String
- self.navigationController?.pushViewController(comment, animated: true)
- }
- ///
- /// JokerCellDelegate 代理方法实现
- ///
- func jokerCell(cell: JokerCell, didClickPicture picutre: String) {
- let browser = PhotoBrowserController()
- browser.bigImageUrlString = picutre
- self.navigationController?.pushViewController(browser, animated: true)
- }
- }
如果使用的cell不一样,那么子类需要重写对应的代理方法,像查看评论这个就需要重写:
CommentController:
- import UIKit
- ///
- /// @brief 查看评价信息
- /// @data 2014-10-09
- /// @author huangyibiao
- class CommentController: BaseRefreshController {
- // 不可空
- var jokeId: String?
- override func viewDidLoad() {
- super.viewDidLoad()
- self.title = "评论"
- var nib = UINib(nibName: "CommentCell", bundle: nil)
- self.tableView!.registerNib(nib, forCellReuseIdentifier: "CommentCellIdentifier")
- self.tableView!.height(kScreenHeight - 64)
- }
- override func viewWillAppear(animated: Bool) {
- super.viewWillAppear(animated)
- if let root = self.tabBarController as? RootTabBarController {
- root.tabBarView?.hidden = true
- }
- if self.jokeId != nil {
- downloadData(urlString: "http://m2.qiushibaike.com/article/\(self.jokeId!)/comments?count=20&page=")
- }
- }
- override func viewWillDisappear(animated: Bool) {
- super.viewWillDisappear(animated)
- if let root = self.tabBarController as? RootTabBarController {
- root.tabBarView?.hidden = false
- }
- }
- ///
- /// RefreshViewDelegate
- func refresh(refreshView: RefreshView, didClickButton button: UIButton) {
- if self.jokeId != nil {
- downloadData(urlString: "http://m2.qiushibaike.com/article/\(self.jokeId!)/comments?count=20&page=")
- }
- }
- ///
- /// UITableViewDataSource
- ///
- override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
- var cell = tableView.dequeueReusableCellWithIdentifier("CommentCellIdentifier", forIndexPath: indexPath) as? CommentCell
- if indexPath.row < self.dataSource.count {
- var dataDict = self.dataSource[indexPath.row] as NSDictionary
- cell?.data = dataDict
- }
- return cell!
- }
- ///
- /// UITableViewDelegate
- ///
- override func tableView(tableView: UITableView!, heightForRowAtIndexPath indexPath: NSIndexPath!) -> CGFloat {
- var index = indexPath.row
- var data = self.dataSource[index] as NSDictionary
- return CommentCell.cellHeight(data)
- }
- override func tableView(tableView: UITableView, didSelectRowAtIndexPath indexPath: NSIndexPath) {
- // 重写而已,以防调用父类的方法
- }
- }
这样封装好以后,下面的最新、热门、真相这三个模块就很简化了:
- import Foundation
- import UIKit
- ///
- /// @brief 热门 模块视图控制器
- /// @author huangyibiao
- class HotController: BaseRefreshController {
- ///
- /// @brief 生命周期函数
- ///
- override func viewDidLoad() {
- super.viewDidLoad()
- self.title = "热门";
- downloadData(urlString: "http://m2.qiushibaike.com/article/list/suggest?count=20&page=")
- }
- ///
- /// RefreshViewDelegate
- func refresh(refreshView: RefreshView, didClickButton button: UIButton) {
- downloadData(urlString: "http://m2.qiushibaike.com/article/list/suggest?count=20&page=")
- }
- }
- import Foundation
- import UIKit
- ///
- /// @brief 最新 模块视图控制器
- /// @author huangyibiao
- class LatestController: BaseRefreshController {
- ///
- /// @brief 生命周期函数
- ///
- override func viewDidLoad() {
- super.viewDidLoad()
- self.title = "最新";
- downloadData(urlString: "http://m2.qiushibaike.com/article/list/latest?count=20&page=")
- }
- ///
- /// RefreshViewDelegate
- func refresh(refreshView: RefreshView, didClickButton button: UIButton) {
- downloadData(urlString: "http://m2.qiushibaike.com/article/list/latest?count=20&page=")
- }
- }
- import Foundation
- import UIKit
- ///
- /// @brief 真相 模块视图控制器
- /// @author huangyibiao
- class TruthController: BaseRefreshController {
- ///
- /// @brief 生命周期函数
- ///
- override func viewDidLoad() {
- super.viewDidLoad()
- self.title = "真相";
- downloadData(urlString: "http://m2.qiushibaike.com/article/list/imgrank?count=20&page=")
- }
- ///
- /// RefreshViewDelegate
- func refresh(refreshView: RefreshView, didClickButton button: UIButton) {
- downloadData(urlString: "http://m2.qiushibaike.com/article/list/imgrank?count=20&page=")
- }
- }
那么剩下定制cell及动态调整内容就让大家自己去查看源码了!!!!!
想要源码吗?猛击这里
版权声明:本文为博主原创文章,未经博主允许不得转载。
- 上一篇处理文本中的超链接
- 下一篇iOS中用到的3DES密码算法
Swift实现糗事百科Demo(实战项目)的更多相关文章
- 8.Python爬虫实战一之爬取糗事百科段子
大家好,前面入门已经说了那么多基础知识了,下面我们做几个实战项目来挑战一下吧.那么这次为大家带来,Python爬取糗事百科的小段子的例子. 首先,糗事百科大家都听说过吧?糗友们发的搞笑的段子一抓一大把 ...
- Python爬虫实战一之爬取糗事百科段子
大家好,前面入门已经说了那么多基础知识了,下面我们做几个实战项目来挑战一下吧.那么这次为大家带来,Python爬取糗事百科的小段子的例子. 首先,糗事百科大家都听说过吧?糗友们发的搞笑的段子一抓一大把 ...
- Python爬虫实战之爬取糗事百科段子
首先,糗事百科大家都听说过吧?糗友们发的搞笑的段子一抓一大把,这次我们尝试一下用爬虫把他们抓取下来. 友情提示 糗事百科在前一段时间进行了改版,导致之前的代码没法用了,会导致无法输出和CPU占用过高的 ...
- Python爬虫实战之爬取糗事百科段子【华为云技术分享】
首先,糗事百科大家都听说过吧?糗友们发的搞笑的段子一抓一大把,这次我们尝试一下用爬虫把他们抓取下来. 友情提示 糗事百科在前一段时间进行了改版,导致之前的代码没法用了,会导致无法输出和CPU占用过高的 ...
- python scrapy实战糗事百科保存到json文件里
编写qsbk_spider.py爬虫文件 # -*- coding: utf-8 -*- import scrapy from qsbk.items import QsbkItem from scra ...
- 转 Python爬虫实战一之爬取糗事百科段子
静觅 » Python爬虫实战一之爬取糗事百科段子 首先,糗事百科大家都听说过吧?糗友们发的搞笑的段子一抓一大把,这次我们尝试一下用爬虫把他们抓取下来. 友情提示 糗事百科在前一段时间进行了改版,导致 ...
- 芝麻HTTP:Python爬虫实战之爬取糗事百科段子
首先,糗事百科大家都听说过吧?糗友们发的搞笑的段子一抓一大把,这次我们尝试一下用爬虫把他们抓取下来. 友情提示 糗事百科在前一段时间进行了改版,导致之前的代码没法用了,会导致无法输出和CPU占用过高的 ...
- 【Python爬虫实战】多线程爬虫---糗事百科段子爬取
多线程爬虫:即程序中的某些程序段并行执行,合理地设置多线程,可以让爬虫效率更高糗事百科段子普通爬虫和多线程爬虫分析该网址链接得出:https://www.qiushibaike.com/8hr/pag ...
- Python爬虫实战(一):爬糗事百科段子
代码: # _*_ coding:utf-8 _*_ import urllib2 import re from datetime import datetime class QSBK: def __ ...
随机推荐
- Reverse Integer - Palindrome Number - 简单模拟
第一个题目是将整数进行反转,这个题实现反转并不难,主要关键点在于如何进行溢出判断.溢出判断再上一篇字符串转整数中已有介绍,本题采用其中的第三种方法,将数字转为字符串,使用字符串比较大小的方法进行比较. ...
- js 完成单继承
//1.使用prototype完成单继承. //定义一个A类 function A(){ } //为A类动态调用属性color,与方法sayColor A.prototype.color = &quo ...
- XML 学习之保存节点
XmlDocumnet WriteTo 方法 说明: 保存xmldocument对象节点到xmlwriter(即xml文件写入流) 实例: public static void WriteXml( X ...
- System.Web.Security 在winform中是什么命名空间呢
des.Key = ASCIIEncoding.ASCII.GetBytes(System.Web.Security.FormsAuthentication.HashPasswordForStorin ...
- PHPExcel 生成图形报表
db.php为数据库操作类, $config为数据库配置,PHPExcel版本为PHPExcel_1.8.0, PHP代码: $dir = dirname(__FILE__); require $d ...
- OpenSSL命令---req
用途: 本指令用来创建和处理PKCS#10格式的证书.它还能够建立自签名证书,做Root CA. 用法: openssl req [-inform PEM|DER] [-outform PEM|DER ...
- 20150706 js之定时器
对应智能社:09 定时器的使用 开启定时器: setInterval(xxx(),1000);//间隔型 第一个参数为函数,第二个为时间,单位为毫秒 setTimeout(xxx(),1000);// ...
- POJ1054 枚举【STL__binary_search()_的应用】
①使用binary_search前要先保证有序 ②binary_search函数仅返回true或false ③binary_search(first element, laste lment + 1, ...
- Clojure学习:表达式与函数
Clojure是一门Lisp方言——确切地说,是一门JVM上的Lisp方言——也是一门非纯粹的函数式语言. Clojure理所当然地秉承了Lisp“代码即数据( code is data! )”的设计 ...
- c# 数据库编程(通过SqlCommand 执行DML语句)
原来一直是java,python等语言,最近用c#语言,并编写数据库访问代码.使用了之后,这里总结下,分享下c#如何操作数据库. 在java等其它语言中,有一套标准的api来完成数据库访问,并且一般都 ...