You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

203 lines
7.5 KiB

  1. //
  2. // FLEXSystemLogTableViewController.m
  3. // FLEX
  4. //
  5. // Created by Ryan Olson on 1/19/15.
  6. // Copyright (c) 2015 f. All rights reserved.
  7. //
  8. #import "FLEXSystemLogTableViewController.h"
  9. #import "FLEXUtility.h"
  10. #import "FLEXColor.h"
  11. #import "FLEXASLLogController.h"
  12. #import "FLEXOSLogController.h"
  13. #import "FLEXSystemLogTableViewCell.h"
  14. @interface FLEXSystemLogTableViewController ()
  15. @property (nonatomic, readonly) id<FLEXLogController> logController;
  16. @property (nonatomic, readonly) NSMutableArray<FLEXSystemLogMessage *> *logMessages;
  17. @property (nonatomic, copy) NSArray<FLEXSystemLogMessage *> *filteredLogMessages;
  18. @end
  19. @implementation FLEXSystemLogTableViewController
  20. - (id)init {
  21. return [super initWithStyle:UITableViewStylePlain];
  22. }
  23. - (void)viewDidLoad
  24. {
  25. [super viewDidLoad];
  26. self.showsSearchBar = YES;
  27. __weak __typeof(self) weakSelf = self;
  28. id logHandler = ^(NSArray<FLEXSystemLogMessage *> *newMessages) {
  29. __strong __typeof(weakSelf) self = weakSelf;
  30. [self handleUpdateWithNewMessages:newMessages];
  31. };
  32. _logMessages = [NSMutableArray array];
  33. if (FLEXOSLogAvailable()) {
  34. _logController = [FLEXOSLogController withUpdateHandler:logHandler];
  35. } else {
  36. _logController = [FLEXASLLogController withUpdateHandler:logHandler];
  37. }
  38. [self.tableView registerClass:[FLEXSystemLogTableViewCell class] forCellReuseIdentifier:kFLEXSystemLogTableViewCellIdentifier];
  39. self.tableView.separatorStyle = UITableViewCellSeparatorStyleNone;
  40. self.title = @"Loading...";
  41. UIBarButtonItem *scrollDown = [[UIBarButtonItem alloc] initWithTitle:@" ⬇︎ "
  42. style:UIBarButtonItemStylePlain
  43. target:self
  44. action:@selector(scrollToLastRow)];
  45. UIBarButtonItem *settings = [[UIBarButtonItem alloc] initWithTitle:@"Settings"
  46. style:UIBarButtonItemStylePlain
  47. target:self
  48. action:@selector(showLogSettings)];
  49. if (FLEXOSLogAvailable()) {
  50. self.navigationItem.rightBarButtonItems = @[scrollDown, settings];
  51. } else {
  52. self.navigationItem.rightBarButtonItem = scrollDown;
  53. }
  54. }
  55. - (void)handleUpdateWithNewMessages:(NSArray<FLEXSystemLogMessage *> *)newMessages
  56. {
  57. self.title = @"System Log";
  58. [self.logMessages addObjectsFromArray:newMessages];
  59. // "Follow" the log as new messages stream in if we were previously near the bottom.
  60. BOOL wasNearBottom = self.tableView.contentOffset.y >= self.tableView.contentSize.height - self.tableView.frame.size.height - 100.0;
  61. [self.tableView reloadData];
  62. if (wasNearBottom) {
  63. [self scrollToLastRow];
  64. }
  65. }
  66. - (void)viewWillAppear:(BOOL)animated
  67. {
  68. [super viewWillAppear:animated];
  69. [self.logController startMonitoring];
  70. }
  71. - (void)scrollToLastRow
  72. {
  73. NSInteger numberOfRows = [self.tableView numberOfRowsInSection:0];
  74. if (numberOfRows > 0) {
  75. NSIndexPath *lastIndexPath = [NSIndexPath indexPathForRow:numberOfRows - 1 inSection:0];
  76. [self.tableView scrollToRowAtIndexPath:lastIndexPath atScrollPosition:UITableViewScrollPositionBottom animated:YES];
  77. }
  78. }
  79. - (void)showLogSettings
  80. {
  81. FLEXOSLogController *logController = (FLEXOSLogController *)self.logController;
  82. BOOL persistent = [[NSUserDefaults standardUserDefaults] boolForKey:kFLEXiOSPersistentOSLogKey];
  83. NSString *toggle = persistent ? @"Disable" : @"Enable";
  84. NSString *title = [@"Persistent logging: " stringByAppendingString:persistent ? @"ON" : @"OFF"];
  85. NSString *body = @"In iOS 10 and up, ASL is gone. The OS Log API is much more limited. "
  86. "To get as close to the old behavior as possible, logs must be collected manually at launch and stored.\n\n"
  87. "Turn this feature on only when you need it.";
  88. [FLEXAlert makeAlert:^(FLEXAlert *make) {
  89. make.title(title).message(body).button(toggle).handler(^(NSArray<NSString *> *strings) {
  90. [[NSUserDefaults standardUserDefaults] setBool:!persistent forKey:kFLEXiOSPersistentOSLogKey];
  91. logController.persistent = !persistent;
  92. [logController.messages addObjectsFromArray:self.logMessages];
  93. });
  94. make.button(@"Dismiss").cancelStyle();
  95. } showFrom:self];
  96. }
  97. #pragma mark - FLEXGlobalsEntry
  98. + (NSString *)globalsEntryTitle:(FLEXGlobalsRow)row {
  99. return @"⚠️ System Log";
  100. }
  101. + (UIViewController *)globalsEntryViewController:(FLEXGlobalsRow)row {
  102. return [self new];
  103. }
  104. #pragma mark - Table view data source
  105. - (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
  106. {
  107. return 1;
  108. }
  109. - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
  110. {
  111. return self.searchController.isActive ? self.filteredLogMessages.count : self.logMessages.count;
  112. }
  113. - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
  114. {
  115. FLEXSystemLogTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:kFLEXSystemLogTableViewCellIdentifier forIndexPath:indexPath];
  116. cell.logMessage = [self logMessageAtIndexPath:indexPath];
  117. cell.highlightedText = self.searchText;
  118. if (indexPath.row % 2 == 0) {
  119. cell.backgroundColor = [FLEXColor primaryBackgroundColor];
  120. } else {
  121. cell.backgroundColor = [FLEXColor secondaryBackgroundColor];
  122. }
  123. return cell;
  124. }
  125. - (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
  126. {
  127. FLEXSystemLogMessage *logMessage = [self logMessageAtIndexPath:indexPath];
  128. return [FLEXSystemLogTableViewCell preferredHeightForLogMessage:logMessage inWidth:self.tableView.bounds.size.width];
  129. }
  130. #pragma mark - Copy on long press
  131. - (BOOL)tableView:(UITableView *)tableView shouldShowMenuForRowAtIndexPath:(NSIndexPath *)indexPath
  132. {
  133. return YES;
  134. }
  135. - (BOOL)tableView:(UITableView *)tableView canPerformAction:(SEL)action forRowAtIndexPath:(NSIndexPath *)indexPath withSender:(id)sender
  136. {
  137. return action == @selector(copy:);
  138. }
  139. - (void)tableView:(UITableView *)tableView performAction:(SEL)action forRowAtIndexPath:(NSIndexPath *)indexPath withSender:(id)sender
  140. {
  141. if (action == @selector(copy:)) {
  142. // We usually only want to copy the log message itself, not any metadata associated with it.
  143. UIPasteboard.generalPasteboard.string = [self logMessageAtIndexPath:indexPath].messageText;
  144. }
  145. }
  146. - (FLEXSystemLogMessage *)logMessageAtIndexPath:(NSIndexPath *)indexPath
  147. {
  148. return self.searchController.isActive ? self.filteredLogMessages[indexPath.row] : self.logMessages[indexPath.row];
  149. }
  150. #pragma mark - Search bar
  151. - (void)updateSearchResults:(NSString *)searchString
  152. {
  153. [self onBackgroundQueue:^NSArray *{
  154. return [self.logMessages filteredArrayUsingPredicate:[NSPredicate predicateWithBlock:^BOOL(FLEXSystemLogMessage *logMessage, NSDictionary<NSString *, id> *bindings) {
  155. NSString *displayedText = [FLEXSystemLogTableViewCell displayedTextForLogMessage:logMessage];
  156. return [displayedText rangeOfString:searchString options:NSCaseInsensitiveSearch].length > 0;
  157. }]];
  158. } thenOnMainQueue:^(NSArray *filteredLogMessages) {
  159. if ([self.searchText isEqual:searchString]) {
  160. self.filteredLogMessages = filteredLogMessages;
  161. [self.tableView reloadData];
  162. }
  163. }];
  164. }
  165. @end