(IOS)从0到Double系列 - 如何刻出一个可拖动的导航浮動按钮
来源:互联网 发布:cf真正刷枪软件 编辑:程序博客网 时间:2024/05/17 22:07
本文转自简书:http://www.jianshu.com/p/7c8002b60a2e
本篇教程使用Swift 源碼:https://github.com/jamesdouble/JDJellyButton
,以下稱JDJellyButton
0)何謂導航浮動按鈕
当你的应用开发到一定程度的规模时,必须要有个十分清晰明了的导航功能,才不会让使用者卡在某一页,不知道如何前往他们想去的页面。
常见的导航方式,不外乎最常用UITabBarController、UINavigationBar,另外有一种虽然常见,但是因为不是IOS原生就有的UIControl,所以还是比较少人使用,那就是 "floating navigation button"。
之所以会有'Floating'这个字眼,是大多这样的导航按钮会凌驾在所有视图控制器(UI...ViewController)上,不管底下的视图控制器如何切页他都会保持在同样的位置。
0.1)JDJellyButton特色:按鈕群組
源碼其中一個特色就是浮動按鈕附有群組的功能,能讓一個浮動按鈕能包含更多的子按鈕以處理更多不同的事件。
0.2)UIView or UIButton?
大部分的按钮控件虽然都是‘按钮’,但是比起继承实作UIButton,还不如继承实作他的父类别UIView, 可做的事比较多,限制也比较少,本文的JDJellyButton继承自UIView。
0.3)Gesturer or UIResponder
因为我们是自己实作继承UIView的类别,比起每个按钮都要加上手势,我比较偏好在类别下实作几个常见的UIResponder方法 - touchesBegan, touchesMoved。一来省去还要宣告selector这样拐个弯的做法。
1)代碼架構&解析
以下是JDJelllyButton的元件,我將由底層的子元件往上講解。
var MainButton:JDJellyMainButton!var Container:JelllyContainer!var RootView:UIView?var delegate:JellyButtonDelegate?var _datasource:JDJellyButtonDataSource?var jellybutton:JDJellyButtonView?
1.1)ButtonGroups
紀錄了多個JDJellyButtonView跟它們個別的位置,此為“一組”Button
struct ButtonGroups { var buttongroup:[JDJellyButtonView]! var groupPositionDiff:[CGPoint]?}
1.2)JDJellyButtonView:UIView
此一类别是实作每个按钮的基础样式与点击,一个圆配一张图片。
别忘了要处理点击的事件。我做的方法是通知委任(上层接口JDJellyButton)被点击的是第几的Group的第几个Button。
protocol JellyButtonDelegate { func JellyButtonHasBeenTap(touch:UITouch,image:UIImage,groupindex:Int,arrindex:Int)}
override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) { let image = self.imgView?.image let groupindex = dependingMainButton?.getGroupIndex() let arrindex = dependingMainButton?.getJellyButtonIndex(jelly: self) print("\(groupindex),\(arrindex)") tapdelegate?.JellyButtonHasBeenTap(touch: touches.first!,image: image!,groupindex: groupindex!,arrindex: arrindex!) }
1.3)JDJellyMainButton:JDJellyButtonView
本控件最主要的类别,也是整个导航浮动按钮的主体。样式跟其他的按钮一样,差别是在点击后的事件以及它可以拖动,所以就直接继承
JDJellyButtonView并且覆写touchesBegan, touchesMoved,并且也由它来管理ButtonGroups。
func appendButtonGroup(bgs:ButtonGroups) { var temp_bgs:ButtonGroups = bgs for jelly in temp_bgs.buttongroup { //让每个按钮知道自己依附的是谁 //因為只有MainButton知道子Button位在第幾個Group jelly.dependingMainButton = self } temp_bgs.groupPositionDiff = [CGPoint]() for i in 0..<bgs.buttongroup.count { //计算位置 let cgpoint:CGPoint = CGPoint(x: x[i] , y: y[i]) temp_bgs.groupPositionDiff?.append(cgpoint) } buttongroups.append(temp_bgs) }
需要注意的是因为JDJellyButton有分群组,而触发的条件是“长按”,因此我们不再touchesBegan做立即展开,而是在touchesEnded处理。
override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) { /* 略 */ if(Expanding) { expandignMove = true closingButtonGroup(expandagain: false) } //紀錄點下去的時間 LastTime = touches.first!.timestamp /* 略 */ }override func touchesEnded(_ touches: Set<UITouch>, with event: UIEvent?) { /* 略 */ //短按 if(touches.first!.timestamp - LastTime! < 0.15){ if(!Expanding) { expandButtonGroup() } else { closingButtonGroup(expandagain: false) } } else //長按 { if(!Moving) { switchButtonGroup() } if(expandignMove && Moving) { expandButtonGroup() } } Moving = false expandignMove = false /* 略 */ }
1.4)JelllyContainer:UIView
本来并没有打算制作这个类别,后来遇到了一个非常严重的问题:虽然按钮以外透明的地方看似可点击后方的其他View,但是其实会点到浮动导航按钮的整个背景,进而无法触发后方使用者原本的东西。上网爬了之后,发现需覆写point这个Function
override func point(inside point: CGPoint, with event: UIEvent?) -> Bool { for subview in subviews { if !subview.isHidden && subview.alpha > 0 && subview.isUserInteractionEnabled && subview.point(inside: convert(point, to: subview), with: event) { return true } } return false }
- (IOS)从0到Double系列 - 如何刻出一个可拖动的导航浮動按钮
- iOS 悬浮可拖动可点击按钮
- 【Demo】iOS可吸附拖动的悬浮窗按钮插件
- 可拖动的悬浮按钮
- 仿ios touch按钮创建可随意拖动的悬浮按钮
- 可任意拖动的导航条
- iOS自定义悬浮按钮,Objective-C可拖动的浮动按钮,iOS浮动图标,类似AssistiveTouch的浮动漂浮按钮
- 一个可拖动的ScrollView
- 一个可拖动的View
- 实现可拖动的按钮视图
- Android 可拖动的悬浮按钮
- APP内可拖动的悬浮按钮
- 如何从网页代码中看出一个元素到浏览器顶端的距离?
- android 可拖动按钮
- 可拖动按钮
- Android下图片或按钮等可拖动到任意位置的效果实现源码
- iOS:如何修改导航栏按钮或者字体的颜色
- java 程序里如何实现从一个列表中拖动一个元素到另一个列表中?
- table-layout写成的表格
- 编译FFmpeg是configure文件帮助选项详解
- 深入理解Java:注解(Annotation)--注解处理器
- 学会使用QT的帮助文档
- js小功能
- (IOS)从0到Double系列 - 如何刻出一个可拖动的导航浮動按钮
- Android Studio 2.2.2 鼠标悬停显示注释 提示函数用法
- BarTender批量打印标签的教程
- 理解一致性哈希算法(consistent hashing)
- Linux 3.x 内核学习笔记——x86 64位内存管理
- 使用git管理项目
- html5的localStorage的操作
- OC - WebView使用 之 获取 WebView 实际内容高度 及 其中一种导致WebView展示不全的bug
- 自定义log4j日志文件存放路径