IOS APP 侧边栏滑动

作者 : 二月长河

import UIKit

// 此 View Controller 为根容器,本身并不包含任何 UI 元素

class ViewController: UIViewController {

// 该 TabBar Controller 不是传统意义上的容器,在此只负责提供 UITabBar 这个 UI 组件
var mainTabBarController: MainTabBarController!

// 主界面点击手势,用于在菜单划出状态下点击主页后自动关闭菜单
var tapGesture: UITapGestureRecognizer!

// 首页的 Navigation Bar 的提供者,是首页的容器
var homeNavigationController: UINavigationController!
// 首页中间的主要视图的来源
var homeViewController: HomeViewController!
// 侧滑菜单视图的来源
var leftViewController: LeftViewController!

// 构造主视图,实现 UINavigationController.view 和 HomeViewController.view 一起缩放
var mainView: UIView!

// 侧滑所需参数
var distance: CGFloat = 0
let FullDistance: CGFloat = 0.78
let Proportion: CGFloat = 0.77
var centerOfLeftViewAtBeginning: CGPoint!
var proportionOfLeftView: CGFloat = 1
var distanceOfLeftView: CGFloat = 50

// 侧滑菜单黑色半透明遮罩层
var blackCover: UIView!

override func viewDidLoad() {
    super.viewDidLoad()

    // 给根容器设置背景
    let imageView = UIImageView(image: UIImage(named: "back"))
    imageView.frame = UIScreen.mainScreen().bounds
    self.view.addSubview(imageView)

    // 通过 StoryBoard 取出左侧侧滑菜单视图
    leftViewController = UIStoryboard(name: "Main", bundle: nil).instantiateViewControllerWithIdentifier("LeftViewController") as! LeftViewController
    // 适配 4.7 和 5.5 寸屏幕的缩放操作,有偶发性小 bug
    if Common.screenWidth > 320 {
        proportionOfLeftView = Common.screenWidth / 320
        distanceOfLeftView += (Common.screenWidth - 320) * FullDistance / 2
    }
    leftViewController.view.center = CGPointMake(leftViewController.view.center.x - 50, leftViewController.view.center.y)
    leftViewController.view.transform = CGAffineTransformScale(CGAffineTransformIdentity, 0.8, 0.8)

    // 动画参数初始化
    centerOfLeftViewAtBeginning = leftViewController.view.center
    // 把侧滑菜单视图加入根容器
    self.view.addSubview(leftViewController.view)

    // 在侧滑菜单之上增加黑色遮罩层,目的是实现视差特效
    blackCover = UIView(frame: CGRectOffset(self.view.frame, 0, 0))
    blackCover.backgroundColor = UIColor.blackColor()
    self.view.addSubview(blackCover)

    // 初始化主视图,即包含 TabBar、NavigationBar和首页的总视图
    mainView = UIView(frame: self.view.frame)
    // 初始化 TabBar
    let nibContents = NSBundle.mainBundle().loadNibNamed("MainTabBarController", owner: nil, options: nil)
    mainTabBarController = nibContents.first as! MainTabBarController
    // 取出 TabBar Controller 的视图加入主视图
    let tabBarView = mainTabBarController.view
    mainView.addSubview(tabBarView)
    // 从 StoryBoard 取出首页的 Navigation Controller
    homeNavigationController = UIStoryboard(name: "Main", bundle: nil).instantiateViewControllerWithIdentifier("HomeNavigationController") as! UINavigationController
    // 从 StoryBoard 初始化而来的 Navigation Controller 会自动初始化他的 Root View Controller,即 HomeViewController
    // 我们将其(指针)取出,赋给容器 View Controller 的成员变量 homeViewController
    homeViewController = homeNavigationController.viewControllers.first as! HomeViewController
    // 分别将 Navigation Bar 和 homeViewController 的视图加入 TabBar Controller 的视图
    tabBarView.addSubview(homeViewController.navigationController!.view)
    tabBarView.addSubview(homeViewController.view)

    // 在 TabBar Controller 的视图中,将 TabBar 视图提到最表层
    tabBarView.bringSubviewToFront(mainTabBarController.tabBar)

    // 将主视图加入容器
    self.view.addSubview(mainView)

    // 分别指定 Navigation Bar 左右两侧按钮的事件
    homeViewController.navigationItem.leftBarButtonItem?.action = Selector("showLeft")
    homeViewController.navigationItem.rightBarButtonItem?.action = Selector("showRight")

    // 给主视图绑定 UIPanGestureRecognizer
    let panGesture = homeViewController.panGesture
    panGesture.addTarget(self, action: Selector("pan:"))
    mainView.addGestureRecognizer(panGesture)

    // 生成单击收起菜单手势
    tapGesture = UITapGestureRecognizer(target: self, action: "showHome")
}

override func didReceiveMemoryWarning() {
    super.didReceiveMemoryWarning()
    // Dispose of any resources that can be recreated.
}

// 响应 UIPanGestureRecognizer 事件
func pan(recongnizer: UIPanGestureRecognizer) {
    let x = recongnizer.translationInView(self.view).x
    let trueDistance = distance + x // 实时距离
    let trueProportion = trueDistance / (Common.screenWidth*FullDistance)

    // 如果 UIPanGestureRecognizer 结束,则激活自动停靠
    if recongnizer.state == UIGestureRecognizerState.Ended {

        if trueDistance > Common.screenWidth * (Proportion / 3) {
            showLeft()
        } else if trueDistance < Common.screenWidth * -(Proportion / 3) {
            showRight()
        } else {
            showHome()
        }

        return
    }

    // 计算缩放比例
    var proportion: CGFloat = recongnizer.view!.frame.origin.x >= 0 ? -1 : 1
    proportion *= trueDistance / Common.screenWidth
    proportion *= 1 - Proportion
    proportion /= FullDistance + Proportion/2 - 0.5
    proportion += 1
    if proportion <= Proportion { // 若比例已经达到最小,则不再继续动画
        return
    }
    // 执行视差特效
    blackCover.alpha = (proportion - Proportion) / (1 - Proportion)
    // 执行平移和缩放动画
    recongnizer.view!.center = CGPointMake(self.view.center.x + trueDistance, self.view.center.y)
    recongnizer.view!.transform = CGAffineTransformScale(CGAffineTransformIdentity, proportion, proportion)

    // 执行左视图动画
    let pro = 0.8 + (proportionOfLeftView - 0.8) * trueProportion
    leftViewController.view.center = CGPointMake(centerOfLeftViewAtBeginning.x + distanceOfLeftView * trueProportion, centerOfLeftViewAtBeginning.y - (proportionOfLeftView - 1) * leftViewController.view.frame.height * trueProportion / 2 )
    leftViewController.view.transform = CGAffineTransformScale(CGAffineTransformIdentity, pro, pro)
}

// 封装三个方法,便于后期调用

// 展示左视图
func showLeft() {
    // 给首页 加入 点击自动关闭侧滑菜单功能
    mainView.addGestureRecognizer(tapGesture)
    // 计算距离,执行菜单自动滑动动画
    distance = self.view.center.x * (FullDistance*2 + Proportion - 1)
    doTheAnimate(self.Proportion, showWhat: "left")
    homeNavigationController.popToRootViewControllerAnimated(true)
}
// 展示主视图
func showHome() {
    // 从首页 删除 点击自动关闭侧滑菜单功能
    mainView.removeGestureRecognizer(tapGesture)
    // 计算距离,执行菜单自动滑动动画
    distance = 0
    doTheAnimate(1, showWhat: "home")
}
// 展示右视图
func showRight() {
    // 给首页 加入 点击自动关闭侧滑菜单功能
    mainView.addGestureRecognizer(tapGesture)
    // 计算距离,执行菜单自动滑动动画
    distance = self.view.center.x * -(FullDistance*2 + Proportion - 1)
    doTheAnimate(self.Proportion, showWhat: "right")
}
// 执行三种动画:显示左侧菜单、显示主页、显示右侧菜单
func doTheAnimate(proportion: CGFloat, showWhat: String) {
    UIView.animateWithDuration(0.3, delay: 0, options: UIViewAnimationOptions.CurveEaseInOut, animations: { () -> Void in
        // 移动首页中心
        self.mainView.center = CGPointMake(self.view.center.x + self.distance, self.view.center.y)
        // 缩放首页
        self.mainView.transform = CGAffineTransformScale(CGAffineTransformIdentity, proportion, proportion)
        if showWhat == "left" {
            // 移动左侧菜单的中心
            self.leftViewController.view.center = CGPointMake(self.centerOfLeftViewAtBeginning.x + self.distanceOfLeftView, self.leftViewController.view.center.y)
            // 缩放左侧菜单
            self.leftViewController.view.transform = CGAffineTransformScale(CGAffineTransformIdentity, self.proportionOfLeftView, self.proportionOfLeftView)
        }
        // 改变黑色遮罩层的透明度,实现视差效果
        self.blackCover.alpha = showWhat == "home" ? 1 : 0

        // 为了演示效果,在右侧菜单划出时隐藏漏出的左侧菜单,并无实际意义
        self.leftViewController.view.alpha = showWhat == "right" ? 0 : 1
        }, completion: nil)
}
}
import UIKit

// 侧滑菜单 View Controller

class LeftViewController: UIViewController, UITableViewDelegate, UITableViewDataSource {

let titlesDictionary = ["开通会员", "QQ钱包", "网上营业厅", "个性装扮", "我的收藏", "我的相册", "我的文件"]

@IBOutlet weak var settingTableView: UITableView!
@IBOutlet weak var avatarImageView: UIImageView!

@IBOutlet weak var heightLayoutConstraintOfSettingTableView: NSLayoutConstraint!

override func viewDidLoad() {
    super.viewDidLoad()

    settingTableView.delegate = self
    settingTableView.dataSource = self
    settingTableView.tableFooterView = UIView()

    heightLayoutConstraintOfSettingTableView.constant = Common.screenHeight < 500 ? Common.screenHeight * (568 - 221) / 568 : 347
    self.view.frame = CGRectMake(0, 0, 320 * 0.78, Common.screenHeight)
}

override func didReceiveMemoryWarning() {
    super.didReceiveMemoryWarning()
    // Dispose of any resources that can be recreated.
}

// 处理点击事件
func tableView(tableView: UITableView, didSelectRowAtIndexPath indexPath: NSIndexPath) {
    let viewController = Common.rootViewController
    viewController.homeViewController.titleOfOtherPages = titlesDictionary[indexPath.row]
    viewController.homeViewController.performSegueWithIdentifier("showOtherPages", sender: self)
    Common.contactsVC.view.removeFromSuperview()
    viewController.mainTabBarController.tabBar.hidden = true
    viewController.mainTabBarController.selectedIndex = 0
    viewController.showHome()
    tableView.deselectRowAtIndexPath(indexPath, animated: false)
}

func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
    return 7
}
func numberOfSectionsInTableView(tableView: UITableView) -> Int {
    return 1
}

func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
    let cell = tableView.dequeueReusableCellWithIdentifier("leftViewCell", forIndexPath: indexPath) 

    cell.backgroundColor = UIColor.clearColor()
    cell.textLabel!.text = titlesDictionary[indexPath.row]

    return cell
}

/*
// MARK: - Navigation

// In a storyboard-based application, you will often want to do a little preparation before navigation
override func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?) {
    // Get the new view controller using segue.destinationViewController.
    // Pass the selected object to the new view controller.
}
*/
}

results matching ""

    No results matching ""