自定義iOS選擇器 附農歷選擇器代碼
1.鑒于模擬器和真機的滾動效果的差異,建議大家真機調試的時候注釋掉IDJScrollComponent.m里的如下內容:
- self.decelerationRate=0;
2.我的代碼中使用了iOS5.0的UIImage的縮放圖片的resizableImageWithCapInsets:方法,如果你使用的是iOS5.0以下的SDK,請替換為stretchableImageWithLeftCapWidth:topCapHeight:方法。
3.幾處BUG:
(1.)copy的我忘記釋放內存了。
- NSString *pYear=[cal.year copy];
- NSString *pMonth=[cal.month copy];
這個pYear, pMonth沒釋放!
(2.)UITableViewCell的釋放有些問題。
cellForRowAtIndexPath方法里面, cell的初始化有以下三種情況,
- cell=[[UITableViewCell alloc]initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier];
- cell = (UITableViewCell *)[tableView dequeueReusableCellWithIdentifier:CellIdentifier];
- cell=[[UITableViewCell alloc]initWithStyle:UITableViewCellStyleDefault reuseIdentifier:nil];
假設是調用這句 cell = (UITableViewCell *)[tableView dequeueReusableCellWithIdentifier:CellIdentifier]; 初始化
然后最后return [cell autorelease]; 就有問題了!
這樣改, 把另外2種初始化改成
- cell=[[[UITableViewCell alloc]initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier] autorelease];
- cell=[[[UITableViewCell alloc]initWithStyle:UITableViewCellStyleDefault reuseIdentifier:nil] autorelease];
然后最后return cell;就OK了。
(3.)算法的局限性。
IDJScrollComponent.m里的moveSubViews:方法由于算法欠妥,導致了一個局限性,就是滾輪圖片的高度必須是可以被顯示的行數整除的數字,譬如:我的例子日期顯示三行,那么滾輪的圖片高度就是177像素的,177恰好可以被3整除。
(4.)農歷數據提供器IDJChineseCalendar.mm的- (id)initWithYearStart:(NSUInteger)start end:(NSUInteger)end:方法:
- //獲取當年的公歷年份對應的農歷的年代和甲子年份
- std::pair p=ChineseCalendarDB::GetEraAndYearOfLunar([self.year intValue]);
需要移動到
- //設置當前時間的農歷日期字段
- self.year=[NSString stringWithFormat:@"%d", chineseDate.GetYear()];
的后面。
這個控件并不是對iOS自帶的UIPickerView貼圖,而是重新實現,我一周多的成果,期間走了不少彎路。哈!

此控件開發的原因:
最近有個需求,需要做一個農歷的日期滾輪選擇器,一開始想用iOS自帶的UIPickerView,但是設計人員要求的比較高,必須實現她要求的樣式,也就是上圖中大家看到的樣式。但是UIPickerView根本沒有提供可以定制樣式(也就是更換皮膚)的接口。首先我在論壇上找到了帖子,按照這位兄臺所述,確實可以在一定程度上修改UIPickerView的圖片,我之所以說一定程度上,有以下幾個原因:
(1.)他是用覆蓋原有的UIPickerView上的各個子視圖上圖片的方法做的,那么你提供新的圖片就必須與原圖片嚴絲合縫。這也說明,即便是圖片換了,樣式也是不能換的,譬如:分割線的粗細、上下左右的邊框的大小等,也就是不能完全DIY,不能滿足我的需求。
(2.)由于覆蓋UIPickerView上的各個子視圖上圖片的做法不是蘋果官方給出的,一旦以后某個版本的UIPickerView實現策略改變,例如:3號子視圖不再是滾輪的背景圖片,你的做法就掛了。
于是我就實現了上圖中的滾輪選擇器,它與UIPickerView的區別是:
(1.)皮膚圖片提供了方法可以做替換的。
(2.)可以選擇滾輪上的數據是否循環滾動。
(3.)可以指定選擇條的位置。

為了代碼的復用,也就是所謂的OOP,我按照以下的方式實現(因為我覺得這個滾輪選擇器的實現方案其實算不上亮點,因為有很多的方案可以選擇的,但是程序的結構、代碼的重用是很重要的):
(1.)pickerview目錄:這個目錄中的IDJPickerView.h是滾輪選擇器視圖,沒有任何和與數據相關的東西,也就是你可以在IDJPickerView上顯示任何數據。IDJPickerView通過委托獲取需要顯示的數據。這個目錄下的IDJScrollComponent.h實現了UIScrollView上的內容可以反復循環滾動的功能。
(2.)datepicker目錄:IDJDatePickerView.h實現了IDJPickerView里的協議,并把IDJPickerView做為自己的一個私有成員,實現了一個可以根據type字段顯示公歷、農歷的日期選擇器,IDJDatePickerView.m里的農歷算法用的是solar_chinese_calendar目錄中的C++算法,因此農歷的相關數據類使用了C++混編的代碼。其實iOS自己支持農歷的,但是存在BUG,具體的原因大家可以看我的IDJChineseCalendar.mm的源碼,里面的注釋比較詳細的。我在這里也使用了一層封裝,也就是IDJDatePickerView.m本身也不提供數據,而是通過IDJCalendar.h、IDJChineseCalendar.h、IDJCalendarUtil.h來提供,這樣既實現了數據與視圖的分離(因為農歷算法太復雜了,直接寫在IDJDatePickerView.m里會使得代碼看起來可讀性太差了),而且把C++的調用封裝在了數據層,使得IDJDatePickerView.m的視圖層代碼不會出現C++的API。
(3.)timepicker目錄:這個目錄純粹是一個Demo,展示了一個不循環滾動、顯示行數與選中條位置不對稱的選擇器,沒有什么實際的含義。
一開始我是純粹寫農歷的選擇器,代碼都是耦合在一起的,后來逐個拆開的,這樣代碼就可以復用了,不僅僅是農歷選擇器,而是可以承載任何數據。