模仿TableViewCell的上下移動(dòng)寫(xiě)的一個(gè)長(zhǎng)按UIButton可上下移動(dòng)的原型(swift)

有時(shí)候并沒(méi)有用到TableView嗅绸,但也想實(shí)現(xiàn)長(zhǎng)按上下移動(dòng),比如多個(gè)按鈕實(shí)現(xiàn)長(zhǎng)按移動(dòng)。所以就寫(xiě)了一個(gè)簡(jiǎn)單的方法來(lái)實(shí)現(xiàn)這個(gè)效果配椭。此效果的基礎(chǔ)是所有按鈕全部代碼創(chuàng)建,是基于更改frame來(lái)實(shí)現(xiàn)的

效果圖

效果圖.gif

原理

  • 每個(gè)按鈕要都添加一個(gè)tag
  • 給每個(gè)按鈕添加一個(gè)長(zhǎng)按手勢(shì)
  let longPressGes = UILongPressGestureRecognizer.init(target:   self, action: #selector(longPressButton))
  button.addGestureRecognizer(longPressGes)
  • longPressButton方法里添加實(shí)現(xiàn)

講解

  • 屬性

      lazy var buttonArray = NSMutableArray() //按鈕數(shù)組
      var tag              = 0                //按鈕tag
      var endButtonTag     = 0                //用來(lái)保存移動(dòng)的按鈕最后停止時(shí)的tag
      var upY: CGFloat?                //移動(dòng)的按鈕之上的按鈕的中心y軸
      var downY: CGFloat?             //移動(dòng)的按鈕之下的按鈕的中心y軸
      var beginY: CGFloat?           //用以保存移動(dòng)的按鈕的y軸
      var oldFrame: CGRect?         //用以保存移動(dòng)的按鈕的最后停止的位置frame
    
  • 創(chuàng)建按鈕樣例

    • 循環(huán)調(diào)用
    for _ in 0..<10 {
              let button =  self.addButton(CGRectMake(0, 69 + CGFloat(self.buttonArray.count) * 49, self.view.frame.width, 44), title: "\(tag)")
              self.buttonArray.addObject(button)
    }
    
    • 方法

// MARK: 添加按鈕
func addButton(rect: CGRect, title: String) -> UIButton {
let button = UIButton.init(type: UIButtonType.System)
button.backgroundColor = UIColor(red: 0.94, green: 0.94, blue: 0.94, alpha: 1.00)
button.setTitle(title, forState: UIControlState.Normal)
button.frame = rect
button.tag = tag
tag += 1
let longPressGes = UILongPressGestureRecognizer.init(target: self, action: #selector(longPressButton))
button.addGestureRecognizer(longPressGes)
self.view.addSubview(button)
return button
}
```


  • 實(shí)現(xiàn)上下移動(dòng)

    // MARK: 長(zhǎng)按按鈕移動(dòng)
      func longPressButton(sender: UIGestureRecognizer) {
          let view = sender.view as! UIButton //獲取手勢(shì)的視圖雹姊,既移動(dòng)的按鈕
          let maxY = CGRectGetMaxY(view.frame)//獲取按鈕的最大y軸
          view.center = sender.locationInView(self.view)
          if sender.state == UIGestureRecognizerState.Began {   
                //Func:1
          }
          if sender.state == UIGestureRecognizerState.Changed {  
               //Func:2
          }
          if sender.state == UIGestureRecognizerState.Ended {
              //Func:3
          }   
      }
    
  • Fun:1里面股缸,也就是開(kāi)始手勢(shì)的時(shí)候,將一些屬性賦值

     endButtonTag = view.tag  //用來(lái)保存移動(dòng)的按鈕最后停止時(shí)的tag
     oldFrame = view.frame //用以保存移動(dòng)的按鈕的最后停止的位置frame
     upY = view.frame.origin.y - 27 //移動(dòng)的按鈕之上的按鈕的中心y軸
     downY = maxY + 27  //移動(dòng)的按鈕之下的按鈕的中心y軸
     beginY = view.frame.origin.y //用以保存移動(dòng)的按鈕的y軸
    
    
  • Fun:2手勢(shì)改變的時(shí)候來(lái)處理上下移動(dòng)的邏輯

  • 向上移動(dòng)


    if view.frame.origin.y < upY   {
         if view.tag != 0 {
             let frontButton = self.buttonArray[view.tag-1] as! UIButton
             oldFrame = CGRectMake(oldFrame!.origin.x, frontButton.frame.origin.y, oldFrame!.width, oldFrame!.height)
             upY = frontButton.frame.origin.y - 27
             var frame = frontButton.frame
             frame.origin.y = frame.origin.y + 49
             downY = frame.origin.y + 22
             endButtonTag = frontButton.tag
             self.buttonArray[frontButton.tag + 1] = frontButton
             frontButton.tag = frontButton.tag + 1
             UIView.animateWithDuration(0.1, animations: {
                 frontButton.frame = frame
             })
             view.tag -= 1
        }
    }
  ```
  - 首先第一個(gè)按鈕`tag = 0`不要做向上移動(dòng)的處理,
  - 然后具體實(shí)現(xiàn)的邏輯是:每向上移動(dòng)一格吱雏,就將原來(lái)上面的按鈕的`frame`保存起來(lái)敦姻,將按鈕數(shù)組中的對(duì)應(yīng)的兩個(gè)按鈕互相交換瘾境,同時(shí)上面的按鈕和移動(dòng)中的按鈕交換`tag`,再將上面的按鈕移動(dòng)到移動(dòng)的按鈕的下面镰惦,這樣就完成了向上移動(dòng)的實(shí)現(xiàn)迷守。

-  向下移動(dòng)

  ```
  if maxY > downY  {
              if view.tag != self.buttonArray.count-1 {
                  let afterButton = self.buttonArray[view.tag + 1] as! UIButton
                  oldFrame = CGRectMake(oldFrame!.origin.x, afterButton.frame.origin.y, oldFrame!.width, oldFrame!.height)
                  downY = CGRectGetMaxY(afterButton.frame) + 27
                  var midFrame = afterButton.frame
                  midFrame.origin.y = midFrame.origin.y - 49
                  upY = midFrame.origin.y + 22
                  endButtonTag = afterButton.tag
                  self.buttonArray[afterButton.tag - 1] = afterButton
                  afterButton.tag -= 1
                  UIView.animateWithDuration(0.1, animations: {
                      afterButton.frame = midFrame
                  })
                  view.tag += 1
              }
          }
  ```
  - 最后一個(gè)按鈕不做向下移動(dòng)
  - 具體實(shí)現(xiàn)邏輯與向上移動(dòng)類(lèi)似:每向下移動(dòng)一格,就將原來(lái)下面的按鈕的`frame`保存起來(lái)陨献,將按鈕數(shù)組中的對(duì)應(yīng)的兩個(gè)按鈕互相交換盒犹,同時(shí)下面的按鈕和移動(dòng)中的按鈕交換`tag`懂更,再將下面的按鈕移動(dòng)到移動(dòng)中的按鈕的上面眨业,完成向下移動(dòng)的實(shí)現(xiàn)。

- 在`Fun:3`結(jié)束手勢(shì)時(shí)

self.buttonArray[endButtonTag] = view
if view.frame.origin.x <= self.view.frame.width {
UIView.animateWithDuration(0.1, animations: {
view.frame = self.oldFrame!
})
}

邏輯:將移動(dòng)中的按鈕替換掉最后停止的位置的`tag`對(duì)應(yīng)的數(shù)組中的值沮协,再將移動(dòng)中的按鈕移動(dòng)到最后的位置即可龄捡。

---

## 最后
這個(gè)方法實(shí)現(xiàn)起來(lái)并不是很難,代碼量也不多慷暂∑钢常可能還有更簡(jiǎn)單的方法來(lái)實(shí)現(xiàn),如果有人了解可以告知行瑞,在這先提前感謝奸腺。
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市血久,隨后出現(xiàn)的幾起案子突照,更是在濱河造成了極大的恐慌,老刑警劉巖氧吐,帶你破解...
    沈念sama閱讀 222,000評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件讹蘑,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡筑舅,警方通過(guò)查閱死者的電腦和手機(jī)座慰,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,745評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)翠拣,“玉大人版仔,你說(shuō)我怎么就攤上這事∥竽梗” “怎么了蛮粮?”我有些...
    開(kāi)封第一講書(shū)人閱讀 168,561評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)优烧。 經(jīng)常有香客問(wèn)我蝉揍,道長(zhǎng),這世上最難降的妖魔是什么畦娄? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,782評(píng)論 1 298
  • 正文 為了忘掉前任又沾,我火速辦了婚禮弊仪,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘杖刷。我一直安慰自己励饵,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,798評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布滑燃。 她就那樣靜靜地躺著役听,像睡著了一般。 火紅的嫁衣襯著肌膚如雪表窘。 梳的紋絲不亂的頭發(fā)上典予,一...
    開(kāi)封第一講書(shū)人閱讀 52,394評(píng)論 1 310
  • 那天,我揣著相機(jī)與錄音乐严,去河邊找鬼瘤袖。 笑死,一個(gè)胖子當(dāng)著我的面吹牛昂验,可吹牛的內(nèi)容都是我干的捂敌。 我是一名探鬼主播,決...
    沈念sama閱讀 40,952評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼既琴,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼占婉!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起甫恩,我...
    開(kāi)封第一講書(shū)人閱讀 39,852評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤逆济,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后填物,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體纹腌,經(jīng)...
    沈念sama閱讀 46,409評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,483評(píng)論 3 341
  • 正文 我和宋清朗相戀三年滞磺,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了升薯。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,615評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡击困,死狀恐怖涎劈,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情阅茶,我是刑警寧澤蛛枚,帶...
    沈念sama閱讀 36,303評(píng)論 5 350
  • 正文 年R本政府宣布,位于F島的核電站脸哀,受9級(jí)特大地震影響蹦浦,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜撞蜂,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,979評(píng)論 3 334
  • 文/蒙蒙 一盲镶、第九天 我趴在偏房一處隱蔽的房頂上張望侥袜。 院中可真熱鬧,春花似錦溉贿、人聲如沸枫吧。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,470評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)九杂。三九已至,卻和暖如春宣蠕,著一層夾襖步出監(jiān)牢的瞬間例隆,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,571評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工植影, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留裳擎,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 49,041評(píng)論 3 377
  • 正文 我出身青樓思币,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親羡微。 傳聞我的和親對(duì)象是個(gè)殘疾皇子谷饿,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,630評(píng)論 2 359

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

  • 1.badgeVaule氣泡提示 2.git終端命令方法> pwd查看全部 >cd>ls >之后桌面找到文件夾內(nèi)容...
    i得深刻方得S閱讀 4,675評(píng)論 1 9
  • *7月8日上午 N:Block :跟一個(gè)函數(shù)塊差不多,會(huì)對(duì)里面所有的內(nèi)容的引用計(jì)數(shù)+1妈倔,想要解決就用__block...
    炙冰閱讀 2,492評(píng)論 1 14
  • //設(shè)置尺寸為屏幕尺寸的時(shí)候self.window = [[UIWindow alloc] initWithFra...
    LuckTime閱讀 818評(píng)論 0 0
  • 一博投、控件的屬性 1.CGRect frame 1> 表示控件的位置和尺寸(以父控件的左上角為坐標(biāo)原點(diǎn)(0, 0))...
    清澈Saup閱讀 1,132評(píng)論 0 51
  • 我不是一個(gè)把收納運(yùn)用得很好的人,可能正因?yàn)槿绱硕⒑系墼谖疑磉叞才帕艘恍疤焓埂币慊瑢?duì)于創(chuàng)造整潔的室內(nèi)環(huán)境,她們各有所...
    觀舟閱讀 407評(píng)論 0 0