Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
PojavLauncherTeam
GitHub Repository: PojavLauncherTeam/PojavLauncher_iOS
Path: blob/main/Natives/LauncherNavigationController.m
589 views
1
#import <UniformTypeIdentifiers/UniformTypeIdentifiers.h>
2
#import "authenticator/BaseAuthenticator.h"
3
#import "AFNetworking.h"
4
#import "ALTServerConnection.h"
5
#import "CustomControlsViewController.h"
6
#import "DownloadProgressViewController.h"
7
#import "JavaGUIViewController.h"
8
#import "LauncherMenuViewController.h"
9
#import "LauncherNavigationController.h"
10
#import "LauncherPreferences.h"
11
#import "MinecraftResourceDownloadTask.h"
12
#import "MinecraftResourceUtils.h"
13
#import "PickTextField.h"
14
#import "PLPickerView.h"
15
#import "PLProfiles.h"
16
#import "UIKit+AFNetworking.h"
17
#import "UIKit+hook.h"
18
#import "ios_uikit_bridge.h"
19
#import "utils.h"
20
21
#include <sys/time.h>
22
23
#define AUTORESIZE_MASKS UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleLeftMargin | UIViewAutoresizingFlexibleRightMargin
24
25
static void *ProgressObserverContext = &ProgressObserverContext;
26
27
@interface LauncherNavigationController () <UIDocumentPickerDelegate, UIPickerViewDataSource, PLPickerViewDelegate, UIPopoverPresentationControllerDelegate> {
28
}
29
30
@property(nonatomic) MinecraftResourceDownloadTask* task;
31
@property(nonatomic) DownloadProgressViewController* progressVC;
32
@property(nonatomic) PLPickerView* versionPickerView;
33
@property(nonatomic) UITextField* versionTextField;
34
@property(nonatomic) int profileSelectedAt;
35
36
@end
37
38
@implementation LauncherNavigationController
39
40
- (void)viewDidLoad
41
{
42
[super viewDidLoad];
43
44
if ([self respondsToSelector:@selector(setNeedsUpdateOfScreenEdgesDeferringSystemGestures)]) {
45
[self setNeedsUpdateOfScreenEdgesDeferringSystemGestures];
46
}
47
48
self.versionTextField = [[PickTextField alloc] initWithFrame:CGRectMake(4, 4, self.toolbar.frame.size.width * 0.8 - 8, self.toolbar.frame.size.height - 8)];
49
[self.versionTextField addTarget:self.versionTextField action:@selector(resignFirstResponder) forControlEvents:UIControlEventEditingDidEndOnExit];
50
self.versionTextField.autoresizingMask = AUTORESIZE_MASKS;
51
self.versionTextField.placeholder = @"Specify version...";
52
self.versionTextField.leftView = [[UIImageView alloc] initWithFrame:CGRectMake(0, 0, 40, 40)];
53
self.versionTextField.rightView = [[UIImageView alloc] initWithImage:[[UIImage imageNamed:@"SpinnerArrow"] _imageWithSize:CGSizeMake(30, 30)]];
54
self.versionTextField.rightView.frame = CGRectMake(0, 0, self.versionTextField.frame.size.height * 0.9, self.versionTextField.frame.size.height * 0.9);
55
self.versionTextField.leftViewMode = UITextFieldViewModeAlways;
56
self.versionTextField.rightViewMode = UITextFieldViewModeAlways;
57
self.versionTextField.textAlignment = NSTextAlignmentCenter;
58
59
self.versionPickerView = [[PLPickerView alloc] init];
60
self.versionPickerView.delegate = self;
61
self.versionPickerView.dataSource = self;
62
UIToolbar *versionPickToolbar = [[UIToolbar alloc] initWithFrame:CGRectMake(0.0, 0.0, self.view.frame.size.width, 44.0)];
63
64
[self reloadProfileList];
65
66
UIBarButtonItem *versionFlexibleSpace = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemFlexibleSpace target:self action:nil];
67
UIBarButtonItem *versionDoneButton = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemDone target:self action:@selector(versionClosePicker)];
68
versionPickToolbar.items = @[versionFlexibleSpace, versionDoneButton];
69
self.versionTextField.inputAccessoryView = versionPickToolbar;
70
self.versionTextField.inputView = self.versionPickerView;
71
72
UIView *targetToolbar = self.toolbar;
73
[targetToolbar addSubview:self.versionTextField];
74
75
self.progressViewMain = [[UIProgressView alloc] initWithFrame:CGRectMake(0, 0, self.toolbar.frame.size.width, 4)];
76
self.progressViewMain.autoresizingMask = AUTORESIZE_MASKS;
77
self.progressViewMain.hidden = YES;
78
[targetToolbar addSubview:self.progressViewMain];
79
80
self.buttonInstall = [UIButton buttonWithType:UIButtonTypeSystem];
81
setButtonPointerInteraction(self.buttonInstall);
82
[self.buttonInstall setTitle:localize(@"Play", nil) forState:UIControlStateNormal];
83
self.buttonInstall.autoresizingMask = AUTORESIZE_MASKS;
84
self.buttonInstall.backgroundColor = [UIColor colorWithRed:54/255.0 green:176/255.0 blue:48/255.0 alpha:1.0];
85
self.buttonInstall.layer.cornerRadius = 5;
86
self.buttonInstall.frame = CGRectMake(self.toolbar.frame.size.width * 0.8, 4, self.toolbar.frame.size.width * 0.2, self.toolbar.frame.size.height - 8);
87
self.buttonInstall.tintColor = UIColor.whiteColor;
88
self.buttonInstall.enabled = NO;
89
[self.buttonInstall addTarget:self action:@selector(performInstallOrShowDetails:) forControlEvents:UIControlEventPrimaryActionTriggered];
90
[targetToolbar addSubview:self.buttonInstall];
91
92
self.progressText = [[UILabel alloc] initWithFrame:self.versionTextField.frame];
93
self.progressText.adjustsFontSizeToFitWidth = YES;
94
self.progressText.autoresizingMask = AUTORESIZE_MASKS;
95
self.progressText.font = [self.progressText.font fontWithSize:16];
96
self.progressText.textAlignment = NSTextAlignmentCenter;
97
self.progressText.userInteractionEnabled = NO;
98
[targetToolbar addSubview:self.progressText];
99
100
[self fetchRemoteVersionList];
101
[NSNotificationCenter.defaultCenter addObserver:self
102
selector:@selector(receiveNotification:)
103
name:@"InstallModpack"
104
object:nil];
105
106
if ([BaseAuthenticator.current isKindOfClass:MicrosoftAuthenticator.class]) {
107
// Perform token refreshment on startup
108
[self setInteractionEnabled:NO forDownloading:NO];
109
id callback = ^(NSString* status, BOOL success) {
110
self.progressText.text = status;
111
if (status == nil) {
112
[self setInteractionEnabled:YES forDownloading:NO];
113
} else if (!success) {
114
showDialog(localize(@"Error", nil), status);
115
}
116
};
117
[BaseAuthenticator.current refreshTokenWithCallback:callback];
118
}
119
}
120
121
- (BOOL)isVersionInstalled:(NSString *)versionId {
122
NSString *localPath = [NSString stringWithFormat:@"%s/versions/%@", getenv("POJAV_GAME_DIR"), versionId];
123
BOOL isDirectory;
124
[NSFileManager.defaultManager fileExistsAtPath:localPath isDirectory:&isDirectory];
125
return isDirectory;
126
}
127
128
- (void)fetchLocalVersionList {
129
if (!localVersionList) {
130
localVersionList = [NSMutableArray new];
131
}
132
[localVersionList removeAllObjects];
133
134
NSFileManager *fileManager = [NSFileManager defaultManager];
135
NSString *versionPath = [NSString stringWithFormat:@"%s/versions/", getenv("POJAV_GAME_DIR")];
136
NSArray *list = [fileManager contentsOfDirectoryAtPath:versionPath error:Nil];
137
for (NSString *versionId in list) {
138
if (![self isVersionInstalled:versionId]) continue;
139
[localVersionList addObject:@{
140
@"id": versionId,
141
@"type": @"custom"
142
}];
143
}
144
}
145
146
- (void)fetchRemoteVersionList {
147
self.buttonInstall.enabled = NO;
148
remoteVersionList = @[
149
@{@"id": @"latest-release", @"type": @"release"},
150
@{@"id": @"latest-snapshot", @"type": @"snapshot"}
151
].mutableCopy;
152
153
AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];
154
[manager GET:@"https://piston-meta.mojang.com/mc/game/version_manifest_v2.json" parameters:nil headers:nil progress:^(NSProgress * _Nonnull progress) {
155
self.progressViewMain.progress = progress.fractionCompleted;
156
} success:^(NSURLSessionTask *task, NSDictionary *responseObject) {
157
[remoteVersionList addObjectsFromArray:responseObject[@"versions"]];
158
NSDebugLog(@"[VersionList] Got %d versions", remoteVersionList.count);
159
setPrefObject(@"internal.latest_version", responseObject[@"latest"]);
160
self.buttonInstall.enabled = YES;
161
} failure:^(NSURLSessionTask *operation, NSError *error) {
162
NSDebugLog(@"[VersionList] Warning: Unable to fetch version list: %@", error.localizedDescription);
163
self.buttonInstall.enabled = YES;
164
}];
165
}
166
167
// Invoked by: startup, instance change event
168
- (void)reloadProfileList {
169
// Reload local version list
170
[self fetchLocalVersionList];
171
// Reload launcher_profiles.json
172
[PLProfiles updateCurrent];
173
[self.versionPickerView reloadAllComponents];
174
// Reload selected profile info
175
self.profileSelectedAt = [PLProfiles.current.profiles.allKeys indexOfObject:PLProfiles.current.selectedProfileName];
176
if (self.profileSelectedAt == -1) {
177
// This instance has no profiles?
178
return;
179
}
180
[self.versionPickerView selectRow:self.profileSelectedAt inComponent:0 animated:NO];
181
[self pickerView:self.versionPickerView didSelectRow:self.profileSelectedAt inComponent:0];
182
}
183
184
#pragma mark - Options
185
- (void)enterCustomControls {
186
CustomControlsViewController *vc = [[CustomControlsViewController alloc] init];
187
vc.modalPresentationStyle = UIModalPresentationOverFullScreen;
188
vc.setDefaultCtrl = ^(NSString *name){
189
setPrefObject(@"control.default_ctrl", name);
190
};
191
vc.getDefaultCtrl = ^{
192
return getPrefObject(@"control.default_ctrl");
193
};
194
[self presentViewController:vc animated:YES completion:nil];
195
}
196
197
- (void)enterModInstaller {
198
UIDocumentPickerViewController *documentPicker = [[UIDocumentPickerViewController alloc]
199
initForOpeningContentTypes:@[[UTType typeWithMIMEType:@"application/java-archive"]]
200
asCopy:YES];
201
documentPicker.delegate = self;
202
documentPicker.modalPresentationStyle = UIModalPresentationFormSheet;
203
[self presentViewController:documentPicker animated:YES completion:nil];
204
}
205
206
- (void)enterModInstallerWithPath:(NSString *)path hitEnterAfterWindowShown:(BOOL)hitEnter {
207
JavaGUIViewController *vc = [[JavaGUIViewController alloc] init];
208
vc.filepath = path;
209
vc.hitEnterAfterWindowShown = hitEnter;
210
if (!vc.requiredJavaVersion) {
211
return;
212
}
213
[self invokeAfterJITEnabled:^{
214
vc.modalPresentationStyle = UIModalPresentationFullScreen;
215
NSLog(@"[ModInstaller] launching %@", vc.filepath);
216
[self presentViewController:vc animated:YES completion:nil];
217
}];
218
}
219
220
- (void)documentPicker:(UIDocumentPickerViewController *)controller didPickDocumentAtURL:(NSURL *)url {
221
[self enterModInstallerWithPath:url.path hitEnterAfterWindowShown:NO];
222
}
223
224
- (void)setInteractionEnabled:(BOOL)enabled forDownloading:(BOOL)downloading {
225
for (UIControl *view in self.toolbar.subviews) {
226
if ([view isKindOfClass:UIControl.class]) {
227
view.alpha = enabled ? 1 : 0.2;
228
view.enabled = enabled;
229
}
230
}
231
self.progressViewMain.hidden = enabled;
232
self.progressText.text = nil;
233
if (downloading) {
234
[self.buttonInstall setTitle:localize(enabled ? @"Play" : @"Details", nil) forState:UIControlStateNormal];
235
self.buttonInstall.alpha = 1;
236
self.buttonInstall.enabled = YES;
237
}
238
UIApplication.sharedApplication.idleTimerDisabled = !enabled;
239
}
240
241
- (void)launchMinecraft:(UIButton *)sender {
242
if (!self.versionTextField.hasText) {
243
[self.versionTextField becomeFirstResponder];
244
return;
245
}
246
247
if (BaseAuthenticator.current == nil) {
248
// Present the account selector if none selected
249
UIViewController *view = [(UINavigationController *)self.splitViewController.viewControllers[0]
250
viewControllers][0];
251
[view performSelector:@selector(selectAccount:) withObject:sender];
252
return;
253
}
254
255
[self setInteractionEnabled:NO forDownloading:YES];
256
257
NSString *versionId = PLProfiles.current.profiles[self.versionTextField.text][@"lastVersionId"];
258
NSDictionary *object = [remoteVersionList filteredArrayUsingPredicate:[NSPredicate predicateWithFormat:@"(id == %@)", versionId]].firstObject;
259
if (!object) {
260
object = @{
261
@"id": versionId,
262
@"type": @"custom"
263
};
264
}
265
266
self.task = [MinecraftResourceDownloadTask new];
267
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
268
__weak LauncherNavigationController *weakSelf = self;
269
self.task.handleError = ^{
270
dispatch_async(dispatch_get_main_queue(), ^{
271
[weakSelf setInteractionEnabled:YES forDownloading:YES];
272
weakSelf.task = nil;
273
weakSelf.progressVC = nil;
274
});
275
};
276
[self.task downloadVersion:object];
277
dispatch_async(dispatch_get_main_queue(), ^{
278
self.progressViewMain.observedProgress = self.task.progress;
279
[self.task.progress addObserver:self
280
forKeyPath:@"fractionCompleted"
281
options:NSKeyValueObservingOptionInitial
282
context:ProgressObserverContext];
283
});
284
});
285
}
286
287
- (void)performInstallOrShowDetails:(UIButton *)sender {
288
if (self.task) {
289
if (!self.progressVC) {
290
self.progressVC = [[DownloadProgressViewController alloc] initWithTask:self.task];
291
}
292
UINavigationController *nav = [[UINavigationController alloc] initWithRootViewController:self.progressVC];
293
nav.modalPresentationStyle = UIModalPresentationPopover;
294
nav.popoverPresentationController.sourceView = sender;
295
[self presentViewController:nav animated:YES completion:nil];
296
} else {
297
[self launchMinecraft:sender];
298
}
299
}
300
301
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context {
302
if (context != ProgressObserverContext) {
303
[super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
304
return;
305
}
306
307
// Calculate download speed and ETA
308
static CGFloat lastMsTime;
309
static NSUInteger lastSecTime, lastCompletedUnitCount;
310
NSProgress *progress = self.task.textProgress;
311
struct timeval tv;
312
gettimeofday(&tv, NULL);
313
NSInteger completedUnitCount = self.task.progress.totalUnitCount * self.task.progress.fractionCompleted;
314
progress.completedUnitCount = completedUnitCount;
315
if (lastSecTime < tv.tv_sec) {
316
CGFloat currentTime = tv.tv_sec + tv.tv_usec / 1000000.0;
317
NSInteger throughput = (completedUnitCount - lastCompletedUnitCount) / (currentTime - lastMsTime);
318
progress.throughput = @(throughput);
319
progress.estimatedTimeRemaining = @((progress.totalUnitCount - completedUnitCount) / throughput);
320
lastCompletedUnitCount = completedUnitCount;
321
lastSecTime = tv.tv_sec;
322
lastMsTime = currentTime;
323
}
324
325
dispatch_async(dispatch_get_main_queue(), ^{
326
self.progressText.text = progress.localizedAdditionalDescription;
327
328
if (!progress.finished) return;
329
[self.progressVC dismissModalViewControllerAnimated:NO];
330
331
self.progressViewMain.observedProgress = nil;
332
if (self.task.metadata) {
333
[self invokeAfterJITEnabled:^{
334
UIKit_launchMinecraftSurfaceVC(self.view.window, self.task.metadata);
335
}];
336
} else {
337
self.task = nil;
338
[self setInteractionEnabled:YES forDownloading:YES];
339
[self reloadProfileList];
340
}
341
});
342
}
343
344
- (void)receiveNotification:(NSNotification *)notification {
345
if (![notification.name isEqualToString:@"InstallModpack"]) {
346
return;
347
}
348
[self setInteractionEnabled:NO forDownloading:YES];
349
self.task = [MinecraftResourceDownloadTask new];
350
NSDictionary *userInfo = notification.userInfo;
351
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
352
__weak LauncherNavigationController *weakSelf = self;
353
self.task.handleError = ^{
354
dispatch_async(dispatch_get_main_queue(), ^{
355
[weakSelf setInteractionEnabled:YES forDownloading:YES];
356
weakSelf.task = nil;
357
weakSelf.progressVC = nil;
358
});
359
};
360
[self.task downloadModpackFromAPI:notification.object detail:userInfo[@"detail"] atIndex:[userInfo[@"index"] unsignedLongValue]];
361
dispatch_async(dispatch_get_main_queue(), ^{
362
self.progressViewMain.observedProgress = self.task.progress;
363
[self.task.progress addObserver:self
364
forKeyPath:@"fractionCompleted"
365
options:NSKeyValueObservingOptionInitial
366
context:ProgressObserverContext];
367
});
368
});
369
}
370
371
- (void)invokeAfterJITEnabled:(void(^)(void))handler {
372
localVersionList = remoteVersionList = nil;
373
BOOL hasTrollStoreJIT = getEntitlementValue(@"com.apple.private.local.sandboxed-jit");
374
375
if (isJITEnabled(false)) {
376
[ALTServerManager.sharedManager stopDiscovering];
377
handler();
378
return;
379
} else if (hasTrollStoreJIT) {
380
NSURL *jitURL = [NSURL URLWithString:[NSString stringWithFormat:@"apple-magnifier://enable-jit?bundle-id=%@", NSBundle.mainBundle.bundleIdentifier]];
381
[UIApplication.sharedApplication openURL:jitURL options:@{} completionHandler:nil];
382
// Do not return, wait for TrollStore to enable JIT and jump back
383
} else if (getPrefBool(@"debug.debug_skip_wait_jit")) {
384
NSLog(@"Debug option skipped waiting for JIT. Java might not work.");
385
handler();
386
return;
387
}
388
389
self.progressText.text = localize(@"launcher.wait_jit.title", nil);
390
391
UIAlertController* alert = [UIAlertController alertControllerWithTitle:localize(@"launcher.wait_jit.title", nil)
392
message:hasTrollStoreJIT ? localize(@"launcher.wait_jit_trollstore.message", nil) : localize(@"launcher.wait_jit.message", nil)
393
preferredStyle:UIAlertControllerStyleAlert];
394
/* TODO:
395
UIAlertAction *cancel = [UIAlertAction actionWithTitle:localize(@"Cancel", nil) style:UIAlertActionStyleCancel handler:^{
396
397
}];
398
[alert addAction:cancel];
399
*/
400
[self presentViewController:alert animated:YES completion:nil];
401
402
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
403
while (!isJITEnabled(false)) {
404
// Perform check for every 200ms
405
usleep(1000*200);
406
}
407
dispatch_async(dispatch_get_main_queue(), ^{
408
[alert dismissViewControllerAnimated:YES completion:handler];
409
});
410
});
411
}
412
413
#pragma mark - UIPopoverPresentationControllerDelegate
414
- (UIModalPresentationStyle)adaptivePresentationStyleForPresentationController:(UIPresentationController *)controller traitCollection:(UITraitCollection *)traitCollection {
415
return UIModalPresentationNone;
416
}
417
418
#pragma mark - UIPickerView stuff
419
- (void)pickerView:(PLPickerView *)pickerView didSelectRow:(NSInteger)row inComponent:(NSInteger)component {
420
self.profileSelectedAt = row;
421
//((UIImageView *)self.versionTextField.leftView).image = [pickerView imageAtRow:row column:component];
422
((UIImageView *)self.versionTextField.leftView).image = [pickerView imageAtRow:row column:component];
423
self.versionTextField.text = [self pickerView:pickerView titleForRow:row forComponent:component];
424
PLProfiles.current.selectedProfileName = self.versionTextField.text;
425
}
426
427
- (NSInteger)numberOfComponentsInPickerView:(UIPickerView *)pickerView {
428
return 1;
429
}
430
431
- (NSInteger)pickerView:(UIPickerView *)pickerView numberOfRowsInComponent:(NSInteger)component {
432
return PLProfiles.current.profiles.count;
433
}
434
435
- (NSString *)pickerView:(UIPickerView *)pickerView titleForRow:(NSInteger)row forComponent:(NSInteger)component {
436
return PLProfiles.current.profiles.allValues[row][@"name"];
437
}
438
439
- (void)pickerView:(UIPickerView *)pickerView enumerateImageView:(UIImageView *)imageView forRow:(NSInteger)row forComponent:(NSInteger)component {
440
UIImage *fallbackImage = [[UIImage imageNamed:@"DefaultProfile"] _imageWithSize:CGSizeMake(40, 40)];
441
NSString *urlString = PLProfiles.current.profiles.allValues[row][@"icon"];
442
[imageView setImageWithURL:[NSURL URLWithString:urlString] placeholderImage:fallbackImage];
443
}
444
445
- (void)versionClosePicker {
446
[self.versionTextField endEditing:YES];
447
[self pickerView:self.versionPickerView didSelectRow:[self.versionPickerView selectedRowInComponent:0] inComponent:0];
448
}
449
450
#pragma mark - View controller UI mode
451
452
- (BOOL)prefersHomeIndicatorAutoHidden {
453
return YES;
454
}
455
456
- (void)viewDidLayoutSubviews {
457
[super viewDidLayoutSubviews];
458
[sidebarViewController updateAccountInfo];
459
}
460
461
@end
462
463