原文:http://blog.csdn.net/ch_soft/article/details/6947695/html
UIScrollView爲了顯示多於一個屏幕的內容或者超過你能放在內存中的內容。架構
Scroll View爲你處理縮小放大手勢,UIScrollView實現了這些手勢,而且替你處理對於它們的探測和迴應。其中須要注意的子類是UITableView以及UITextView(用來顯示大量的文字)。還有一個UIWebView,儘管那不是UIScrollView的直接子類,它適用UIScrollView去顯示網頁內容框架
contentsize是內容的寬和高,contentsize.width是內容的寬度,contentsize.heght是高度,contentsize是UIScrollView的一個屬性,它是一個CGSize,是由核心圖形所定義的架構,那定義了你能夠滾軸內容的寬度和高度,你也能夠添加能夠上下滾動的額外區域。第一種方法是你能夠經過添加內容的大小來完成。另一個比較動態的選擇是UIScrollView的另外一個屬性contentInset,contentInset增長你在contentsize中指定的內容可以滾動的上下左右區域數量contentInset.top以及contentInset.buttom分別表示上面和下面的距離。iphone
在滾軸視圖中,有一個叫作ContentOffset的屬性跟蹤UIScrollView的具體位置,你可以本身獲取和設置它,ContentOffset是你當前可視內容在滾軸視圖邊界的左上角那個點。如圖:動畫
能夠看出,ContentOffset內容中的那個點不是從contentInset的左上角開始的,而是內容的左上角,此時的ContentOffset是正值,但有時也是負值,以下圖所示:ui
使用一個ScrollView.net
建立一個UIScrollView代理
CGRectframe = CGRectMake( 0, 0, 200, 200);htm
scrollView= [[UIScrollView alloc] initWithFrame: frame];對象
添加子視圖(框架能夠超過scroll view的邊界)
frame= CGRectMake( 0, 0, 500, 500);
myImageView= [[UIImageView alloc] initWithFrame: frame];
[scrollViewaddSubview:myImageView];
設置內容尺寸
scrollView.contentSize= CGSize(500,500);
擴展Scroll View 的行爲
應用程序一般須要知道有關的滾圖的事件
scrolloffset改變的時候
拖動開始和結束
減速的開始和結束
經過子類化擴展Scroll View 的行爲
建立一個子類
重寫一些功能並改變行爲
關於這種方式的爭議
應用程序的邏輯和行爲變成了視圖自己的一部分,就像,你可能有一些定製的滾軸邏輯,,在那你只在乎一個視圖控制,但你想在不一樣地方重複使用你的滾軸視圖,若是你必須爲每一個都子類化,你最後會有不少不一樣的滾軸視圖子類以及在視圖中的特定應用邏輯。
編寫不少子類是很沉悶的事情,你最後會有不少沒法重複使用的單獨視圖,而MVC的視圖部分的一個重點是視圖是能夠在不一樣的控制器和不一樣的模式之中重複使用的,若是咱們把全部邏輯都放在視圖中,它減小了可複用性。
你的代碼變得很牢固地配對在一塊兒,它實際上變成了超類的一部分,你沒法從UIScrollView中析取它,以後用其它東西代替,若是它在你控制器中且爲控制器的一部分,在以後更容易改變它工做的方式和從新安排你應用程序的一些部分。
經過委派來擴展Scroll View 的行爲(經常使用的)
委派是一個單獨的對象,協議,定義了委派會實現的一系列功能的Objective-C協議,它建立了一系列很清晰的撤銷點,在那裏你能定製行爲和外觀。它在這些對象之間保持了鬆散的配對,視圖自己與視圖控制器或任何其它的控制器對象,委派不是滾軸視圖的直接子類,它比起牢固配對的子類更加的鬆散。
首先引用別人分析一篇不錯的文章,同時也感謝這位朋友的分享:http://www.cocoachina.com/iphonedev/sdk/2010/1224/2503.html
在滾動過程中,實際上是在修改原點座標。當手指觸摸後, scroll view會暫時攔截觸摸事件,使用一個計時器。假如在計時器到點後沒有發生手指移動事件,那麼 scroll view 發送 tracking events 到被點擊的 subview。假如在計時器到點前發生了移動事件,那麼 scroll view 取消 tracking 本身發生滾動。
子類能夠重載
touchesShouldBegin:withEvent:inContentView: 決定本身是否接收 touch 事件
pagingEnabled:當值是 YES 會自動滾動到 subview 的邊界,默認是NO
touchesShouldCancelInContentView: 開始發送 tracking messages 消息給 subview 的時候調用這個方法,決定是否發送 tracking messages 消息到subview。假如返回 NO,發送。YES 則不發送。
假如 canCancelContentTouches屬性是NO,則不調用這個方法來影響如何處理滾動手勢。
scroll view 還處理縮放和平移手勢,要實現縮放和平移,必須實現委託 viewForZoomingInScrollView:、scrollViewDidEndZooming:withView:atScale:
兩個方法。另外 maximumZoomScale和minimumZoomScale 兩個屬性要不同。
幾個屬性介紹
tracking
當 touch 後尚未拖動的時候值是YES,不然NO
zoomBouncing
當內容放大到最大或者最小的時候值是 YES,不然 NO
zooming
當正在縮放的時候值是 YES,不然 NO
decelerating
當滾動後,手指放開可是還在繼續滾動中。這個時候是 YES,其它時候是 NO
decelerationRate
設置手指放開後的減速率
maximumZoomScale
一個浮點數,表示能放最大的倍數
minimumZoomScale
一個浮點數,表示能縮最小的倍數
pagingEnabled
當值是 YES 會自動滾動到 subview 的邊界。默認是NO
scrollEnabled
決定是否能夠滾動
delaysContentTouches
是個布爾值,當值是 YES 的時候,用戶觸碰開始,scroll view要延遲一會,看看是否用戶有意圖滾動。假如滾動了,那麼捕捉 touch-down 事件,不然就不捕捉。假如值是NO,當用戶觸碰, scroll view 會當即觸發 touchesShouldBegin:withEvent:inContentView:,默認是 YES
canCancelContentTouches
當值是 YES 的時候,用戶觸碰後,而後在必定時間內沒有移動,scrollView 發送 tracking events,而後用戶移動手指足夠長度觸發滾動事件,這個時候,scrollView 發送了 touchesCancelled:withEvent: 到 subview,而後 scroView 開始滾動。假如值是 NO,scrollView 發送 tracking events 後,就算用戶移動手指,scrollView 也不會滾動。
contentSize
裏面內容的大小,也就是能夠滾動的大小,默認是0,沒有滾動效果。
showsHorizontalScrollIndicator
滾動時是否顯示水平滾動條
showsVerticalScrollIndicator
滾動時是否顯示垂直滾動條
bounces
默認是 yes,就是滾動超過邊界會反彈有反彈回來的效果。假如是 NO,那麼滾動到達邊界會馬上中止。
bouncesZoom
和 bounces 相似,區別在於:這個效果反映在縮放上面,假如縮放超過最大縮放,那麼會反彈效果;假如是 NO,則到達最大或者最小的時候當即中止。
directionalLockEnabled
默認是 NO,能夠在垂直和水平方向同時運動。當值是 YES 時,假如一開始是垂直或者是水平運動,那麼接下來會鎖定另一個方向的滾動。 假如一開始是對角方向滾動,則不會禁止某個方向
indicatorStyle
滾動條的樣式,基本只是設置顏色。總共3個顏色:默認、黑、白
scrollIndicatorInsets
設置滾動條的位置
最後加上,我剛定的一個例子,能夠縮放,及對使用的代理方法加了註釋
// ScrollViewViewController.m
// ScrollView
//
// Created by ch_soft on 11-11-8.
// Copyright 2011年 __MyCompanyName__. All rights reserved.
//
#import"ScrollViewViewController.h"
#import"CustomA.h"
#import"CustomB.h"
@implementation ScrollViewViewController
- (void)didReceiveMemoryWarning
{
// Releases the view if it doesn't have a superview.
[superdidReceiveMemoryWarning];
// Release any cached data, images, etc that aren't in use.
}
#pragma mark - View lifecycle
// Implement viewDidLoad to do additional setup after loading the view, typically from a nib.
- (void)viewDidLoad
{
scrollview =[[UIScrollViewalloc]initWithFrame:CGRectMake(50,0,200,400)];
scrollview.contentSize=CGSizeMake(400,400);
scrollview.backgroundColor=[UIColororangeColor];
scrollview.pagingEnabled=YES;//是否本身動適應
viewA=[[CustomAalloc]initWithFrame:CGRectMake(50,0,100,400)];
viewA.backgroundColor=[UIColorblueColor];
[scrollviewaddSubview:viewA];
[viewArelease];
CustomB * viewB=[[CustomB alloc] initWithFrame:CGRectMake(250,0,100,400)];
viewB.backgroundColor=[UIColoryellowColor];
[scrollviewaddSubview:viewB];
[viewB release];
[self.viewaddSubview:scrollview];
scrollview.maximumZoomScale=2.0;
scrollview.minimumZoomScale=0.5;
// scrollview.decelerationRate=1;
scrollview.delegate=self;
//canCancelContentTouches:YES-移動手指足夠長度觸發滾動事件,NO-scrollView發送 tracking events 後,就算用戶移動手指,scrollView也不會滾動。
scrollview.canCancelContentTouches=NO;
//當值是 YES的時候,用戶觸碰開始.要延遲一會,看看是否用戶有意圖滾動。假如滾動了,那麼捕捉 touch-down事件,不然就不捕捉。假如值是NO,當用戶觸碰, scroll view會當即觸發
scrollview.delaysContentTouches=YES;
[scrollviewrelease];
[superviewDidLoad];
}
#pragma mark UIScrollViewDelegate
//只要滾動了就會觸發
- (void)scrollViewDidScroll:(UIScrollView *)scrollView;
{
// NSLog(@" scrollViewDidScroll");
NSLog(@"ContentOffset x is %f,yis %f",scrollView.contentOffset.x,scrollView.contentOffset.y);
}
//開始拖拽視圖
- (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView;
{
NSLog(@"scrollViewWillBeginDragging");
}
//完成拖拽
- (void)scrollViewDidEndDragging:(UIScrollView *)scrollView willDecelerate:(BOOL)decelerate;
{
NSLog(@"scrollViewDidEndDragging");
}
//將開始降速時
- (void)scrollViewWillBeginDecelerating:(UIScrollView *)scrollView;
{
NSLog(@"scrollViewWillBeginDecelerating");
}
//減速中止了時執行,手觸摸時執行執行
- (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView;
{
NSLog(@"scrollViewDidEndDecelerating");
}
//滾動動畫中止時執行,代碼改變時出發,也就是setContentOffset改變時
- (void)scrollViewDidEndScrollingAnimation:(UIScrollView *)scrollView;
{
NSLog(@"scrollViewDidEndScrollingAnimation");
}
//設置放大縮小的視圖,要是uiscrollview的subview
- (UIView *)viewForZoomingInScrollView:(UIScrollView *)scrollView;
{
NSLog(@"viewForZoomingInScrollView");
return viewA;
}
//完成放大縮小時調用
- (void)scrollViewDidEndZooming:(UIScrollView *)scrollView withView:(UIView *)view atScale:(float)scale;
{
viewA.frame=CGRectMake(50,0,100,400);
NSLog(@"scale between minimum and maximum. called after any 'bounce' animations");
}// scale between minimum and maximum. called after any 'bounce' animations
//若是你不是徹底滾動到滾軸視圖的頂部,你能夠輕點狀態欄,那個可視的滾軸視圖會一直滾動到頂部,那是默認行爲,你能夠經過該方法返回NO來關閉它
- (BOOL)scrollViewShouldScrollToTop:(UIScrollView *)scrollView;
{
NSLog(@"scrollViewShouldScrollToTop");
returnYES;
}
- (void)scrollViewDidScrollToTop:(UIScrollView *)scrollView;
{
NSLog(@"scrollViewDidScrollToTop");
}
- (void)viewDidUnload
{
[super viewDidUnload];
// Release any retained subviews of the main view.
// e.g. self.myOutlet = nil;
}
- (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation
{
// Return YES for supported orientations
return (interfaceOrientation == UIInterfaceOrientationPortrait);
}
@end