即刻App 點(diǎn)贊數(shù)字變化動(dòng)畫效果的實(shí)現(xiàn)

前言

久違的再次動(dòng)筆寫博客匾嘱。撬讽。。

說明

從一年前開始使用即刻這款A(yù)PP靖榕, 一直覺得它的細(xì)節(jié)做的特別好, 最近開始動(dòng)筆颤枪,仿寫它的一些效果實(shí)現(xiàn)淑际, 從點(diǎn)贊數(shù)字變化開始畏纲。

Demo 地址

https://github.com/94haox/AnimationNumber

實(shí)現(xiàn)

開始

先看下效果(GIF圖,沒做好需要等一等春缕,當(dāng)然我建議是直接下載Demo盗胀,運(yùn)行。锄贼。票灰。 )


效果圖

在點(diǎn)贊的時(shí)候, 點(diǎn)贊數(shù)+1宅荤, 或者取消點(diǎn)贊 -1屑迂, 只在個(gè)位(當(dāng)然臨界點(diǎn)是牽扯到各個(gè)位)實(shí)現(xiàn) 向上平滑而出, 或者向下平滑而出的動(dòng)畫

思考

  • 首先點(diǎn)贊數(shù)是一個(gè)字符串冯键,我們需要對字符串中的單個(gè)字符進(jìn)行操作惹盼,那么首先, 我們需要將字符串分割惫确,便于操作手报;
  • 分割成單個(gè)字符后蚯舱, 需要用UILabel 顯示, 意味著需要相同個(gè)數(shù)的Label掩蛤;
  • 對Label做動(dòng)畫晓淀;

動(dòng)手敲代碼

新建一個(gè)類 繼承于UIView, AnimationNumber;

添加一些便于自定義label的屬性

// 顯示label的字體
@property (nonatomic, strong)UIFont *numberFont;
// 顯示label的顏色
@property (nonatomic, strong)UIColor *numberColor;
// 用于接收字符串
@property (nonatomic, copy) NSString *currentNumber;

在 Extension 中添加一些我們不需要暴露在外的屬性

// 之前顯示的數(shù)字
@property (nonatomic, strong) NSMutableArray<NSString *> *oldNumbers;
// 之前顯示的Label
@property (nonatomic, strong) NSMutableArray<UILabel *> *oldLabelList;
// 當(dāng)前顯示的數(shù)字
@property (nonatomic, strong) NSMutableArray<NSString *> *currentNumbers;
// 當(dāng)前顯示的Label
@property (nonatomic, strong) NSMutableArray<UILabel *> *currentLabelList;
@property (nonatomic, strong) UIView *contentView;

在第一次接收字符串的時(shí)候, oldNumbers盏档,oldLabelList 應(yīng)該是空的凶掰;只有在第二次接收的時(shí)候才會(huì)儲(chǔ)存上一次的分割后的number, 和Label蜈亩;

初始化

- (instancetype)initWithFrame:(CGRect)frame{
  if (self = [super initWithFrame:frame]) {
    self.oldNumbers = [NSMutableArray arrayWithCapacity:1];
    self.currentNumbers = [NSMutableArray arrayWithCapacity:1];
    self.oldLabelList = [NSMutableArray arrayWithCapacity:1];
    self.currentLabelList = [NSMutableArray arrayWithCapacity:1];

    // contentView 寫成了懶加載懦窘, 看Demo
    [self addSubview:self.contentView]; 
  }
  return self;
}

下面說明, 主要實(shí)現(xiàn)的方法稚配;

 // 分割 字符串
- (void)carveUpNumberWith:(NSString *)number{
  NSMutableArray<UILabel *> *labelsList = [NSMutableArray array];
  NSMutableArray<NSString *> *numbersList = [NSMutableArray array];
  for (int i = 0; i < number.length; i++) {
    NSString *stringItem = [number substringWithRange:NSMakeRange(i, 1)];
    // Label創(chuàng)建 看Demo
    UILabel *label = [self createLabels:stringItem];
    CGRect frame = label.frame;
    // 第一個(gè)Label
    frame.origin.x = labelsList.count > 0 ? CGRectGetMaxX(labelsList.lastObject.frame) : 0;
    frame.origin.y = 0;
    label.frame = frame;
    [labelsList addObject:label];
    [numbersList addObject:stringItem];
  }
  self.currentLabelList = labelsList;
  self.currentNumbers = numbersList;
}

最重要的方法


實(shí)現(xiàn)動(dòng)畫
- (void)updateLabelsWithNumber:(NSString *)number{
    //  通過oldLabelList.count判斷是否是第一次接收字符串
  if (self.oldLabelList.count > 0) {
    // 判斷兩次數(shù)字的差別畅涂, 從最后一位開始比較
    NSInteger length = number.length;
    NSInteger oldLength = self.oldLabelList.count;
    for (int i = 0; i < self.currentNumbers.count; i ++) {
      NSString *item = [number substringWithRange:NSMakeRange(length - i-1, 1)];
      UILabel *label = self.currentLabelList[length-i-1];
      // 判斷 防止數(shù)組越界
      if (i < self.oldLabelList.count) {
        NSString *oldItem = self.oldNumbers[oldLength - i-1];
        UILabel *oldLabel = self.oldLabelList[oldLength-i-1];
        // 判斷相同位置, 是否數(shù)字相同
        if (![oldItem isEqualToString:item]) {
        //  相同位置道川, 單個(gè)數(shù)字午衰, 現(xiàn)在比之前大, 動(dòng)畫從上往下冒萄, 現(xiàn)在比之前小則從下往上
          CGRect frame = label.frame;
          if (oldItem.integerValue < item.integerValue) {
           frame.origin.y = - label.frame.size.height;
          }else{
           frame.origin.y = label.frame.size.height;
          }
          label.frame = frame;

          [UIView animateWithDuration:animationDuration delay:0 options:UIViewAnimationOptionCurveEaseOut animations:^{
            CGRect frame = label.frame;
            frame.origin.y = 0;
            label.frame = frame;

            CGRect oldFrame = oldLabel.frame;
            if (oldItem.integerValue < item.integerValue) {
              oldFrame.origin.y = oldLabel.frame.size.height;
            }else{
              oldFrame.origin.y = -oldLabel.frame.size.height;
            }
            oldLabel.frame = oldFrame;
          } completion:^(BOOL finished) {
            // 做完動(dòng)畫臊岸, 移出視圖
            [oldLabel removeFromSuperview];
          }];
        }else{
          [oldLabel removeFromSuperview];
        }
        
      }else{
        CGRect frame = label.frame;
        frame.origin.y = - label.frame.size.height;
        label.frame = frame;
        
        [UIView animateWithDuration:animationDuration delay:0 options:UIViewAnimationOptionCurveEaseOut animations:^{
          CGRect frame = label.frame;
          frame.origin.y = 0;
          label.frame = frame;
        } completion:nil];
        
      }
    }
  }
  
  // 當(dāng)之前的數(shù)值比較大時(shí), 移除多出的位數(shù)
  if (self.oldLabelList.count > self.currentLabelList.count) {
    for (int i = 0; i < self.oldLabelList.count - self.currentLabelList.count; i ++) {
      UILabel *label = self.oldLabelList[i];
      [label removeFromSuperview];
    }
  }
  
  self.oldLabelList = self.currentLabelList;
  self.oldNumbers = self.currentNumbers;
  
}

結(jié)束

其實(shí)效果挺簡單尊流, 當(dāng)然實(shí)現(xiàn)也挺簡單帅戒, 可能有更好的實(shí)現(xiàn)方法, 希望看到的朋友崖技,告訴我逻住。。迎献。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末瞎访,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子吁恍,更是在濱河造成了極大的恐慌扒秸,老刑警劉巖,帶你破解...
    沈念sama閱讀 207,113評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件践盼,死亡現(xiàn)場離奇詭異鸦采,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)咕幻,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評論 2 381
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來顶霞,“玉大人肄程,你說我怎么就攤上這事锣吼。” “怎么了蓝厌?”我有些...
    開封第一講書人閱讀 153,340評論 0 344
  • 文/不壞的土叔 我叫張陵玄叠,是天一觀的道長。 經(jīng)常有香客問我拓提,道長读恃,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,449評論 1 279
  • 正文 為了忘掉前任代态,我火速辦了婚禮寺惫,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘蹦疑。我一直安慰自己西雀,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,445評論 5 374
  • 文/花漫 我一把揭開白布歉摧。 她就那樣靜靜地躺著艇肴,像睡著了一般。 火紅的嫁衣襯著肌膚如雪叁温。 梳的紋絲不亂的頭發(fā)上再悼,一...
    開封第一講書人閱讀 49,166評論 1 284
  • 那天,我揣著相機(jī)與錄音膝但,去河邊找鬼帮哈。 笑死,一個(gè)胖子當(dāng)著我的面吹牛锰镀,可吹牛的內(nèi)容都是我干的娘侍。 我是一名探鬼主播,決...
    沈念sama閱讀 38,442評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼泳炉,長吁一口氣:“原來是場噩夢啊……” “哼憾筏!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起花鹅,我...
    開封第一講書人閱讀 37,105評論 0 261
  • 序言:老撾萬榮一對情侶失蹤氧腰,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后刨肃,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體古拴,經(jīng)...
    沈念sama閱讀 43,601評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,066評論 2 325
  • 正文 我和宋清朗相戀三年真友,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了黄痪。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,161評論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡盔然,死狀恐怖桅打,靈堂內(nèi)的尸體忽然破棺而出是嗜,到底是詐尸還是另有隱情,我是刑警寧澤挺尾,帶...
    沈念sama閱讀 33,792評論 4 323
  • 正文 年R本政府宣布鹅搪,位于F島的核電站,受9級(jí)特大地震影響遭铺,放射性物質(zhì)發(fā)生泄漏丽柿。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,351評論 3 307
  • 文/蒙蒙 一魂挂、第九天 我趴在偏房一處隱蔽的房頂上張望甫题。 院中可真熱鬧,春花似錦锰蓬、人聲如沸幔睬。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽麻顶。三九已至,卻和暖如春舱卡,著一層夾襖步出監(jiān)牢的瞬間辅肾,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評論 1 261
  • 我被黑心中介騙來泰國打工轮锥, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留矫钓,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,618評論 2 355
  • 正文 我出身青樓舍杜,卻偏偏與公主長得像新娜,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子既绩,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,916評論 2 344

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

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理概龄,服務(wù)發(fā)現(xiàn),斷路器饲握,智...
    卡卡羅2017閱讀 134,601評論 18 139
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 171,525評論 25 707
  • 發(fā)現(xiàn) 關(guān)注 消息 iOS 第三方庫私杜、插件、知名博客總結(jié) 作者大灰狼的小綿羊哥哥關(guān)注 2017.06.26 09:4...
    肇東周閱讀 12,033評論 4 62
  • synthesize 和 dynamic區(qū)別 救欧? @synthesize 的語義是如果你沒有手動(dòng)實(shí)現(xiàn) setter...
    英雄出少年閱讀 133評論 0 0
  • 那日 你踱水而來衰粹,含情脈脈 我執(zhí)卷繾綰,淡淡一笑 那月 你水邊牧馬笆怠,林下飲茶 我溪里浣紗铝耻,洗手羹湯 那...
    剪痕閱讀 253評論 0 7