TestsTested | ✗ |
LangLanguage | Obj-CObjective C |
License | MIT |
ReleasedLast Release | Dec 2014 |
Maintained by Unclaimed.
Extension to UITableViewController in attempt to provide additional features that are reusable in most scenarios.
Configure the loading style, currently support header and footer.
- (void)viewDidLoad
{
[super viewDidLoad];
theTableViewController.loadingStyle = MwfTableViewLoadingStyleFooter; // default is MwfTableViewLoadingStyleHeader
}
Programmatically trigger loading animation
[theTableViewController setLoading:YES animated:YES];
Programmatically stop loading animation
[theTableViewController setLoading:YES animated:NO];
If you need to, you can override few methods to provide custom look and feel for your app.
- (UIView *)createLoadingView
{
// ... construct your custom loading view
return yourAwesomeCustomLoadingView;
}
- (void)willShowLoadingView:(UIView *)loadingView
{
// cast to your implementation
YourAwesomeLoadingView * view = (YourAwesomeLoadingView *)loadingView;
// ... do something about it, e.g. start animating the activity indicator view
}
- (void)didHideLoadingView:(UIView *)loadingView
{
// cast to your implementation
YourAwesomeLoadingView * view = (YourAwesomeLoadingView *)loadingView;
// ... do something about it, e.g. stop animating the activity indicator view
}
A class MwfTableData
is provided to manage your table backing store.
Instead of using NSArray
, this class provides all the convenience you need working with table view.
Creating Table Data
+ (MwfTableData *) createTableData;
+ (MwfTableData *) createTableDataWithSections;
Accessing data
- (NSUInteger) numberOfSections;
- (NSUInteger) numberOfRowsInSection:(NSUInteger)section;
- (NSUInteger) numberOfRows;
- (id) objectForSectionAtIndex:(NSUInteger)section;
- (id) objectForRowAtIndexPath:(NSIndexPath *)ip;
- (NSIndexPath *) indexPathForRow:(id)object;
- (NSUInteger) indexForSection:(id)sectionObject;
Inserting data
- (NSUInteger)addSection:(id)sectionObject;
- (NSUInteger)insertSection:(id)sectionObject atIndex:(NSUInteger)sectionIndex;
- (NSIndexPath *)addRow:(id)object inSection:(NSUInteger)sectionIndex;
- (NSIndexPath *)insertRow:(id)object atIndexPath:(NSIndexPath *)indexPath;
- (NSIndexPath *)addRow:(id)object;
- (NSIndexPath *)insertRow:(id)object atIndex:(NSUInteger)index;
Deleting data
- (NSIndexPath *)removeRowAtIndexPath:(NSIndexPath *)indexPath;
- (NSUInteger)removeSectionAtIndex:(NSUInteger)sectionIndex;
Updating data
- (NSUInteger)updateSection:(id)object atIndex:(NSUInteger)section;
- (NSIndexPath *)updateRow:(id)object atIndexPath:(NSIndexPath *)indexPath;
MwfTableData
in MwfTableViewController subclassesOverride createAndInitTableData
In this method, you can create MwfTableData
instance using one of the creation methods (with or without section) and initialize with some data.
- (MwfTableData *)createAndInitTableData;
{
MwfTableData * tableData = [MwfTableData createTableData];
[tableData addRow:@"Row 1"];
[tableData addRow:@"Row 2"];
[tableData addRow:@"Row 3"];
[tableData addRow:@"Load More"];
return tableData;
}
Performing bulk updates
MwfTableViewController
provides method performUpdates:(void(^)(MwfTableData *))updatesBlock
which you can call to perform bulk updates to your table view (add,remove,update sections/rows). This method will update the backing store as well as updating the table view (using UITableViewRowAnimationAutomatic
for row animation). This method frees you from tracking the changed index sets and paths, and lets you focus on the application logic that updates your table.
- (void)loadMoreData {
[self performUpdates:^(MwfTableData * data) {
// ... call the insert/delete/update method of MwfTableData
}];
}
Reload table view
Setting table data via tableData
property of MwfTableViewController
will trigger tableView's reloadData
.
- (void)reloadEntireTable { // a method in your MwfTableViewController subclass
MwfTableData * tableData = [MwfTableData createTableDataWithSections];
[tableData addSection:@"Section 1"];
[tableData addRow:@"Row 1" inSection:0];
[tableData addRow:@"Row 2" inSection:0];
self.tableData = tableData;
}
This feature aims to simplify the implementation of tableView:cellForRowAtIndexPath:
method.
If you have implemented table view with mix of multiple table view cell types, you will probably have a fat implementation of the tableView:cellForRowAtIndexPath:
method.
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)ip;
{
UITableViewCell * cell = nil;
id dataObject = ...; // get data object from backing store
NSString * reuseIdentifier = @"DefaultCell"; // default
Class cellClass = [UITableViewCell class]; // default
if ([dataObject isKindOf:[DataType1 class]]) {
reuseIdentifier = @"DataType1Cell";
cellClass = [DataType1Cell class];
}
else if ([dataObject isKindOf:[DataType2 class]]) {
reuseIdentifier = @"DataType2Cell";
cellClass = [DataType2Cell class];
}
... // and so on
cell = [tableView dequeueReusableCellWithIdentifier:reuseIdentifier];
if (!cell) {
cell = [[cellClass alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:reuseIdentifier];
}
// configure cell
if ([dataObject isKindOf:[DataType1 class]]) {
DataType1Cell * theCell = (DataType1Cell *)cell;
// configure using dataObject
}
else if ([dataObject isKindOf:[DataType2 class]]) {
DataType2Cell * theCell = (DataType2Cell *)cell;
// configure using dataObject
}
return cell;
}
The above example shows how bad it can become when the types of cell to display grow. It can become big nightmare to maintain.
Hence, MwfTableViewController
implemented a method tableView:cellForObject:atIndexPath:
trying to improve the big-fat-messy tableView:cellForRowAtIndexPath:
method you may have.
By default, MwfTableViewController
implements tableView:cellForRowAtIndexPath:
by calling MwfTableData
and the method.
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath;
{
id rowItem = [self.tableData objectForRowAtIndexPath:indexPath];
UITableViewCell * cell = [self tableView:tableView cellForObject:rowItem];
// to prevent app crashing when returning nil
if (!cell) {
cell = [self.tableView dequeueReusableCellWithIdentifier:@"NilCell"];
if (!cell) cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:@"NilCell"];
}
return cell;
}
If you don't like the default implementation (because you are not using MwfTableData
), you can override the method in your MwfTableViewController
subclass to suit your need, by still utilizing tableView:cellForObject:atIndexPath:
method.
So, how does tableView:cellForObject:atIndexPath:
work? In short, the design embraces coding-by-convention. It calls creation and configuration methods based on the name of the row data type.
Example: assuming you have your row data type Tweet
and Comment
, it will attempt to call the following methods for you:
- (UITableViewCell *)tableView:(UITableView *)tableView cellForTweetAtIndexPath:(NSIndexPath *)ip
// assume it returns TweetCell instance- (UITableViewCell *)tableView:(UITableView *)tableView cellForCommenAtIndexPatht:(NSIndexPath *)ip
// assume it returns CommentCell instance- (UITableViewCell *)tableView:(UITableView *)tableView configCell:(TweetCell *)cell forTweet:(Tweet *)tweet
- (UITableViewCell *)tableView:(UITableView *)tableView configCell:(CommentCell *)cell forTweet:(Comment *)comment
If you do not implement the creation and/or configuration methods for specific row type, but you implement the methods for its superclass, they will be called. For example, you have a class Shape and Box, where Box is subclass of Shape. In your codes, you implement the creation and configuration method for Shape, but not for Box:
- (UITableViewCell*)tableView:(UITableView*)tableView cellForShapeAtIndexPath:(NSIndexPath*)ip;
{
UITableViewCell * cell = ...; // dequeue and create the cell
return cell
}
- (void)tableView:(UITableView*)tableView configCell:(UITableViewCell*)cell forShape:(Shape*)shape;
{
// ... configure the cell
}
Whenever MwfTableViewController
encounter row with type Box, it will call the methods for Shape instead (because Shape is superclass of Box).
Note: if you want to further simplify your code by not writing those methods at all... Read on!
MwfTableItem
is a super simple class to represent a model for your row cell. It has 3 properties, i.e. cellClass
, reuseIdentifier
and userInfo
.
cellClass
, by convention it will return the item class name with suffix Cell
. e.g. MwfTableItem
and MwfTableItemCell
, Tweet
and TweetCell
.reuseIdentifier
, by convention it will return the cellClass
name.userInfo
, default to nil
. You can literally set anything to it (which you can use to further configure the cell).Benefits of using this:
MwfTableViewController
's tableView:cellForObject:atIndexPath:
(explained in previous section), you eliminate the need to implement the creation and configuration methods. MwfTableViewController
's tableView:cellForObject:atIndexPath:
and userInfo
is set, you can implement tableView:configCell:for<TableItemClassName>UserInfo:
method to configure the cell using data in userInfo
.Example:
// from demo project
@interface MwfDemoLoadingItem : MwfTableItem
@property (nonatomic,retain) NSString * loadingText;
@end
@interface MwfDemoLoadingItemCell : MwfTableItemCell
@property (nonatomic,retain) UIActivityIndicatorView * activityIndicatorView;
@end
// the implementation
@implementation MwfDemoLoadingItem
@synthesize loadingText = _loadingText;
@end
@implementation MwfDemoLoadingItemCell
@synthesize activityIndicatorView = _activityIndicatorView;
- (id)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier;
{
self = [super initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:reuseIdentifier];
if (self) {
_activityIndicatorView = [[UIActivityIndicatorView alloc] initWithActivityIndicatorStyle:UIActivityIndicatorViewStyleGray];
self.accessoryView = _activityIndicatorView;
}
return self;
}
- (void)setItem:(MwfDemoLoadingItem *)item;
{
[super setItem:item];
if (item.loadingText) self.textLabel.text = item.loadingText;
else self.textLabel.text = @"Loading...";
[_activityIndicatorView startAnimating];
}
@end
Example when using with tableView:cellForObject:atIndexPath:
and userInfo
is set.
// from demo project
// The following method will be automatically invoked if userInfo is set.
- (void)tableView:(UITableView *)tableView configCell:(MwfDemoLoadingItemCell *)cell forMwfDemoLoadingItemUserInfo:(NSString *)userInfo; // I know the userInfo is always NSString
{
cell.detailTextLabel.text = userInfo;
}
3 macros definitions are available to help writing the above callback methods easier (less keystroke and less typing error).
You can either choose ones that is prefixed with mwf_
or more concise $
sign.
Simply define CK_SHORTHAND
if you prefer to use the $
sign.
#define CK_SHORTHAND
#import "MwfTableViewController.h"
Examples:
// -(UITableViewCell*)tableView:(UITableView*)tableView cellForTweetAtIndexPath:(NSIndexPath*)indexPath
// can be written using macro as following
-$cellFor(Tweet)
{
// ... implementation
return cell;
}
// -(void)tableView:(UITableView*)tableView configCell:(TweetCell *)cell forTweet:(Tweet *)item
// can be written using macro as following
-$configCell(Tweet,TweetCell)
{
// ... implementation
}
// -(void)tableView:(UITableView*)tableView configCell:(AwesomeItemCell*)cell forAwesomeItemUserInfo:(NSString *)userInfo
// can be written using macro as following
-$configCellWithUserInfo(AwesomeItem,AwesomeItemCell,NSString)
{
// ... implementation
}
Search is a very common function that comes together with table view. For that, MwfTableViewController
provides some basic functionalities to easily implement search for your table view.
Enable search for your table view controller
To enable search, just set wantSearch
to YES. Additionally, implement createAndInitSearchResultsTableData
to customize initial data for your search results.
- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil;
{
self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
if (self) {
self.wantSearch = YES;
}
return self;
}
- (MwfTableData *)createAndInitSearchResultsTableData {
MwfTableData * td = ...; // create and perhaps initialize with some rows
return td;
}
Handle search query
When user types on the search bar or select search option, - (MwfTableData *)createSearchResultsTableDataForSearchText:(NSString *)searchText scope:(NSString *)scope;
will be invoked. You should implement this method to populate an instance of MwfTableData
that contains results of the search. By default, the invocation of this method is delayed by 0.4 second
. You can change this value by setting searchDelayInSeconds
property.
If you wish you perform search in background, you should return nil
from this method and dispatch the search to the background. Once search results are available, simply set searchResultsTableData
to display them.
Bonus
Similar to tableData
, searchResultsTableData
also comes with equivalence update method performUpdatesForSearchResults:(void(^)(MwfTableData *))updateBlock
.
Attention
When wantSearch
is YES
. Responding to UITableViewController
delegate or dataSource callbacks requires special attention. You should not call tableData
or searchResultsTableData
directly to avoid mistake in using the incorrect table data. Instead, a method - (MwfTableData *) tableDataForTableView:(UITableView *)tableView;
is available for you to use to use the correct table data for specified tableView. The implementation of this method is very simple actually:
- (MwfTableData *)tableDataForTableView:(UITableView *)tableView;
{
if (tableView == self.tableView) return _tableData;
return _searchResultsTableData;
}
If you remember this, it will save you tons of debugging time that could sometimes really hard to solve.
MwfTableViewController is licensed under MIT License Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.