視錯覺:從一個看似簡單的自定義控件提及

爲何要寫今天這篇博客那就說來話長了,那是在一個大雪紛飛的冬天……而後……。好了,不扯淡了,直接進入今天的主題吧,這篇博客是關於iOS自定義組件的東西。一些UI效果看起來彷佛是這個樣子,其實本質不是這個樣子。在作一些UI效果時咱們能夠利用視錯覺的一些東西,讓用戶看到的是一個東西,其實你實現的又是一個東西。原則是千方百計騙過用戶的眼睛。視錯覺若是和UI實現結合起來,有時會產生意想不到的效果。html

另外一篇使用Mask實現一樣效果的博客地址以下:http://www.cnblogs.com/ludashi/p/5671133.htmlgit

一.視錯覺的概述github

引用--「視錯覺就是當人觀察物體時,基於經驗主義或不當的參照造成的錯誤的判斷和感知。視錯:是指 觀察者在客觀因素干擾下或者自身的心理因素支配下,對圖形產生的與客觀事實不相符的錯誤的感受。編程

看上面這個解釋彷佛有點複雜,其實說白了就是欺騙你本身的眼睛。眼見不必定爲實。視錯覺的經常使用例子:矮中見高、虛中見實、冷調降溫、粗中見細、曲中見直等等經常使用的手法。說這麼多,接下來我想用一組圖來直觀的感覺一下視錯覺。(圖片來源與網絡數組

  1.這裏不是起點,那裏也不是終點。網絡

    2.平衡?別想了。工具

    3.一個很是經典的幻覺。字體

上面是在網上找的視錯覺的圖片,像上面這些視錯覺的經典案例還有不少呢,請你們自行百度吧。上面算是今天博客的引子吧,接下來就和開發有關了。動畫

 

2、一個利用視錯覺的自定義組件atom

當第一次看到這個組件效果時,感受沒有什麼特別之處,就是一個普通在普通不過的組件。但是再仔細看就感受不同了,一些細節處理的很是好。看到組件效果時,由於沒有源碼,因此當時也不知道如何實現的。就想唄,而後就想到了用「視錯覺」這個高大上的東西來實現。固然這個實現方案是我本身想出來的,我不知道原做者是如何實現的,下面給出了我本身的實現方案。 說這麼多,先來看一下運行效果吧。

1. 該組件的運行效果以下,其實就是一個自定義的SegmentControl。看到該組件的時候,個人第一印象是:「這個組件應該挺好作的」。當時感受就是幾個Button, 而後紅色的是一個UIView, 點擊那個Button時,就把UIView經過動畫的形式移動到當前點擊的Button。頂多就是封裝一下,成爲一個自定義組件,而後給別人使用。

2. 在仔細看效果,感受本身仍是太年輕,太單純了。這個組件遠遠不是我想的那樣,上面組件的實現的重點與難點不在於如何去運動,如何去封裝它。而在於下面這個截圖中的東西。若是把動畫放慢,你會發現一個細節,這個細節處理的很是的巧妙,也是這個組件的亮點與難點所在。下方是切換時放慢的一個效果。看到這個細節時,瞬間顛覆了我以前單純的想法。這個組件遠遠沒有我想的這麼簡單。

在切換時,有一個細節,就是在紅色區域中的文字(或文字的一部分)隨着紅色區域的移動,文字的顏色也會隨之部分改變。當紅色區域移動事後,字體顏色又變爲原來的了看到這個效果,對這個組件的崇拜感就油然而生了。瞬間也蒙圈了,一時沒有解決思路。大腦中充滿了無數的問號。這究竟用了什麼黑科技!?實現了這麼NX的效果。晚上作地鐵回家的時候也一直在想其解決方案。果真在地鐵上靈光一現,應該就是用它:「視錯覺」。因而乎就回家晚飯都沒吃,就拿出筆記本開始按着本身的思路去實現。功夫不負有心人呢,因此纔有了今天的博客。固然實現上述效果是我本身的思路,若是還有其餘更好的實現方式歡迎交流。

 

3、實現原理

1.原理介紹

實現上述效果時,若是按着我看到的就是看到的來實現的話,估計會無從下手的。一個原則:「眼見爲虛」,就OK了。今天這篇博客,我不想往上粘貼過多的代碼,仍是把個人思路給你們分享一下就好。其實編程這東西,有時候思路比代碼更爲重要。下方我想用在Reveal工具上的分析層次來給你們聊一下實現原理。

經過Reveal上面的UI層次咱們很容易看出,這個組件遠比咱們想象的要複雜的多。個人實現方式以下:

      (1) 先在View上加上一層的Label, 這些Label用來顯示常規的字體顏色(未選中時的顏色)(黑色的字)

      (2) 在以前的Label上添加一層View , 動畫元素,高亮顯示的字體,點擊的按鈕都在這個View上

      (3) 在這個View上添加一層高亮的Label(白色的字), Label的字體,大小,位置等要和底層的Label一致(除了顏色除外)

      (4) 上層的View的大小要和一個Label的大小一致,而且設置超出View的子視圖不顯示。

      (5) 移動View(紅色部分)時,也要移動View上白色的字。要保持移動的過程當中,白色Label和黑色label徹底重合。

這樣View移動到那個label上時,就會把後邊的黑色Label給擋上,顯示的是上面白色的Label.原理大概就是這個原理,原理一旦知道怎麼回事了,至於實現起來就簡單許多了。 


2.該自定組件可配置項以下:

 1 //
 2 //  ZLCustomeSegmentControlView.h
 3 //  CustomeSegmentControl
 4 //
 5 //  Created by ZeluLi on 15/11/19.
 6 //  Copyright © 2015年 zeluli. All rights reserved.
 7 //
 8 
 9 #import <UIKit/UIKit.h>
10 
11 typedef void(^ButtonOnClickBlock)(NSInteger tag, NSString * title);
12 
13 @interface ZLCustomeSegmentControlView : UIView
14 
15 @property (nonatomic, strong) NSArray *titles;                      //標題數組
16 @property (nonatomic, strong) UIColor *titlesCustomeColor;          //標題的常規顏色
17 @property (nonatomic, strong) UIColor *titlesHeightLightColor;      //標題高亮顏色
18 @property (nonatomic, strong) UIColor *backgroundHeightLightColor;  //高亮時的顏色
19 @property (nonatomic, strong) UIFont *titlesFont;                   //標題的字號
20 @property (nonatomic, assign) CGFloat duration;                     //運動時間
21 
22 /**
23  *  點擊按鈕的回調
24  *
25  *  @param block 點擊按鈕的Block
26  */
27 -(void) setButtonOnClickBlock: (ButtonOnClickBlock) block;
28 
29 @end

 

   3.該自定義組件的調用方式

 1     ZLCustomeSegmentControlView *v = [[ZLCustomeSegmentControlView alloc] initWithFrame:CGRectMake(30, 100, SCREEN_WIDTH - 60, 50)];
 2     
 3     v.titles = @[@"Hello", @"Apple", @"Swift", @"Objc"];
 4     v.duration = 0.7f;
 5     
 6     [v setButtonOnClickBlock:^(NSInteger tag, NSString *title) {
 7         NSLog(@"index = %ld, title = %@", (long)tag, title);
 8     }];
 9     
10     [self.view addSubview:v];

 

因爲某些緣由呢,今天博客中就不一一往上粘貼代碼了。有好的思路歡迎交流~
上述Demo在GitHub上分享連接以下:https://github.com/lizelu/ZLCustomeSegmentControlView

相關文章
相關標籤/搜索