iOS 給view加一個(gè)拖拽功能

前言

現(xiàn)在的直播app都具有懸浮窗功能,懸浮窗可以拖拽,并且具有回彈動(dòng)畫暮屡,可以設(shè)計(jì)一個(gè)UIView的分類實(shí)現(xiàn)驻龟,減少侵入性温眉。

主要代碼及思路

思路

  • 給view添加一個(gè)pan的手勢,在gesture的狀態(tài)中進(jìn)行判斷(例如邊界的回彈)翁狐,最終拖拽手勢結(jié)束時(shí)类溢,通過block把手勢回調(diào)出去。

代碼

#import "UIView+dragable.h"
#import <objc/runtime.h>

#define ScreenWidth                         [[UIScreen mainScreen] bounds].size.width
#define ScreenHeight                        [[UIScreen mainScreen] bounds].size.height

static const char *ActionHandlerPanGestureKey;

@implementation UIView (dragable)

- (void)addDragableActionWithEnd:(void (^)(CGRect endFrame))endBlock; {
    // 添加拖拽手勢
    UIPanGestureRecognizer *panGestureRecognizer = [[UIPanGestureRecognizer alloc] initWithTarget:self action:@selector(handlePanAction:)];
    [self addGestureRecognizer:panGestureRecognizer];
    
    // 記錄block
    objc_setAssociatedObject(self, ActionHandlerPanGestureKey, endBlock, OBJC_ASSOCIATION_COPY);
}

@end

  • 由于在uiview的分類中露懒,所以使用了runtime關(guān)聯(lián)一下需要回調(diào)的block
- (void)handlePanAction:(UIPanGestureRecognizer *)sender {
    CGPoint point = [sender translationInView:[sender.view superview]];
    
    CGFloat senderHalfViewWidth = sender.view.frame.size.width / 2;
    CGFloat senderHalfViewHeight = sender.view.frame.size.height / 2;
    
    __block CGPoint viewCenter = CGPointMake(sender.view.center.x + point.x, sender.view.center.y + point.y);
    // 拖拽狀態(tài)結(jié)束
    if (sender.state == UIGestureRecognizerStateEnded) {
        [UIView animateWithDuration:0.4 animations:^{
            if ((sender.view.center.x + point.x - senderHalfViewWidth) <= 12) {
                viewCenter.x = senderHalfViewWidth + 12;
            }
            if ((sender.view.center.x + point.x + senderHalfViewWidth) >= (ScreenWidth - 12)) {
                viewCenter.x = ScreenWidth - senderHalfViewWidth - 12;
            }
            if ((sender.view.center.y + point.y - senderHalfViewHeight) <= 12) {
                viewCenter.y = senderHalfViewHeight + 12;
            }
            if ((sender.view.center.y + point.y + senderHalfViewHeight) >= (ScreenHeight - 12)) {
                viewCenter.y = ScreenHeight - senderHalfViewHeight - 12;
            }
            sender.view.center = viewCenter;
        } completion:^(BOOL finished) {
            void (^endBlock)(CGRect endFrame) = objc_getAssociatedObject(self, ActionHandlerPanGestureKey);
            if (endBlock) {
                endBlock(sender.view.frame);
            }
        }];
        [sender setTranslation:CGPointMake(0, 0) inView:[sender.view superview]];
    } else {
        // UIGestureRecognizerStateBegan || UIGestureRecognizerStateChanged
        viewCenter.x = sender.view.center.x + point.x;
        viewCenter.y = sender.view.center.y + point.y;
        sender.view.center = viewCenter;
        [sender setTranslation:CGPointMake(0, 0) inView:[sender.view superview]];
    }
}
  • UIGestureRecognizerState有個(gè)狀態(tài)是UIGestureRecognizerStateEnded闯冷,當(dāng)拖拽結(jié)束時(shí),需要判斷拖拽視圖的位置是否超出屏幕邊界隐锭,如果超出邊界窃躲,需要回彈到邊界內(nèi)。
  • 回彈動(dòng)畫結(jié)束后钦睡,用runtime取出關(guān)聯(lián)的block蒂窒,把最終的視圖的frame回調(diào),供外部使用荞怒。
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末洒琢,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子褐桌,更是在濱河造成了極大的恐慌衰抑,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,755評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件荧嵌,死亡現(xiàn)場離奇詭異呛踊,居然都是意外死亡砾淌,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門谭网,熙熙樓的掌柜王于貴愁眉苦臉地迎上來汪厨,“玉大人,你說我怎么就攤上這事愉择〗俾遥” “怎么了?”我有些...
    開封第一講書人閱讀 165,138評(píng)論 0 355
  • 文/不壞的土叔 我叫張陵锥涕,是天一觀的道長衷戈。 經(jīng)常有香客問我,道長层坠,這世上最難降的妖魔是什么殖妇? 我笑而不...
    開封第一講書人閱讀 58,791評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮窿春,結(jié)果婚禮上拉一,老公的妹妹穿的比我還像新娘。我一直安慰自己旧乞,他們只是感情好蔚润,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,794評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著尺栖,像睡著了一般嫡纠。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上延赌,一...
    開封第一講書人閱讀 51,631評(píng)論 1 305
  • 那天除盏,我揣著相機(jī)與錄音,去河邊找鬼挫以。 笑死者蠕,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的掐松。 我是一名探鬼主播踱侣,決...
    沈念sama閱讀 40,362評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼大磺!你這毒婦竟也來了抡句?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,264評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤杠愧,失蹤者是張志新(化名)和其女友劉穎待榔,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體流济,經(jīng)...
    沈念sama閱讀 45,724評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡锐锣,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,900評(píng)論 3 336
  • 正文 我和宋清朗相戀三年腌闯,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片雕憔。...
    茶點(diǎn)故事閱讀 40,040評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡绑嘹,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出橘茉,到底是詐尸還是另有隱情,我是刑警寧澤姨丈,帶...
    沈念sama閱讀 35,742評(píng)論 5 346
  • 正文 年R本政府宣布畅卓,位于F島的核電站,受9級(jí)特大地震影響蟋恬,放射性物質(zhì)發(fā)生泄漏翁潘。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,364評(píng)論 3 330
  • 文/蒙蒙 一歼争、第九天 我趴在偏房一處隱蔽的房頂上張望拜马。 院中可真熱鬧,春花似錦沐绒、人聲如沸俩莽。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,944評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽扮超。三九已至,卻和暖如春蹋肮,著一層夾襖步出監(jiān)牢的瞬間出刷,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,060評(píng)論 1 270
  • 我被黑心中介騙來泰國打工坯辩, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留馁龟,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,247評(píng)論 3 371
  • 正文 我出身青樓漆魔,卻偏偏與公主長得像坷檩,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子有送,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,979評(píng)論 2 355

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

  • 發(fā)現(xiàn) 關(guān)注 消息 iOS 第三方庫淌喻、插件、知名博客總結(jié) 作者大灰狼的小綿羊哥哥關(guān)注 2017.06.26 09:4...
    肇東周閱讀 12,105評(píng)論 4 62
  • 在假期的最后一天大掃除時(shí),終于把那拆得七零八散的熱水器給丟樓下垃圾桶了匕荸。但是心情還是不舍爹谭。畢竟陪我在這個(gè)城市...
    閑趣的肖小姐閱讀 294評(píng)論 0 0
  • 朋友,今天的你榛搔,過28歲了嗎诺凡?開始懷念青春了嗎? 那時(shí)候践惑,桌子上堆滿了各科的習(xí)題腹泌,床頭貼著“戰(zhàn)勝自我”之類的座右銘...
    我是東北仁兒閱讀 464評(píng)論 0 3
  • 無味的生活總需要一個(gè)有趣的人,原諒我的膽怯尔觉、懦弱凉袱,我不想失去,也未曾奢望得到過
    密秘的密秘閱讀 274評(píng)論 0 0
  • 九月她陪我買支牙膏 與她早已不再相見 二月干癟仍能擠出 一支牙膏足夠五月有余 一生一人行至耄耋 收縮牙床零落黃牙 ...
    任一蓑閱讀 418評(píng)論 0 1