iOS中矩形視圖顯示圓形區(qū)域

前段時(shí)間項(xiàng)目中有個(gè)特殊的需求烁挟,折騰了好久破衔,覺得有必要總結(jié)一下辟躏。我自己寫了一個(gè)日歷控件若债,產(chǎn)品要求選中的日期要用圓形圖片顯示出來乔妈,如圖所示


圖1.jpeg

方案一

剛開始以為只要用layer.cornerRadius就可以了序目,但是真正實(shí)現(xiàn)起來的時(shí)候才發(fā)現(xiàn)不是那么簡(jiǎn)單犀斋。因?yàn)槲业娜諝v控件寬度設(shè)置為屏幕寬度奏篙,那么問題就來了润努,日歷一行只能顯示七天关斜,所以不同屏幕寬度下每個(gè)單元view的寬度就不固定了,而且還不是正方形铺浇,用layer.cornerRadius的話就會(huì)出來一個(gè)橢圓形的選中效果痢畜,第一種方案就放棄了。

方案二

第二種方案我是想著放一個(gè)遮罩,既然itemView不一定是正方形丁稀,那我們想辦法讓它只顯示出來正方形的區(qū)域就好了吼拥。于是用UIBezierPath+CAShapeLayer畫出一個(gè)圓形的遮罩添加在每個(gè)單元view的layer上,設(shè)置為itemView.layer.mask為畫出的圓形CAShapeLayer线衫,運(yùn)行一看滿足了要求凿可。但出現(xiàn)一個(gè)新的問題就是,每次進(jìn)到這個(gè)界面的時(shí)候屏幕都會(huì)卡頓授账,能看出明顯的抖動(dòng)枯跑,這肯定是不能忍的。只能查找原因優(yōu)化~
大概原因是設(shè)置layer的mask屬性會(huì)觸發(fā)離屏渲染白热,大大的消耗了性能敛助,所以卡頓。其中有一種優(yōu)化方案就是開啟GPU的柵格化棘捣,然后把需要渲染的畫面緩存起來,等下次進(jìn)來的時(shí)候可以直接加載

 self.layer.shouldRasterize = YES;
 self.layer.rasterizationScale = self.layer.contentsScale;

試了一下果然不卡了休建,但是新的問題又出現(xiàn)了乍恐,就是日歷上的數(shù)字都變得模糊了。我推斷出大概是以下原因测砂,當(dāng)shouldRasterize設(shè)成true時(shí)茵烈,layer被渲染成一個(gè)bitmap,并緩存起來砌些,等下次使用時(shí)不會(huì)再重新去渲染了呜投。實(shí)現(xiàn)圓角本身就是在做顏色混合(blending),如果每次頁(yè)面出來時(shí)都blending存璃,消耗太大仑荐,這時(shí)shouldRasterize = yes,下次就只是簡(jiǎn)單的從渲染引擎的cache里讀取那張bitmap纵东,節(jié)約系統(tǒng)資源粘招。我這邊要渲染的是文字,估計(jì)在layer被渲染成bitmap的過程或者讀取bitmap的時(shí)候產(chǎn)生了一些誤差偎球,所以導(dǎo)致文字看上去有些模糊洒扎。
接著優(yōu)化~

方案三

我在itemView上邊覆蓋一個(gè)中間為圓形透明區(qū)域的圖片,正方形的圖片中心點(diǎn)與itemView的中心點(diǎn)重合衰絮。這樣解決了卡頓問題袍冷,而且也很清晰,但是新的問題出現(xiàn)了猫牡,就是我選中的時(shí)候要改變itemView的顏色胡诗,這時(shí)候如果itemView寬度大于高度的時(shí)候,就會(huì)出現(xiàn)左右兩邊的顏色沒有被覆蓋的情況,沒辦法……
接著優(yōu)化~

方案四

反向選取區(qū)域乃戈,既然處理不了中間的圓形區(qū)域褂痰,那我能不能把周圍的處理一下嘞~最后終于是功夫不負(fù)有心人,就是這個(gè)方法bezierPathByReversingPath症虑,用代碼說話吧

- (void)drawRect:(CGRect)rect{
    [super drawRect:rect];
    CGContextRef context = UIGraphicsGetCurrentContext();
    CGFloat radius = MIN(self.width, self.height)*0.5-2;
    UIBezierPath *path = [UIBezierPath bezierPathWithRect:self.bounds];
    UIBezierPath *clipPath = [[UIBezierPath bezierPathWithRoundedRect:CGRectMake((self.width-radius*2)*0.5, (self.height-radius*2)*0.5, radius*2, radius*2) cornerRadius:radius] bezierPathByReversingPath];
    [path appendPath:clipPath];
    CGContextAddPath(context, path.CGPath);
    CGContextSetFillColorWithColor(context, [UIColor whiteColor].CGColor);
    CGContextFillPath(context);
}

我自定義了一個(gè)繼承自UILabel的GRCalendarTileLabel控件缩歪,在drawRect方法中把中間圓形區(qū)域以外的區(qū)域設(shè)置為了白色,這樣一來不管你外邊是什么形狀谍憔,我只留出中間一塊兒圓形區(qū)域匪蝙,這里記得調(diào)用[super drawRect:rect]方法,不然給Label設(shè)置text的時(shí)候是不會(huì)顯示的习贫。如下圖所示


圖2.jpeg

這樣一來逛球,問題就算完美解決了。雖然過程中費(fèi)了不少周折苫昌,但是問題解決的時(shí)候還是很有成就感滴颤绕,特此記錄總結(jié)一下,希望能給有同樣需求的小伙伴一點(diǎn)幫助祟身,歡迎大家積極提出指導(dǎo)意見~~~

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末奥务,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子袜硫,更是在濱河造成了極大的恐慌氯葬,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,198評(píng)論 6 514
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件婉陷,死亡現(xiàn)場(chǎng)離奇詭異帚称,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)秽澳,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,334評(píng)論 3 398
  • 文/潘曉璐 我一進(jìn)店門闯睹,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人担神,你說我怎么就攤上這事瞻坝。” “怎么了杏瞻?”我有些...
    開封第一講書人閱讀 167,643評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵所刀,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我捞挥,道長(zhǎng)浮创,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,495評(píng)論 1 296
  • 正文 為了忘掉前任砌函,我火速辦了婚禮斩披,結(jié)果婚禮上溜族,老公的妹妹穿的比我還像新娘。我一直安慰自己垦沉,他們只是感情好煌抒,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,502評(píng)論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著厕倍,像睡著了一般寡壮。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上讹弯,一...
    開封第一講書人閱讀 52,156評(píng)論 1 308
  • 那天况既,我揣著相機(jī)與錄音,去河邊找鬼组民。 笑死棒仍,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的臭胜。 我是一名探鬼主播莫其,決...
    沈念sama閱讀 40,743評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼耸三!你這毒婦竟也來了乱陡?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,659評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤吕晌,失蹤者是張志新(化名)和其女友劉穎蛋褥,沒想到半個(gè)月后临燃,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體睛驳,經(jīng)...
    沈念sama閱讀 46,200評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,282評(píng)論 3 340
  • 正文 我和宋清朗相戀三年膜廊,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了乏沸。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,424評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡爪瓜,死狀恐怖蹬跃,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情铆铆,我是刑警寧澤蝶缀,帶...
    沈念sama閱讀 36,107評(píng)論 5 349
  • 正文 年R本政府宣布,位于F島的核電站薄货,受9級(jí)特大地震影響翁都,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜谅猾,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,789評(píng)論 3 333
  • 文/蒙蒙 一柄慰、第九天 我趴在偏房一處隱蔽的房頂上張望鳍悠。 院中可真熱鬧,春花似錦坐搔、人聲如沸藏研。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,264評(píng)論 0 23
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)蠢挡。三九已至,卻和暖如春占锯,著一層夾襖步出監(jiān)牢的瞬間袒哥,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,390評(píng)論 1 271
  • 我被黑心中介騙來泰國(guó)打工消略, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留堡称,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,798評(píng)論 3 376
  • 正文 我出身青樓艺演,卻偏偏與公主長(zhǎng)得像却紧,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子胎撤,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,435評(píng)論 2 359

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

  • 每個(gè)UIView有一個(gè)伙伴稱為layer晓殊,一個(gè)CALayer。UIView實(shí)際上并沒有把自己畫到屏幕上;它繪制本身...
    shenzhenboy閱讀 3,113評(píng)論 0 17
  • 轉(zhuǎn)載:http://www.reibang.com/p/32fcadd12108 每個(gè)UIView有一個(gè)伙伴稱為l...
    F麥子閱讀 6,220評(píng)論 0 13
  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,262評(píng)論 25 707
  • 1.xcode5和xcode7區(qū)別 1.xcode7沒有Frameworks文件夾,xcode7內(nèi)部會(huì)自動(dòng)幫你導(dǎo)入...
    彼岸的黑色曼陀羅閱讀 517評(píng)論 0 2
  • 漫天黃沙伤提,依然想出來走走巫俺。 一個(gè)人走在嘈雜的街,呼嘯而過的風(fēng)肿男,又怎會(huì)讀懂一顆叮叮當(dāng)當(dāng)?shù)男慕樾凇S孀邅硪粚?duì)...
    蘭貴人閱讀 223評(píng)論 0 1