(IOS)從0到Double系列 - 如何刻出一個可拖動的導航浮動按鈕

本篇教程使用Swift 源碼:https://github.com/jamesdouble/JDJellyButton
,以下稱JDJellyButton

0)何謂導航浮動按鈕


當你的應用開發(fā)到一定程度的規(guī)模時,必須要有個十分清晰明了的導航功能,才不會讓使用者卡在某一頁弯蚜,不知道如何前往他們想去的頁面。
常見的導航方式墩剖,不外乎最常用UITabBarController咐低、UINavigationBar杖们,另外有一種雖然常見悉抵,但是因為不是IOS原生就有的UIControl,所以還是比較少人使用摘完,那就是 "floating navigation button"姥饰。
之所以會有'Floating'這個字眼,是大多這樣的導航按鈕會凌駕在所有視圖控制器(UI...ViewController)上孝治,不管底下的視圖控制器如何切頁他都會保持在同樣的位置列粪。

浮動導航按鈕

0.1)JDJellyButton特色:按鈕群組

源碼其中一個特色就是浮動按鈕附有群組的功能,能讓一個浮動按鈕能包含更多的子按鈕以處理更多不同的事件谈飒。

jellybutton_delegate.gif

0.2)UIView or UIButton?

大部分的按鈕控件雖然都是‘按鈕’岂座,但是比起繼承實作UIButton,還不如繼承實作他的父類別UIView, 可做的事比較多,限制也比較少杭措,本文的JDJellyButton繼承自UIView费什。

0.3)Gesturer or UIResponder

因為我們是自己實作繼承UIView的類別,比起每個按鈕都要加上手勢手素,我比較偏好在類別下實作幾個常見的UIResponder方法 - touchesBegan, touchesMoved鸳址。一來省去還要宣告selector這樣拐個彎的做法瘩蚪。

1)代碼架構(gòu)&解析


以下是JDJelllyButton的元件,我將由底層的子元件往上講解稿黍。

var MainButton:JDJellyMainButton!
var Container:JelllyContainer!
var RootView:UIView?
var delegate:JellyButtonDelegate?
var _datasource:JDJellyButtonDataSource?
var jellybutton:JDJellyButtonView?
架構(gòu)圖
1.1)ButtonGroups

紀錄了多個JDJellyButtonView跟它們個別的位置疹瘦,此為“一組”Button

struct ButtonGroups {
    var buttongroup:[JDJellyButtonView]!
    var groupPositionDiff:[CGPoint]?
}
1.2)JDJellyButtonView:UIView

此一類別是實作每個按鈕的基礎(chǔ)樣式與點擊,一個圓配一張圖片巡球。
別忘了要處理點擊的事件言沐。我做的方法是通知委任(上層接口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鲸阻。

JDJellyButtonDemo.gif
    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有分群組跋涣,而觸發(fā)的條件是“長按”,因此我們不再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细诸,但是其實會點到浮動導航按鈕的整個背景沛贪,進而無法觸發(fā)后方使用者原本的東西。上網(wǎng)爬了之后震贵,發(fā)現(xiàn)需覆寫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
    }
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市猩系,隨后出現(xiàn)的幾起案子媚送,更是在濱河造成了極大的恐慌,老刑警劉巖寇甸,帶你破解...
    沈念sama閱讀 206,378評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件塘偎,死亡現(xiàn)場離奇詭異,居然都是意外死亡拿霉,警方通過查閱死者的電腦和手機吟秩,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評論 2 382
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來绽淘,“玉大人涵防,你說我怎么就攤上這事』γ” “怎么了武学?”我有些...
    開封第一講書人閱讀 152,702評論 0 342
  • 文/不壞的土叔 我叫張陵祭往,是天一觀的道長。 經(jīng)常有香客問我火窒,道長硼补,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,259評論 1 279
  • 正文 為了忘掉前任熏矿,我火速辦了婚禮已骇,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘票编。我一直安慰自己褪储,他們只是感情好,可當我...
    茶點故事閱讀 64,263評論 5 371
  • 文/花漫 我一把揭開白布慧域。 她就那樣靜靜地躺著鲤竹,像睡著了一般。 火紅的嫁衣襯著肌膚如雪昔榴。 梳的紋絲不亂的頭發(fā)上辛藻,一...
    開封第一講書人閱讀 49,036評論 1 285
  • 那天,我揣著相機與錄音互订,去河邊找鬼吱肌。 笑死,一個胖子當著我的面吹牛仰禽,可吹牛的內(nèi)容都是我干的氮墨。 我是一名探鬼主播,決...
    沈念sama閱讀 38,349評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼吐葵,長吁一口氣:“原來是場噩夢啊……” “哼规揪!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起温峭,我...
    開封第一講書人閱讀 36,979評論 0 259
  • 序言:老撾萬榮一對情侶失蹤粒褒,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后诚镰,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體奕坟,經(jīng)...
    沈念sama閱讀 43,469評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,938評論 2 323
  • 正文 我和宋清朗相戀三年清笨,在試婚紗的時候發(fā)現(xiàn)自己被綠了月杉。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,059評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡抠艾,死狀恐怖苛萎,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤腌歉,帶...
    沈念sama閱讀 33,703評論 4 323
  • 正文 年R本政府宣布蛙酪,位于F島的核電站,受9級特大地震影響翘盖,放射性物質(zhì)發(fā)生泄漏桂塞。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,257評論 3 307
  • 文/蒙蒙 一馍驯、第九天 我趴在偏房一處隱蔽的房頂上張望阁危。 院中可真熱鬧,春花似錦汰瘫、人聲如沸狂打。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,262評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽趴乡。三九已至,卻和暖如春蝗拿,著一層夾襖步出監(jiān)牢的瞬間晾捏,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工蛹磺, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人同仆。 一個月前我還...
    沈念sama閱讀 45,501評論 2 354
  • 正文 我出身青樓萤捆,卻偏偏與公主長得像,于是被迫代替她去往敵國和親俗批。 傳聞我的和親對象是個殘疾皇子俗或,可洞房花燭夜當晚...
    茶點故事閱讀 42,792評論 2 345

推薦閱讀更多精彩內(nèi)容

  • 1,Search Bar 怎樣去掉背景的顏色(storyboard里只能設(shè)置background顏色,可是發(fā)現(xiàn)cl...
    以德扶人閱讀 2,331評論 2 50
  • 發(fā)現(xiàn) 關(guān)注 消息 iOS 第三方庫岁忘、插件辛慰、知名博客總結(jié) 作者大灰狼的小綿羊哥哥關(guān)注 2017.06.26 09:4...
    肇東周閱讀 12,024評論 4 62
  • 真正的雞湯是很有營養(yǎng)的,一篇文章被冠以雞湯二字就是因為太有營養(yǎng)了干像。 因為雞湯文千篇一律的超正能量內(nèi)容和幾乎相同構(gòu)架...
    謝千千閱讀 370評論 0 0
  • 對于時長帅腌,流量資費的用戶,會周期性的根據(jù)記賬消息進行扣費麻汰,每次扣費會記錄一條計費詳單速客。 計費詳單包含了用戶使用的時...
    飄人閱讀 637評論 0 1
  • 四月連日放晴 黃昏 日色漸慢 葉子一樹一樹 綠得整齊 也等誰嗎 我在市郊 街邊長椅 路燈格外殷勤 空明如月色 愔愔...
    深夜狂想曲閱讀 118評論 0 1