VMDisplayMetalViewController.m 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. //
  2. // Copyright © 2019 osy. All rights reserved.
  3. //
  4. // Licensed under the Apache License, Version 2.0 (the "License");
  5. // you may not use this file except in compliance with the License.
  6. // You may obtain a copy of the License at
  7. //
  8. // http://www.apache.org/licenses/LICENSE-2.0
  9. //
  10. // Unless required by applicable law or agreed to in writing, software
  11. // distributed under the License is distributed on an "AS IS" BASIS,
  12. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. // See the License for the specific language governing permissions and
  14. // limitations under the License.
  15. //
  16. #import "VMDisplayMetalViewController.h"
  17. #import "VMDisplayMetalViewController+Keyboard.h"
  18. #import "VMDisplayMetalViewController+Touch.h"
  19. #import "VMDisplayMetalViewController+Pointer.h"
  20. #import "VMDisplayMetalViewController+Pencil.h"
  21. #import "VMDisplayMetalViewController+Gamepad.h"
  22. #import "VMKeyboardView.h"
  23. #import "UTMRenderer.h"
  24. #import "UTMVirtualMachine.h"
  25. #import "UTMQemuManager.h"
  26. #import "UTMConfiguration.h"
  27. #import "UTMConfiguration+Display.h"
  28. #import "UTMLogging.h"
  29. #import "CSDisplayMetal.h"
  30. #import "UTMScreenshot.h"
  31. #import "UTM-Swift.h"
  32. @implementation VMDisplayMetalViewController {
  33. UTMRenderer *_renderer;
  34. }
  35. @synthesize vmDisplay;
  36. @synthesize vmInput;
  37. - (BOOL)serverModeCursor {
  38. return self.vmInput.serverModeCursor;
  39. }
  40. - (void)viewDidLoad {
  41. [super viewDidLoad];
  42. // set up software keyboard
  43. self.keyboardView.inputAccessoryView = self.inputAccessoryView;
  44. // Set the view to use the default device
  45. self.mtkView.device = MTLCreateSystemDefaultDevice();
  46. if (!self.mtkView.device) {
  47. UTMLog(@"Metal is not supported on this device");
  48. return;
  49. }
  50. _renderer = [[UTMRenderer alloc] initWithMetalKitView:self.mtkView];
  51. if (!_renderer) {
  52. UTMLog(@"Renderer failed initialization");
  53. return;
  54. }
  55. // Initialize our renderer with the view size
  56. [_renderer mtkView:self.mtkView drawableSizeWillChange:self.mtkView.drawableSize];
  57. _renderer.source = self.vmDisplay;
  58. [_renderer changeUpscaler:self.vmConfiguration.displayUpscalerValue
  59. downscaler:self.vmConfiguration.displayDownscalerValue];
  60. self.mtkView.delegate = _renderer;
  61. [self initTouch];
  62. [self initGamepad];
  63. // Pointing device support on iPadOS 13.4 GM or later
  64. if (@available(iOS 13.4, *)) {
  65. // Betas of iPadOS 13.4 did not include this API, that's why I check if the class exists
  66. if (NSClassFromString(@"UIPointerInteraction") != nil) {
  67. [self initPointerInteraction];
  68. }
  69. }
  70. // Apple Pencil 2 double tap support on iOS 12.1+
  71. if (@available(iOS 12.1, *)) {
  72. [self initPencilInteraction];
  73. }
  74. }
  75. - (void)viewDidAppear:(BOOL)animated {
  76. [super viewDidAppear:animated];
  77. if (self.vm.state == kVMStopped || self.vm.state == kVMSuspended) {
  78. if ([self.vm startVM]) {
  79. self.vm.ioDelegate = self;
  80. }
  81. }
  82. }
  83. - (void)viewWillTransitionToSize:(CGSize)size withTransitionCoordinator:(id<UIViewControllerTransitionCoordinator>)coordinator {
  84. [super viewWillTransitionToSize:size withTransitionCoordinator:coordinator];
  85. [self displayResize:size];
  86. }
  87. - (void)virtualMachine:(UTMVirtualMachine *)vm transitionToState:(UTMVMState)state {
  88. [super virtualMachine:vm transitionToState:state];
  89. switch (state) {
  90. case kVMStopped:
  91. case kVMPaused:
  92. case kVMSuspended: {
  93. [UIView transitionWithView:self.view duration:0.5 options:UIViewAnimationOptionTransitionCrossDissolve animations:^{
  94. self.placeholderImageView.hidden = NO;
  95. self.placeholderImageView.image = self.vm.screenshot.image;
  96. self.mtkView.hidden = YES;
  97. } completion:nil];
  98. if (self.vmConfiguration.shareClipboardEnabled) {
  99. [[UTMPasteboard generalPasteboard] releasePollingModeForObject:self];
  100. }
  101. break;
  102. }
  103. case kVMStarted: {
  104. [UIView transitionWithView:self.view duration:0.5 options:UIViewAnimationOptionTransitionCrossDissolve animations:^{
  105. self.placeholderImageView.hidden = YES;
  106. self.mtkView.hidden = NO;
  107. } completion:nil];
  108. self->_renderer.source = self.vmDisplay;
  109. [self displayResize:self.view.bounds.size];
  110. if (self.vmConfiguration.shareClipboardEnabled) {
  111. [[UTMPasteboard generalPasteboard] requestPollingModeForObject:self];
  112. }
  113. break;
  114. }
  115. default: {
  116. break;
  117. }
  118. }
  119. }
  120. #pragma mark - Key handling
  121. - (void)setKeyboardVisible:(BOOL)keyboardVisible {
  122. if (keyboardVisible) {
  123. [self.keyboardView becomeFirstResponder];
  124. } else {
  125. [self.keyboardView resignFirstResponder];
  126. }
  127. [super setKeyboardVisible:keyboardVisible];
  128. }
  129. - (void)sendExtendedKey:(CSInputKey)type code:(int)code {
  130. if ((code & 0xFF00) == 0xE000) {
  131. code = 0x100 | (code & 0xFF);
  132. } else if (code >= 0x100) {
  133. UTMLog(@"warning: ignored invalid keycode 0x%x", code);
  134. }
  135. [self.vmInput sendKey:type code:code];
  136. }
  137. #pragma mark - Toolbar actions
  138. - (void)setLastDisplayChangeResize:(BOOL)lastDisplayChangeResize {
  139. _lastDisplayChangeResize = lastDisplayChangeResize;
  140. if (lastDisplayChangeResize) {
  141. [self.zoomButton setImage:[UIImage imageNamed:@"Toolbar Minimize"] forState:UIControlStateNormal];
  142. } else {
  143. [self.zoomButton setImage:[UIImage imageNamed:@"Toolbar Maximize"] forState:UIControlStateNormal];
  144. }
  145. }
  146. - (void)resizeDisplayToFit {
  147. CGSize viewSize = self.mtkView.drawableSize;
  148. CGSize displaySize = self.vmDisplay.displaySize;
  149. CGSize scaled = CGSizeMake(viewSize.width / displaySize.width, viewSize.height / displaySize.height);
  150. self.vmDisplay.viewportScale = MIN(scaled.width, scaled.height);
  151. self.vmDisplay.viewportOrigin = CGPointMake(0, 0);
  152. }
  153. - (void)resetDisplay {
  154. self.vmDisplay.viewportScale = 1.0;
  155. self.vmDisplay.viewportOrigin = CGPointMake(0, 0);
  156. }
  157. - (IBAction)changeDisplayZoom:(UIButton *)sender {
  158. if (self.lastDisplayChangeResize) {
  159. [self resetDisplay];
  160. } else {
  161. [self resizeDisplayToFit];
  162. }
  163. self.lastDisplayChangeResize = !self.lastDisplayChangeResize;
  164. }
  165. #pragma mark - Resizing
  166. - (void)displayResize:(CGSize)size {
  167. UTMLog(@"resizing to (%f, %f)", size.width, size.height);
  168. CGRect bounds = CGRectMake(0, 0, size.width, size.height);
  169. if (self.vmConfiguration.displayRetina) {
  170. CGFloat scale = [UIScreen mainScreen].scale;
  171. CGAffineTransform transform = CGAffineTransformMakeScale(scale, scale);
  172. bounds = CGRectApplyAffineTransform(bounds, transform);
  173. }
  174. [self.vmDisplay requestResolution:bounds];
  175. }
  176. @end