ish/app/RootsTableViewController.m
2020-06-19 09:13:20 -07:00

217 lines
8.5 KiB
Objective-C

//
// RootsTableViewController.m
// iSH
//
// Created by Theodore Dubois on 6/7/20.
//
#import "Roots.h"
#import "RootsTableViewController.h"
#import "ProgressReportViewController.h"
#import "UIApplication+OpenURL.h"
#import "UIViewController+Extras.h"
@interface RootsTableViewController ()
@end
@interface RootDetailViewController : UITableViewController <UIDocumentPickerDelegate>
@property (nonatomic) NSString *rootName;
@property (nonatomic) NSURL *exportURL;
@property (weak, nonatomic) IBOutlet UILabel *nameLabel;
@property (weak, nonatomic) IBOutlet UILabel *deleteLabel;
@property (weak, nonatomic) IBOutlet UITableViewCell *deleteCell;
@end
@implementation RootsTableViewController
- (void)awakeFromNib {
[super awakeFromNib];
[Roots.instance addObserver:self forKeyPath:@"roots" options:0 context:nil];
[Roots.instance addObserver:self forKeyPath:@"defaultRoot" options:0 context:nil];
NSLog(@"%@ hi", self);
}
- (void)dealloc {
NSLog(@"%@ bye", self);
[Roots.instance removeObserver:self forKeyPath:@"roots"];
[Roots.instance removeObserver:self forKeyPath:@"defaultRoot"];
}
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
[self.tableView reloadData];
}
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView {
return 1;
}
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
return Roots.instance.roots.count;
}
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
NSString *ident = @"Root";
if ([Roots.instance.roots[indexPath.row] isEqual:Roots.instance.defaultRoot])
ident = @"Default Root";
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ident forIndexPath:indexPath];
cell.textLabel.text = Roots.instance.roots[indexPath.row];
return cell;
}
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath {
[tableView deselectRowAtIndexPath:indexPath animated:YES];
}
- (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender {
RootDetailViewController *vc = segue.destinationViewController;
vc.rootName = Roots.instance.roots[self.tableView.indexPathForSelectedRow.row];
}
- (IBAction)importFilesystem:(id)sender {
UIDocumentPickerViewController *picker = [[UIDocumentPickerViewController alloc]
initWithDocumentTypes:@[@"public.tar-archive", @"org.gnu.gnu-zip-archive"]
inMode:UIDocumentPickerModeImport];
[self presentViewController:picker animated:YES completion:nil];
if (@available(iOS 13, *)) {
picker.shouldShowFileExtensions = YES;
}
picker.delegate = self;
}
- (void)documentPicker:(UIDocumentPickerViewController *)controller didPickDocumentsAtURLs:(NSArray<NSURL *> *)urls {
NSAssert(urls.count == 1, @"somehow picked multiple documents");
NSURL *url = urls[0];
NSString *fileName = url.lastPathComponent.stringByDeletingPathExtension;
if ([fileName hasSuffix:@".tar"])
fileName = fileName.stringByDeletingPathExtension;
unsigned i = 2;
NSString *name = fileName;
while ([Roots.instance.roots containsObject:name]) {
name = [NSString stringWithFormat:@"%@ %u", fileName, i++];
}
ProgressReportViewController *progressVC = [self.storyboard instantiateViewControllerWithIdentifier:@"progress"];
progressVC.title = [NSString stringWithFormat:@"Importing %@", name];
[self presentViewController:progressVC animated:YES completion:nil];
dispatch_async(dispatch_get_global_queue(QOS_CLASS_USER_INITIATED, 0), ^{
NSError *error;
BOOL success = [Roots.instance importRootFromArchive:urls[0] name:name error:&error progressReporter:progressVC];
dispatch_async(dispatch_get_main_queue(), ^{
[progressVC dismissViewControllerAnimated:YES completion:nil];
if (!success)
[self presentError:error title:@"Import failed"];
});
});
}
@end
@implementation RootDetailViewController
- (void)viewWillAppear:(BOOL)animated {
self.navigationItem.title = self.rootName;
self.nameLabel.text = self.rootName;
[self update];
}
- (void)update {
self.deleteLabel.enabled = !self.isDefaultRoot;
self.deleteCell.selectionStyle = !self.isDefaultRoot ? UITableViewCellSelectionStyleDefault : UITableViewCellSelectionStyleNone;
[self.tableView reloadData];
}
- (BOOL)isDefaultRoot {
return [self.rootName isEqualToString:Roots.instance.defaultRoot];
}
- (NSString *)tableView:(UITableView *)tableView titleForFooterInSection:(NSInteger)section {
if (section == 2) { // delete
if (self.isDefaultRoot)
return @"This filesystem can't be deleted because it's currently mounted as the root.";
}
return [super tableView:tableView titleForFooterInSection:section];
}
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath {
if (indexPath.section == 0 && indexPath.row == 1)
[self browseFiles];
if (indexPath.section == 0 && indexPath.row == 2)
[self exportFilesystem];
if (indexPath.section == 1 && indexPath.row == 0)
[self bootThis];
if (indexPath.section == 2 && indexPath.row == 0)
[self deleteFilesystem];
[tableView deselectRowAtIndexPath:indexPath animated:YES];
}
- (void)browseFiles {
NSURL *url = [NSFileProviderManager.defaultManager.documentStorageURL URLByAppendingPathComponent:self.rootName];
NSURLComponents *components = [NSURLComponents componentsWithURL:url resolvingAgainstBaseURL:NO];
components.scheme = @"shareddocuments";
[UIApplication openURL:components.string];
}
- (void)exportFilesystem {
self.exportURL = [NSFileManager.defaultManager.temporaryDirectory URLByAppendingPathComponent:[NSString stringWithFormat:@"%@.tar.gz", self.rootName]];
ProgressReportViewController *progressVC = [self.storyboard instantiateViewControllerWithIdentifier:@"progress"];
progressVC.title = [NSString stringWithFormat:@"Exporting %@", self.rootName];
[self presentViewController:progressVC animated:YES completion:nil];
// witness the callback hell
dispatch_async(dispatch_get_global_queue(QOS_CLASS_USER_INITIATED, 0), ^{
NSError *err;
BOOL success = [Roots.instance exportRootNamed:self.rootName toArchive:self.exportURL error:&err progressReporter:progressVC];
dispatch_async(dispatch_get_main_queue(), ^{
[progressVC dismissViewControllerAnimated:YES completion:nil];
if (!success) {
[self presentError:err title:@"Export failed"];
return;
}
UIDocumentPickerViewController *picker = [[UIDocumentPickerViewController alloc]
initWithURL:self.exportURL
inMode:UIDocumentPickerModeExportToService];
picker.delegate = self;
if (@available(iOS 13, *)) {
picker.shouldShowFileExtensions = YES;
}
[self presentViewController:picker animated:YES completion:nil];
});
});
}
- (void)setExportURL:(NSURL *)exportURL {
[NSFileManager.defaultManager removeItemAtURL:_exportURL error:nil];
_exportURL = exportURL;
}
- (void)bootThis {
Roots.instance.defaultRoot = self.rootName;
exit(0);
}
- (void)deleteFilesystem {
if (self.isDefaultRoot)
return;
UIAlertController *alert = [UIAlertController alertControllerWithTitle:@"Really delete?"
message:@"I can't be bothered to implement any undo or regret UI so this is irreversable."
preferredStyle:UIAlertControllerStyleAlert];
[alert addAction:[UIAlertAction actionWithTitle:@"Cancel" style:UIAlertActionStyleCancel handler:nil]];
[alert addAction:[UIAlertAction actionWithTitle:@"Delete" style:UIAlertActionStyleDestructive handler:^(UIAlertAction *action) {
NSError *error;
if (![Roots.instance destroyRootNamed:self.rootName error:&error]) {
[self presentError:error title:@"Delete failed"];
} else {
[self.navigationController popViewControllerAnimated:YES];
}
}]];
[self presentViewController:alert animated:YES completion:nil];
}
- (void)dealloc {
self.exportURL = nil; // get it deleted
}
@end