原文链接:实现 instagram 底部弹出菜单的一个例子(模拟 instagram 系列)
模拟 instagram 项目源码:github 仓库:模拟 instagram
聊胜于无,最近有点低潮期的感觉,但是得坚持下去。写不动代码了,所以写一篇简单的总结。

目标和成果

instagram 截图如下:

成果如下:

要实现的内容:

  1. 黑色半透明遮罩层

  2. 弹出动画

  3. 点击取消按钮或者背景菜单收回

思路

应该是有一种跟 UIPresentationController 有关的解决方法,但是我还不会,所以这次不用。

之前在模拟微博的项目里也用到了弹出层,所以我想这次也义勇和上次同样的方式,在 TabBarController 里面添加 subview。

但有两点不同:

  1. 上次是点击 tabBarItem 来执行弹出,这次要点击第一个 tab 页里的按钮。怎么样让 TabBarController 得到消息?

  2. 上次弹出的按钮是独立的,这次弹出的菜单和指定的图片有关。所以需要有一个标志记录当前点击的是哪一张图片的 more 按钮。

解决方案 Part.1 创建菜单视图

在 storyboard 拖进一个 ViewController,添加好内容,设置好约束。如图:

注意,这个 ViewController 的 root view 要讲背景设置为 黑色、透明度 30%,以模拟遮罩层。

新建一个视图文件 BottomMenuView.swift,继承 UIViewController。修改刚才新建的 ViewController 类为 BottomMenuView,并设置 StoryboardID 为”BottomMenu”。

关联好 IBOutlet 对象和点击事件。

import Foundation
import UIKit

class BottomMenuView: UIViewController {

    @IBOutlet weak var menuBgView: UIView!
    @IBOutlet weak var reportBtn: UIButton!
    @IBOutlet weak var sharetoFacebookBtn: UIButton!
    @IBOutlet weak var tweetBtn: UIButton!
    @IBOutlet weak var copyURLBtn: UIButton!
    @IBOutlet weak var noticeBtn: UIButton!
    @IBOutlet weak var dismissBtn: UIButton!
    @IBOutlet weak var buttonPanel: UIView!
    
    override func viewDidLoad() {
        super.viewDidLoad()
    }
    
    override func viewDidAppear(animated: Bool) {
        //这里是设置颜色、按钮边框形状。
        let maskPathAllCorner = UIBezierPath(roundedRect: noticeBtn.bounds,byRoundingCorners: [.BottomLeft,.Bottomright,.TopLeft,.TopRight],cornerRadii: CGSize(width: 5,height: 5.0))
        let maskPathTopCorner = UIBezierPath(roundedRect: noticeBtn.bounds,byRoundingCorners: [.TopLeft,height: 5.0))
        let maskPathBottomCorner = UIBezierPath(roundedRect: noticeBtn.bounds,.Bottomright],height: 5.0))
        let shapeAllCorner = CAShapeLayer()
        shapeAllCorner.path = maskPathAllCorner.CGPath
        let shapetopCorner = CAShapeLayer()
        shapetopCorner.path = maskPathTopCorner.CGPath
        let shapeBottomCorner = CAShapeLayer()
        shapeBottomCorner.path = maskPathBottomCorner.CGPath
        
        reportBtn.layer.mask = shapetopCorner
        noticeBtn.layer.mask = shapeBottomCorner
        dismissBtn.layer.mask = shapeAllCorner
    }
    @IBAction func bgTapped(sender: UITapGestureRecognizer) {
        //给背景 view 添加一个点击事件
        //当点击时,让菜单收回
    }
    @IBAction func dismissBtnpressed(sender: UIButton) {
            //点击事件,当点击时,让菜单收回
    }
}

在 TabBarViewController.swift 中,修改 viewDidLoad 方法:

override func viewDidLoad() {
    super.viewDidLoad()
    
    self.tabBar.tintColor = UIColor.whiteColor()
    for it in self.tabBar.items! {
        it.badgeValue = "100"
    }
    
    self.delegate = self
    
    //找到对应视图,并添加为 TabBarView 的子视图。子视图默认 alpha 值为0。
    //这样每次只用修改 alpha 值来显示和隐藏,不用重新添加子视图。
    self.bottomMenu = self.storyboard?.instantiateViewControllerWithIdentifier("BottomMenu") as! BottomMenuView
    self.bottomMenu.modalTransitionStyle = .CoverVertical
    self.view.addSubview(self.bottomMenu.view)
    self.bottomMenu.view.alpha = 0
}

解决方案 Part.2 给 TabBarController 传递消息

因为 more 按钮在 tab 页的 viewController 里面,我们暂时没办法直接执行 TabBarController 里面的函数,或者对它的 view 进行操作,所以要想个办法发送通知。

发送通知当然是用 NSNotification。

发送显示菜单通知

在 firstViewController (即第一个 tab 页的 controller)中添加 more 按钮点击事件,在里面发送一个通知。

@IBAction func moreButtonpressed(sender: UIButton) {
    NSNotificationCenter.defaultCenter().postNotification(NSNotification(name: "ShowBottomMenu",object: sender.tag))
}

在这个通知里面,object 是当前按钮的 tag 值。这是在每个 cell 初始化的时候,给当前 cell 的每个 button 的 tag 赋值为当前图片的 id,保证之后能够判断每个点击事件来自哪个 cell/图片。

发送收回菜单通知

回到 BottomMenuView.swift,在两个点击事件添加发送通知:

@IBAction func bgTapped(sender: UITapGestureRecognizer) {
    NSNotificationCenter.defaultCenter().postNotification(NSNotification(name: "dismissBottomMenu",object: nil))
}
@IBAction func dismissBtnpressed(sender: UIButton) {
    
    NSNotificationCenter.defaultCenter().postNotification(NSNotification(name: "dismissBottomMenu",object: nil))
}

接收通知

在 TabBarViewController.swift 的 viewDidLoad 方法中添加一下两行用于接收通知:

NSNotificationCenter.defaultCenter().addobserver(self,selector: "showBottomMenu:",name: "ShowBottomMenu",object: nil)
NSNotificationCenter.defaultCenter().addobserver(self,selector: "dismissBottomMenu:",name: "dismissBottomMenu",object: nil)

经过上面的步骤,TabBarViewController 就能接收到显示或者收回底部菜单的通知了。然后我们可以在自定义的函数里面编写显示和隐藏的动画。

解决方案 Part.3 显示/收回菜单动画

这部分很简单啦。

func showBottomMenu(notification: NSNotification) {
    //获取当前 section id
    let tag = notification.object as! Int
    print(tag)
    
    self.bottomMenu.buttonPanel.frame.origin.y = self.view.frame.height
    UIView.animateWithDuration(0.3) { () -> Void in
        self.bottomMenu.view.alpha = 1
        self.bottomMenu.buttonPanel.frame.origin.y = self.view.frame.height - self.bottomMenu.buttonPanel.frame.height
    }
}

func dismissBottomMenu(notification: NSNotification) {
    print(notification.object)
    UIView.animateWithDuration(0.3) { () -> Void in
        self.bottomMenu.view.alpha = 0
        self.bottomMenu.buttonPanel.frame.origin.y = self.view.frame.height
    }
}

在 showBottomMenu 方法中,获取到了一个 tag 值,这个是留着将来编写对应的菜单按钮点击事件的。

实现 instagram 底部弹出菜单的一个例子模拟 instagram 系列的更多相关文章

  1. ios – 如何通过编程方式为UIButtons组设置单个背景图像

    解决方法要使用选择器数组以编程方式创建按钮,您可以使用以下代码片段://定义策略

  2. ios – UIButton背景图像以编程方式更改

    如何在点击事件中更改我的UIButton的背景图像?并使用上一张图片在几秒钟内刷新它?我的意思是在点击后更改它的背景图像并在点击后重置它.解决方法将您的clickEvent图像作为按钮的突出显示图像.

  3. ios – UIPopoverController出现在错误的位置

    所以我花了一些时间寻找答案,但到目前为止还没有找到任何答案.我正在尝试从UIInputAccessoryView上的按钮呈现弹出窗口.UIBarButtonItem我想显示popover来自定制视图,所以我可以使用图像.我创建这样的按钮:当需要显示popover时,我这样做:但我得到的是:弹出窗口看起来很好,但它应该出现在第一个按钮上时出现在第二个按钮上.然后我发现了这个问题:UIBarButto

  4. ios – 无法在UIButton上更改自定义UIFont大小

    我有一个按钮,我使用自定义UIFont来显示文本.字体正确加载并正确应用于按钮标题.我的问题是我似乎无法改变字体大小:无论我输入什么字体大小,我都会得到一些默认大小.另一方面,如果我做这样的事情:我得到大小32字体,但是,当然,我没有得到我的自定义字体.那么,如何设置自定义字体的大小?解决方法要获取必须传递fontWithName:size:的字体名称,请使用FontBook应用程序打开字体,然后查找PostScript名称属性.确保您已将其添加到Info.plist文件中.

  5. ios – 创建UIButton(UIButtonTypeCustom),顶部带有图像,底部带有AutoLayout标签

    我试图在它的底部显示带有文本的按钮.顶部的图像;图像&单击时必须突出显示文本.像这样:这是viewDidLoad中的代码:问题是文本根本不显示.请参阅下面的链接图像.如何使用UIEdgeInset为文本腾出空间?我也试过创建一个UIButton类,但效果大致相同.解决方法请使用以下代码设置插图,希望这可以帮助..

  6. ios – Touch拖动输入如何工作?

    任何人都可以给我一个关于触摸拖动输入以从一个按钮拖动到另一个触发两个按钮事件的示例.它是如何工作的?例如,我想从Do到Fa拖动Do,Re,Mi,Fa的事件被触发.这是我的代码:哦,我已经看到,当我按住Simulator时,会触发方法btnDragOut.当我从Simulator拖出按钮时,会触发此按钮的方法.现在我希望当我拖出一个按钮时触发方法btnDragOut.有谁知道吗?

  7. ios – 如何让UIButton成为一个圆圈?

    我一直试图让细胞中的UIButton成为一个完美的圆圈.不幸的是,圆形是基于背景图像而不是UIButton框架形成的.我创建圆圈的代码:输出如下:我该怎么做才能获得我想要的完美圆形按钮框架?解决方法尝试这样的事情如果需要创建一个视图圆,则必须将masksToBounds设置为true,不要设置clipsToBounds希望这会帮助你.

  8. Instagram iOS挂钩与非方形图像

    hl=en).然而,它仍然似乎是将我的图像裁剪成方形,而不是让我选择将它们扩展到非方形尺寸.任何人都有运气发送非方形图像?我希望有一些调整可以使它工作.解决方法我也希望他们会在更新时拍摄非方形照片,但你仍然坚持使用旧的解决方案发布非方形照片……

  9. ios – UIButton在我不希望它时是透明的

    我有一个按钮,我放在自定义UICollectionViewCell的顶角.出于某种原因,它显得半透明,以便当我不喜欢这种情况时,通过按钮显示单元格的边框.在我的CustomCollectionviewCell中:这导致以下结果:这些图像都不是半透明的.他们有扎实的背景,所以我很困惑为什么他们在应用程序中显示为半透明.我原本以为他们可能被放在了单元格的后面,所以我在setupCellLayout()

  10. iOS – 将图像发送到Instagram – DocumentInteraction

    是否可以绕过行动共享表将照片分享给Instagram?问题是,这将提供一个“操作表”,我想避免这样做,如果可能的话,我想使用instagram.ige并跳过此ActionSheet.那可能吗?

随机推荐

  1. Swift UITextField,UITextView,UISegmentedControl,UISwitch

    下面我们通过一个demo来简单的实现下这些控件的功能.首先,我们拖将这几个控件拖到storyboard,并关联上相应的属性和动作.如图:关联上属性和动作后,看看实现的代码:

  2. swift UISlider,UIStepper

    我们用两个label来显示slider和stepper的值.再用张图片来显示改变stepper值的效果.首先,这三个控件需要全局变量声明如下然后,我们对所有的控件做个简单的布局:最后,当slider的值改变时,我们用一个label来显示值的变化,同样,用另一个label来显示stepper值的变化,并改变图片的大小:实现效果如下:

  3. preferredFontForTextStyle字体设置之更改

    即:

  4. Swift没有异常处理,遇到功能性错误怎么办?

    本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

  5. 字典实战和UIKit初探

    ios中数组和字典的应用Applicationschedule类别子项类别名称优先级数据包contactsentertainment接触UIKit学习用Swift调用CocoaTouchimportUIKitletcolors=[]varbackView=UIView(frame:CGRectMake(0.0,0.0,320.0,CGFloat(colors.count*50)))backView

  6. swift语言IOS8开发战记21 Core Data2

    上一话中我们简单地介绍了一些coredata的基本知识,这一话我们通过编程来实现coredata的使用。还记得我们在coredata中定义的那个Model么,上面这段代码会加载这个Model。定义完方法之后,我们对coredata的准备都已经完成了。最后强调一点,coredata并不是数据库,它只是一个框架,协助我们进行数据库操作,它并不关心我们把数据存到哪里。

  7. swift语言IOS8开发战记22 Core Data3

    上一话我们定义了与coredata有关的变量和方法,做足了准备工作,这一话我们来试试能不能成功。首先打开上一话中生成的Info类,在其中引用头文件的地方添加一个@objc,不然后面会报错,我也不知道为什么。

  8. swift实战小程序1天气预报

    在有一定swift基础的情况下,让我们来做一些小程序练练手,今天来试试做一个简单地天气预报。然后在btnpressed方法中依旧增加loadWeather方法.在loadWeather方法中加上信息的显示语句:运行一下看看效果,如图:虽然显示出来了,但是我们的text是可编辑状态的,在storyboard中勾选Editable,再次运行:大功告成,而且现在每次单击按钮,就会重新请求天气情况,大家也来试试吧。

  9. 【iOS学习01】swift ? and !  的学习

    如果不初始化就会报错。

  10. swift语言IOS8开发战记23 Core Data4

    接着我们需要把我们的Rest类变成一个被coredata管理的类,点开Rest类,作如下修改:关键字@NSManaged的作用是与实体中对应的属性通信,BinaryData对应的类型是NSData,CoreData没有布尔属性,只能用0和1来区分。进行如下操作,输入类名:建立好之后因为我们之前写的代码有些地方并不适用于coredata,所以编译器会报错,现在来一一解决。

返回
顶部