| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209 |
- //
- // FLEXLiveObjectsTableViewController.m
- // Flipboard
- //
- // Created by Ryan Olson on 5/28/14.
- // Copyright (c) 2014 Flipboard. All rights reserved.
- //
- #import "FLEXLiveObjectsTableViewController.h"
- #import "FLEXHeapEnumerator.h"
- #import "FLEXInstancesTableViewController.h"
- #import "FLEXUtility.h"
- #import <objc/runtime.h>
- static const NSInteger kFLEXLiveObjectsSortAlphabeticallyIndex = 0;
- static const NSInteger kFLEXLiveObjectsSortByCountIndex = 1;
- @interface FLEXLiveObjectsTableViewController () <UISearchBarDelegate>
- @property (nonatomic, strong) NSDictionary *instanceCountsForClassNames;
- @property (nonatomic, readonly) NSArray *allClassNames;
- @property (nonatomic, strong) NSArray *filteredClassNames;
- @property (nonatomic, strong) UISearchBar *searchBar;
- @end
- @implementation FLEXLiveObjectsTableViewController
- - (void)viewDidLoad
- {
- [super viewDidLoad];
-
- self.searchBar = [[UISearchBar alloc] init];
- self.searchBar.placeholder = [FLEXUtility searchBarPlaceholderText];
- self.searchBar.delegate = self;
- self.searchBar.showsScopeBar = YES;
- self.searchBar.scopeButtonTitles = @[@"Sort Alphabetically", @"Sort by Count"];
- [self.searchBar sizeToFit];
- self.tableView.tableHeaderView = self.searchBar;
-
- self.refreshControl = [[UIRefreshControl alloc] init];
- [self.refreshControl addTarget:self action:@selector(refreshControlDidRefresh:) forControlEvents:UIControlEventValueChanged];
-
- [self reloadTableData];
- }
- - (NSArray *)allClassNames
- {
- return [self.instanceCountsForClassNames allKeys];
- }
- - (void)reloadTableData
- {
- // Set up a CFMutableDictionary with class pointer keys and NSUInteger values.
- // We abuse CFMutableDictionary a little to have primitive keys through judicious casting, but it gets the job done.
- // The dictionary is intialized with a 0 count for each class so that it doesn't have to expand during enumeration.
- // While it might be a little cleaner to populate an NSMutableDictionary with class name string keys to NSNumber counts,
- // we choose the CF/primitives approach because it lets us enumerate the objects in the heap without allocating any memory during enumeration.
- // The alternative of creating one NSString/NSNumber per object on the heap ends up polluting the count of live objects quite a bit.
- unsigned int classCount = 0;
- Class *classes = objc_copyClassList(&classCount);
- CFMutableDictionaryRef mutableCountsForClasses = CFDictionaryCreateMutable(NULL, classCount, NULL, NULL);
- for (unsigned int i = 0; i < classCount; i++) {
- CFDictionarySetValue(mutableCountsForClasses, (__bridge const void *)classes[i], (const void *)0);
- }
-
- // Enumerate all objects on the heap to build the counts of instances for each class.
- [FLEXHeapEnumerator enumerateLiveObjectsUsingBlock:^(__unsafe_unretained id object, __unsafe_unretained Class actualClass) {
- NSUInteger instanceCount = (NSUInteger)CFDictionaryGetValue(mutableCountsForClasses, (__bridge const void *)actualClass);
- instanceCount++;
- CFDictionarySetValue(mutableCountsForClasses, (__bridge const void *)actualClass, (const void *)instanceCount);
- }];
-
- // Convert our CF primitive dictionary into a nicer mapping of class name strings to counts that we will use as the table's model.
- NSMutableDictionary *mutableCountsForClassNames = [NSMutableDictionary dictionary];
- for (unsigned int i = 0; i < classCount; i++) {
- Class class = classes[i];
- NSUInteger instanceCount = (NSUInteger)CFDictionaryGetValue(mutableCountsForClasses, (__bridge const void *)(class));
- if (instanceCount > 0) {
- NSString *className = @(class_getName(class));
- [mutableCountsForClassNames setObject:@(instanceCount) forKey:className];
- }
- }
- free(classes);
-
- self.instanceCountsForClassNames = mutableCountsForClassNames;
-
- [self updateTableDataForSearchFilter];
- }
- - (void)refreshControlDidRefresh:(id)sender
- {
- [self reloadTableData];
- [self.refreshControl endRefreshing];
- }
- - (void)updateTitle
- {
- NSString *title = @"Live Objects";
-
- NSUInteger totalCount = 0;
- for (NSString *className in self.allClassNames) {
- totalCount += [self.instanceCountsForClassNames[className] unsignedIntegerValue];
- }
- NSUInteger filteredCount = 0;
- for (NSString *className in self.filteredClassNames) {
- filteredCount += [self.instanceCountsForClassNames[className] unsignedIntegerValue];
- }
-
- if (filteredCount == totalCount) {
- // Unfiltered
- title = [title stringByAppendingFormat:@" (%lu)", (unsigned long)totalCount];
- } else {
- title = [title stringByAppendingFormat:@" (filtered, %lu)", (unsigned long)filteredCount];
- }
-
- self.title = title;
- }
- #pragma mark - Search
- - (void)searchBar:(UISearchBar *)searchBar textDidChange:(NSString *)searchText
- {
- [self updateTableDataForSearchFilter];
- }
- - (void)searchBarSearchButtonClicked:(UISearchBar *)searchBar
- {
- [searchBar resignFirstResponder];
- }
- - (void)searchBar:(UISearchBar *)searchBar selectedScopeButtonIndexDidChange:(NSInteger)selectedScope
- {
- [self updateTableDataForSearchFilter];
- }
- - (void)scrollViewDidScroll:(UIScrollView *)scrollView
- {
- // Dismiss the keyboard when interacting with filtered results.
- [self.searchBar endEditing:YES];
- }
- - (void)updateTableDataForSearchFilter
- {
- if ([self.searchBar.text length] > 0) {
- NSPredicate *searchPreidcate = [NSPredicate predicateWithFormat:@"SELF CONTAINS[cd] %@", self.searchBar.text];
- self.filteredClassNames = [self.allClassNames filteredArrayUsingPredicate:searchPreidcate];
- } else {
- self.filteredClassNames = self.allClassNames;
- }
-
- if (self.searchBar.selectedScopeButtonIndex == kFLEXLiveObjectsSortAlphabeticallyIndex) {
- self.filteredClassNames = [self.filteredClassNames sortedArrayUsingSelector:@selector(caseInsensitiveCompare:)];
- } else if (self.searchBar.selectedScopeButtonIndex == kFLEXLiveObjectsSortByCountIndex) {
- self.filteredClassNames = [self.filteredClassNames sortedArrayUsingComparator:^NSComparisonResult(NSString *className1, NSString *className2) {
- NSNumber *count1 = self.instanceCountsForClassNames[className1];
- NSNumber *count2 = self.instanceCountsForClassNames[className2];
- // Reversed for descending counts.
- return [count2 compare:count1];
- }];
- }
-
- [self updateTitle];
- [self.tableView reloadData];
- }
- #pragma mark - Table view data source
- - (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
- {
- return 1;
- }
- - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
- {
- return [self.filteredClassNames count];
- }
- - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
- {
- static NSString *CellIdentifier = @"Cell";
- UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
- if (!cell) {
- cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier];
- cell.accessoryType = UITableViewCellAccessoryDisclosureIndicator;
- cell.textLabel.font = [FLEXUtility defaultTableViewCellLabelFont];
- }
-
- NSString *className = self.filteredClassNames[indexPath.row];
- NSNumber *count = self.instanceCountsForClassNames[className];
- cell.textLabel.text = [NSString stringWithFormat:@"%@ (%ld)", className, (long)[count integerValue]];
-
- return cell;
- }
- #pragma mark - Table view delegate
- - (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
- {
- NSString *className = self.filteredClassNames[indexPath.row];
- FLEXInstancesTableViewController *instancesViewController = [FLEXInstancesTableViewController instancesTableViewControllerForClassName:className];
- [self.navigationController pushViewController:instancesViewController animated:YES];
- }
- @end
|