iOS 自定義拖拽式控件:QiDragView

級(jí)別: ★★☆☆☆
標(biāo)簽:「iOS」「拖拽式控件」「QiDragView」
作者: MrLiuQ
審校: QiShare團(tuán)隊(duì)


首先寄纵,我們先看一下QiDragView的效果圖:

一宾濒、QiDragView整體架構(gòu)設(shè)計(jì)

話不多說,上架構(gòu)圖~

QiDragView(QiDragSortView的簡稱)是一種可選擇可拖拽的自定義控件捞慌,可以滿足一些拖拽排序的業(yè)務(wù)需求場景淤堵。

二寝衫、如何自定義使用QiDragView?

在上Demo之前拐邪,先介紹幾個(gè)可以自定義的UI配置屬性:

屬性 類型 介紹
rowHeight CGFloat 行高
rowMargin CGFloat 行邊距
rowPadding CGFloat 行間距
columnMargin CGFloat 列邊距
columnPadding CGFloat 列間距
columnCount NSInteger 列數(shù)
normalColor UIColor 按鈕基本字體顏色
selectedColor UIColor 按鈕選擇字體顏色

以及一些邏輯配置屬性:

屬性 類型 介紹
enabledTitles NSArray<NSString *> 可以被點(diǎn)擊的titles(上行參數(shù)慰毅,默認(rèn)全選)
selectedTitles NSArray<NSString *> 可以被選擇的titles(上行參數(shù),默認(rèn)全選)
titles NSArray<NSString *> 按鈕的titles(上行參數(shù)扎阶,會(huì)根據(jù)titles汹胃,創(chuàng)建出對(duì)應(yīng)的button)

使用起來也很方便:

  • 直接設(shè)置titles即可創(chuàng)建出對(duì)應(yīng)title的Buttons。
  • 通過dragSortEnded的block方法回調(diào)东臀,處理拖拽后的業(yè)務(wù)邏輯:按鈕的排序着饥、按鈕是否選擇等屬性

默認(rèn)配置用法:

QiDragSortView *dragSortView = [[QiDragSortView alloc] initWithFrame:CGRectMake(.0, 100.0, self.view.bounds.size.width, .0)];
dragSortView.backgroundColor = [[UIColor lightGrayColor] colorWithAlphaComponent:.5];

dragSortView.titles = @[@"首頁推薦", @"奇舞周刊", @"眾成翻譯", @"QiShare", @"HULK一線雜談", @"Qtest之道"];//!< 初始的Buttons(必填)
[self.view addSubview:dragSortView];

//! 拖拽方法回調(diào):能拿到Button數(shù)組的排序和選擇狀態(tài)
dragSortView.dragSortEnded = ^(NSArray<UIButton *> * _Nonnull buttons) {
    for (UIButton *button in buttons) {
        NSLog(@"title: %@, selected: %i", button.currentTitle, button.isSelected);
    }
};

自定義配置用法:

QiDragSortView *dragSortView = [[QiDragSortView alloc] initWithFrame:CGRectMake(.0, 100.0, self.view.bounds.size.width, .0)];
dragSortView.backgroundColor = [[UIColor lightGrayColor] colorWithAlphaComponent:.5];
dragSortView.rowHeight = 50.0;
dragSortView.rowMargin = 30.0;
dragSortView.rowPadding = 20.0;
dragSortView.columnCount = 3;
dragSortView.columnMargin = 30.0;
dragSortView.columnPadding = 20.0;
dragSortView.normalColor = [UIColor redColor];
dragSortView.selectedColor = [UIColor purpleColor];
dragSortView.enabledTitles = @[@"奇舞周刊", @"眾成翻譯", @"QiShare", @"HULK一線雜談", @"Qtest之道"];//!< 可以點(diǎn)擊選擇的Buttons(選填,默認(rèn)全選)
dragSortView.selectedTitles = @[@"首頁推薦", @"HULK一線雜談", @"Qtest之道"];//!< 初始選擇的Buttons(選填惰赋,默認(rèn)全選)
dragSortView.titles = @[@"首頁推薦", @"奇舞周刊", @"眾成翻譯", @"QiShare", @"HULK一線雜談", @"Qtest之道"];//!< 初始的Buttons(必填)
[self.view addSubview:dragSortView];

//! 拖拽方法回調(diào):能拿到Button數(shù)組的排序和選擇狀態(tài)
dragSortView.dragSortEnded = ^(NSArray<UIButton *> * _Nonnull buttons) {
    for (UIButton *button in buttons) {
        NSLog(@"title: %@, selected: %i", button.currentTitle, button.isSelected);
    }
};

三宰掉、QiDragView的技術(shù)點(diǎn)

3.1 長按手勢(shì):

長按手勢(shì)分別對(duì)應(yīng)三種狀態(tài):UIGestureRecognizerStateBeganUIGestureRecognizerStateChanged赁濒、UIGestureRecognizerStateEnded贵扰。

狀態(tài) 說明
UIGestureRecognizerStateBegan 長按手勢(shì)開始
UIGestureRecognizerStateChanged 長按手勢(shì)拖拽中(進(jìn)行時(shí))
UIGestureRecognizerStateEnded 長按手勢(shì)結(jié)束
//! 長按手勢(shì)
- (void)longPress:(UILongPressGestureRecognizer *)gesture {
    
    UIButton *currentButton = (UIButton *)gesture.view;
    
    if (gesture.state == UIGestureRecognizerStateBegan) {
        
        [self bringSubviewToFront:currentButton];
        
        [UIView animateWithDuration:.25 animations:^{
            self.originButtonCenter = currentButton.center;
            self.originGesturePoint = [gesture locationInView:currentButton];
            currentButton.transform = CGAffineTransformScale(currentButton.transform, 1.2, 1.2);
        }];
    }
    else if (gesture.state == UIGestureRecognizerStateEnded) {
        
        [UIView animateWithDuration:.25 animations:^{
            currentButton.center = self.originButtonCenter;
            currentButton.transform = CGAffineTransformIdentity;
        } completion:^(BOOL finished) {
            if (self.dragSortEnded) {
                self.dragSortEnded(self.buttons);
            }
        }];
    }
    else if (gesture.state == UIGestureRecognizerStateChanged) {
        
        CGPoint gesturePoint = [gesture locationInView:currentButton];
        CGFloat deltaX = gesturePoint.x - _originGesturePoint.x;
        CGFloat deltaY = gesturePoint.y - _originGesturePoint.y;
        currentButton.center = CGPointMake(currentButton.center.x + deltaX, currentButton.center.y + deltaY);
        
        NSInteger fromIndex = currentButton.tag;
        NSInteger toIndex = [self toIndexWithCurrentButton:currentButton];
        
        if (toIndex >= 0) {
            currentButton.tag = toIndex;
            
            if (toIndex > fromIndex) {
                for (NSInteger i = fromIndex; i < toIndex; i++) {
                    UIButton *nextButton = _buttons[i + 1];
                    CGPoint tempPoint = nextButton.center;
                    [UIView animateWithDuration:.5 animations:^{
                        nextButton.center = self.originButtonCenter;
                    }];
                    _originButtonCenter = tempPoint;
                    nextButton.tag = i;
                }
            }
            else if (toIndex < fromIndex) {
                for (NSInteger i = fromIndex; i > toIndex; i--) {
                    UIButton *previousButton = self.buttons[i - 1];
                    CGPoint tempPoint = previousButton.center;
                    [UIView animateWithDuration:.5 animations:^{
                        previousButton.center = self.originButtonCenter;
                    }];
                    _originButtonCenter = tempPoint;
                    previousButton.tag = i;
                }
            }
            [_buttons sortUsingComparator:^NSComparisonResult(UIButton *obj1, UIButton *obj2) {
                return obj1.tag > obj2.tag;
            }];
        }
    }
}

3.2 配置按鈕:

設(shè)計(jì)思路:在屬性titles的setter方法中,初始化并配置好各個(gè)Buttons流部。

- (void)setTitles:(NSArray<NSString *> *)titles {
    
    _titles = titles;
    
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(.01 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        NSInteger differCount = titles.count - self.buttons.count;
        
        if (differCount > 0) {
            for (NSInteger i = self.buttons.count; i < titles.count; i++) {
                [self.buttons addObject:[self buttonWithTag:i]];
            }
        }
        else if (differCount < 0) {
            NSArray *extraButtons = [self.buttons subarrayWithRange:(NSRange){titles.count, self.buttons.count - titles.count}];
            [self.buttons removeObjectsInArray:extraButtons];
            for (UIButton *button in extraButtons) {
                [button removeFromSuperview];
            }
        }
        
        self.enabledTitles = self.enabledTitles ?: titles;//!< 如果有,就傳入纹坐,否則傳入titles
        self.selectedTitles = self.selectedTitles ?: titles;
        
        for (NSInteger i = 0; i < self.buttons.count; i++) {
            [self.buttons[i] setTitle:titles[i] forState:UIControlStateNormal];
            [self.buttons[i] addGestureRecognizer:[[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(longPress:)]];//!< 長按手勢(shì)
            [self selectButton:self.buttons[i] forStatus:[self.selectedTitles containsObject:titles[i]]];
            if ([self.enabledTitles containsObject:titles[i]]) {
                [self.buttons[i] addTarget:self action:@selector(buttonClicked:) forControlEvents:UIControlEventTouchUpInside];
            }
        }
        
        for (NSInteger i = 0; i < self.buttons.count; i++) {
            NSInteger rowIndex = i / self.columnCount;
            NSInteger columnIndex = i % self.columnCount;
            CGFloat buttonWidth = (self.bounds.size.width - self.columnMargin * 2 - self.columnPadding * (self.columnCount - 1))  / self.columnCount;
            CGFloat buttonX = self.columnMargin + columnIndex * (buttonWidth + self.columnPadding);
            CGFloat buttonY = self.rowMargin + rowIndex * (self.rowHeight + self.rowPadding);
            self.buttons[i].frame = CGRectMake(buttonX, buttonY, buttonWidth, self.rowHeight);
        }
        
        CGRect frame = self.frame;
        NSInteger rowCount = ceilf((CGFloat)self.buttons.count / (CGFloat)self.columnCount);
        frame.size.height = self.rowMargin * 2 + self.rowHeight * rowCount  + self.rowPadding * (rowCount - 1);
        self.frame = frame;
    });
}

源碼地址:QiDragView


推薦文章:
iOS Wireshark抓包
iOS Charles抓包
初探TCP
IP枝冀、UDP初探

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末舞丛,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子果漾,更是在濱河造成了極大的恐慌球切,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,406評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件绒障,死亡現(xiàn)場離奇詭異吨凑,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)户辱,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,732評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門鸵钝,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人庐镐,你說我怎么就攤上這事恩商。” “怎么了必逆?”我有些...
    開封第一講書人閱讀 163,711評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵怠堪,是天一觀的道長。 經(jīng)常有香客問我名眉,道長粟矿,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,380評(píng)論 1 293
  • 正文 為了忘掉前任损拢,我火速辦了婚禮陌粹,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘探橱。我一直安慰自己申屹,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,432評(píng)論 6 392
  • 文/花漫 我一把揭開白布隧膏。 她就那樣靜靜地躺著哗讥,像睡著了一般。 火紅的嫁衣襯著肌膚如雪胞枕。 梳的紋絲不亂的頭發(fā)上杆煞,一...
    開封第一講書人閱讀 51,301評(píng)論 1 301
  • 那天,我揣著相機(jī)與錄音腐泻,去河邊找鬼决乎。 笑死,一個(gè)胖子當(dāng)著我的面吹牛派桩,可吹牛的內(nèi)容都是我干的构诚。 我是一名探鬼主播,決...
    沈念sama閱讀 40,145評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼铆惑,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼范嘱!你這毒婦竟也來了送膳?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,008評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤丑蛤,失蹤者是張志新(化名)和其女友劉穎叠聋,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體受裹,經(jīng)...
    沈念sama閱讀 45,443評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡碌补,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,649評(píng)論 3 334
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了棉饶。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片厦章。...
    茶點(diǎn)故事閱讀 39,795評(píng)論 1 347
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖砰盐,靈堂內(nèi)的尸體忽然破棺而出闷袒,到底是詐尸還是另有隱情,我是刑警寧澤岩梳,帶...
    沈念sama閱讀 35,501評(píng)論 5 345
  • 正文 年R本政府宣布囊骤,位于F島的核電站,受9級(jí)特大地震影響冀值,放射性物質(zhì)發(fā)生泄漏也物。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,119評(píng)論 3 328
  • 文/蒙蒙 一列疗、第九天 我趴在偏房一處隱蔽的房頂上張望滑蚯。 院中可真熱鬧,春花似錦抵栈、人聲如沸告材。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,731評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽斥赋。三九已至,卻和暖如春产艾,著一層夾襖步出監(jiān)牢的瞬間疤剑,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,865評(píng)論 1 269
  • 我被黑心中介騙來泰國打工闷堡, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留隘膘,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,899評(píng)論 2 370
  • 正文 我出身青樓杠览,卻偏偏與公主長得像弯菊,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子踱阿,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,724評(píng)論 2 354

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