Flutter Engine
 
Loading...
Searching...
No Matches
FlutterPlatformPlugin.mm
Go to the documentation of this file.
1// Copyright 2013 The Flutter Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
6
7#import <AudioToolbox/AudioToolbox.h>
8#import <Foundation/Foundation.h>
9#import <UIKit/UIApplication.h>
10#import <UIKit/UIKit.h>
11
12#include "flutter/fml/logging.h"
13#import "flutter/shell/platform/darwin/common/InternalFlutterSwiftCommon/InternalFlutterSwiftCommon.h"
19
21
22namespace {
23
24constexpr char kTextPlainFormat[] = "text/plain";
25// Some of the official iOS system sounds. A full list can be found in many places online, such as:
26// https://github.com/p-x9/swift-system-sound/blob/cb4327b223d55d01e9156539c8442db16f4b1f85/SystemSoundTable.md
27const UInt32 kKeyPressClickSoundId = 1306;
28const UInt32 kWheelsOfTimeSoundId = 1157;
29
30NSString* const kSearchURLPrefix = @"x-web-search://?";
31
32} // namespace
33
34namespace flutter {
35
36// TODO(abarth): Move these definitions from system_chrome_impl.cc to here.
38 "io.flutter.plugin.platform.SystemChromeOrientationNotificationName";
40 "io.flutter.plugin.platform.SystemChromeOrientationNotificationKey";
42 "io.flutter.plugin.platform.SystemChromeOverlayNotificationName";
44 "io.flutter.plugin.platform.SystemChromeOverlayNotificationKey";
45
46} // namespace flutter
47
48using namespace flutter;
49
51 UIApplication* flutterApplication = FlutterSharedApplication.application;
52 if (flutterApplication) {
53 flutterApplication.statusBarHidden = hidden;
54 } else {
55 [FlutterLogger logWarning:@"Application based status bar styling is not available in app "
56 "extension."];
57 }
58}
59
60static void SetStatusBarStyleForSharedApplication(UIStatusBarStyle style) {
61 UIApplication* flutterApplication = FlutterSharedApplication.application;
62 if (flutterApplication) {
63 // Note: -[UIApplication setStatusBarStyle] is deprecated in iOS9
64 // in favor of delegating to the view controller.
65 [flutterApplication setStatusBarStyle:style];
66 } else {
67 [FlutterLogger logWarning:@"Application based status bar styling is not available in app "
68 "extension."];
69 }
70}
71
73
74/**
75 * @brief Whether the status bar appearance is based on the style preferred for this ViewController.
76 *
77 * The default value is YES.
78 * Explicitly add `UIViewControllerBasedStatusBarAppearance` as `false` in
79 * info.plist makes this value to be false.
80 */
81@property(nonatomic, assign) BOOL enableViewControllerBasedStatusBarAppearance;
82@property(nonatomic, weak) FlutterEngine* engine;
83
84/**
85 * @brief Used to detect whether or not this device supports live text input from the camera.
86 */
87@property(nonatomic, strong) UITextField* textField;
88@end
89
90@implementation FlutterPlatformPlugin
91
92- (instancetype)initWithEngine:(FlutterEngine*)engine {
93 FML_DCHECK(engine) << "engine must be set";
94 self = [super init];
95
96 if (self) {
97 _engine = engine;
98 NSObject* infoValue = [[NSBundle mainBundle]
99 objectForInfoDictionaryKey:@"UIViewControllerBasedStatusBarAppearance"];
100#if FLUTTER_RUNTIME_MODE == FLUTTER_RUNTIME_MODE_DEBUG
101 if (infoValue != nil && ![infoValue isKindOfClass:[NSNumber class]]) {
102 [FlutterLogger logError:@"The value of UIViewControllerBasedStatusBarAppearance in "
103 "Info.plist must be a Boolean type."];
104 }
105#endif
106 _enableViewControllerBasedStatusBarAppearance =
107 (infoValue == nil || [(NSNumber*)infoValue boolValue]);
108 }
109
110 return self;
111}
112
113- (void)handleMethodCall:(FlutterMethodCall*)call result:(FlutterResult)result {
114 NSString* method = call.method;
115 id args = call.arguments;
116 if ([method isEqualToString:@"SystemSound.play"]) {
117 [self playSystemSound:args];
118 result(nil);
119 } else if ([method isEqualToString:@"HapticFeedback.vibrate"]) {
120 [self vibrateHapticFeedback:args];
121 result(nil);
122 } else if ([method isEqualToString:@"SystemChrome.setPreferredOrientations"]) {
123 [self setSystemChromePreferredOrientations:args];
124 result(nil);
125 } else if ([method isEqualToString:@"SystemChrome.setApplicationSwitcherDescription"]) {
126 [self setSystemChromeApplicationSwitcherDescription:args];
127 result(nil);
128 } else if ([method isEqualToString:@"SystemChrome.setEnabledSystemUIOverlays"]) {
129 [self setSystemChromeEnabledSystemUIOverlays:args];
130 result(nil);
131 } else if ([method isEqualToString:@"SystemChrome.setEnabledSystemUIMode"]) {
132 [self setSystemChromeEnabledSystemUIMode:args];
133 result(nil);
134 } else if ([method isEqualToString:@"SystemChrome.restoreSystemUIOverlays"]) {
135 [self restoreSystemChromeSystemUIOverlays];
136 result(nil);
137 } else if ([method isEqualToString:@"SystemChrome.setSystemUIOverlayStyle"]) {
138 [self setSystemChromeSystemUIOverlayStyle:args];
139 result(nil);
140 } else if ([method isEqualToString:@"SystemNavigator.pop"]) {
141 NSNumber* isAnimated = args;
142 [self popSystemNavigator:isAnimated.boolValue];
143 result(nil);
144 } else if ([method isEqualToString:@"Clipboard.getData"]) {
145 result([self getClipboardData:args]);
146 } else if ([method isEqualToString:@"Clipboard.setData"]) {
147 [self setClipboardData:args];
148 result(nil);
149 } else if ([method isEqualToString:@"Clipboard.hasStrings"]) {
150 result([self clipboardHasStrings]);
151 } else if ([method isEqualToString:@"LiveText.isLiveTextInputAvailable"]) {
152 result(@([self isLiveTextInputAvailable]));
153 } else if ([method isEqualToString:@"SearchWeb.invoke"]) {
154 [self searchWeb:args];
155 result(nil);
156 } else if ([method isEqualToString:@"LookUp.invoke"]) {
157 [self showLookUpViewController:args];
158 result(nil);
159 } else if ([method isEqualToString:@"Share.invoke"]) {
160 [self showShareViewController:args];
161 result(nil);
162 } else if ([method isEqualToString:@"ContextMenu.showSystemContextMenu"]) {
163 [self showSystemContextMenu:args];
164 result(nil);
165 } else if ([method isEqualToString:@"ContextMenu.hideSystemContextMenu"]) {
166 [self hideSystemContextMenu];
167 result(nil);
168 } else {
170 }
171}
172
173- (void)showSystemContextMenu:(NSDictionary*)args {
174 if (@available(iOS 16.0, *)) {
175 FlutterTextInputPlugin* textInputPlugin = [self.engine textInputPlugin];
176 BOOL shownEditMenu = [textInputPlugin showEditMenu:args];
177 if (!shownEditMenu) {
178 [FlutterLogger logError:@"Only text input supports system context menu for now. Ensure the "
179 "system context menu is shown with an active text input connection. "
180 "See https://github.com/flutter/flutter/issues/143033."];
181 }
182 }
183}
184
185- (void)hideSystemContextMenu {
186 if (@available(iOS 16.0, *)) {
187 FlutterTextInputPlugin* textInputPlugin = [self.engine textInputPlugin];
188 [textInputPlugin hideEditMenu];
189 }
190}
191
192- (void)showShareViewController:(NSString*)content {
193 UIViewController* engineViewController = [self.engine viewController];
194
195 NSArray* itemsToShare = @[ content ?: [NSNull null] ];
196 UIActivityViewController* activityViewController =
197 [[UIActivityViewController alloc] initWithActivityItems:itemsToShare
198 applicationActivities:nil];
199
200 if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad) {
201 // On iPad, the share screen is presented in a popover view, and requires a
202 // sourceView and sourceRect
203 FlutterTextInputPlugin* _textInputPlugin = [self.engine textInputPlugin];
204 UITextRange* range = _textInputPlugin.textInputView.selectedTextRange;
205
206 // firstRectForRange cannot be used here as it's current implementation does
207 // not always return the full rect of the range.
208 CGRect firstRect = [(FlutterTextInputView*)_textInputPlugin.textInputView
209 caretRectForPosition:(FlutterTextPosition*)range.start];
210 CGRect transformedFirstRect = [(FlutterTextInputView*)_textInputPlugin.textInputView
211 localRectFromFrameworkTransform:firstRect];
212 CGRect lastRect = [(FlutterTextInputView*)_textInputPlugin.textInputView
213 caretRectForPosition:(FlutterTextPosition*)range.end];
214 CGRect transformedLastRect = [(FlutterTextInputView*)_textInputPlugin.textInputView
215 localRectFromFrameworkTransform:lastRect];
216
217 activityViewController.popoverPresentationController.sourceView = engineViewController.view;
218 // In case of RTL Language, get the minimum x coordinate
219 activityViewController.popoverPresentationController.sourceRect =
220 CGRectMake(fmin(transformedFirstRect.origin.x, transformedLastRect.origin.x),
221 transformedFirstRect.origin.y,
222 abs(transformedLastRect.origin.x - transformedFirstRect.origin.x),
223 transformedFirstRect.size.height);
224 }
225
226 [engineViewController presentViewController:activityViewController animated:YES completion:nil];
227}
228
229- (void)searchWeb:(NSString*)searchTerm {
230 UIApplication* flutterApplication = FlutterSharedApplication.application;
231 if (flutterApplication == nil) {
232 [FlutterLogger logWarning:@"SearchWeb.invoke is not availabe in app extension."];
233 return;
234 }
235
236 NSString* escapedText = [searchTerm
237 stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet
238 URLHostAllowedCharacterSet]];
239 NSString* searchURL = [NSString stringWithFormat:@"%@%@", kSearchURLPrefix, escapedText];
240
241 [flutterApplication openURL:[NSURL URLWithString:searchURL] options:@{} completionHandler:nil];
242}
243
244- (void)playSystemSound:(NSString*)soundType {
245 if ([soundType isEqualToString:@"SystemSoundType.click"]) {
246 // All feedback types are specific to Android and are treated as equal on
247 // iOS.
248 AudioServicesPlaySystemSound(kKeyPressClickSoundId);
249 } else if ([soundType isEqualToString:@"SystemSoundType.tick"]) {
250 AudioServicesPlaySystemSound(kWheelsOfTimeSoundId);
251 }
252}
253
254- (void)vibrateHapticFeedback:(NSString*)feedbackType {
255 if (!feedbackType) {
256 AudioServicesPlaySystemSound(kSystemSoundID_Vibrate);
257 return;
258 }
259
260 if ([@"HapticFeedbackType.lightImpact" isEqualToString:feedbackType]) {
261 [[[UIImpactFeedbackGenerator alloc] initWithStyle:UIImpactFeedbackStyleLight] impactOccurred];
262 } else if ([@"HapticFeedbackType.mediumImpact" isEqualToString:feedbackType]) {
263 [[[UIImpactFeedbackGenerator alloc] initWithStyle:UIImpactFeedbackStyleMedium] impactOccurred];
264 } else if ([@"HapticFeedbackType.heavyImpact" isEqualToString:feedbackType]) {
265 [[[UIImpactFeedbackGenerator alloc] initWithStyle:UIImpactFeedbackStyleHeavy] impactOccurred];
266 } else if ([@"HapticFeedbackType.selectionClick" isEqualToString:feedbackType]) {
267 [[[UISelectionFeedbackGenerator alloc] init] selectionChanged];
268 }
269}
270
271- (void)setSystemChromePreferredOrientations:(NSArray*)orientations {
272 UIInterfaceOrientationMask mask = 0;
273
274 if (orientations.count == 0) {
275 mask |= UIInterfaceOrientationMaskAll;
276 } else {
277 for (NSString* orientation in orientations) {
278 if ([orientation isEqualToString:@"DeviceOrientation.portraitUp"]) {
279 mask |= UIInterfaceOrientationMaskPortrait;
280 } else if ([orientation isEqualToString:@"DeviceOrientation.portraitDown"]) {
281 mask |= UIInterfaceOrientationMaskPortraitUpsideDown;
282 } else if ([orientation isEqualToString:@"DeviceOrientation.landscapeLeft"]) {
283 mask |= UIInterfaceOrientationMaskLandscapeLeft;
284 } else if ([orientation isEqualToString:@"DeviceOrientation.landscapeRight"]) {
285 mask |= UIInterfaceOrientationMaskLandscapeRight;
286 }
287 }
288 }
289
290 if (!mask) {
291 return;
292 }
293 [[NSNotificationCenter defaultCenter]
294 postNotificationName:@(kOrientationUpdateNotificationName)
295 object:nil
296 userInfo:@{@(kOrientationUpdateNotificationKey) : @(mask)}];
297}
298
299- (void)setSystemChromeApplicationSwitcherDescription:(NSDictionary*)object {
300 // No counterpart on iOS but is a benign operation. So no asserts.
301}
302
303- (void)setSystemChromeEnabledSystemUIOverlays:(NSArray*)overlays {
304 BOOL statusBarShouldBeHidden = ![overlays containsObject:@"SystemUiOverlay.top"];
305 if ([overlays containsObject:@"SystemUiOverlay.bottom"]) {
306 [[NSNotificationCenter defaultCenter]
307 postNotificationName:FlutterViewControllerShowHomeIndicator
308 object:nil];
309 } else {
310 [[NSNotificationCenter defaultCenter]
311 postNotificationName:FlutterViewControllerHideHomeIndicator
312 object:nil];
313 }
314 if (self.enableViewControllerBasedStatusBarAppearance) {
315 [self.engine viewController].prefersStatusBarHidden = statusBarShouldBeHidden;
316 } else {
317 // Checks if the top status bar should be visible. This platform ignores all
318 // other overlays
319
320 // We opt out of view controller based status bar visibility since we want
321 // to be able to modify this on the fly. The key used is
322 // UIViewControllerBasedStatusBarAppearance.
323 SetStatusBarHiddenForSharedApplication(statusBarShouldBeHidden);
324 }
325}
326
327- (void)setSystemChromeEnabledSystemUIMode:(NSString*)mode {
328 BOOL edgeToEdge = [mode isEqualToString:@"SystemUiMode.edgeToEdge"];
329 if (self.enableViewControllerBasedStatusBarAppearance) {
330 [self.engine viewController].prefersStatusBarHidden = !edgeToEdge;
331 } else {
332 // Checks if the top status bar should be visible, reflected by edge to edge setting. This
333 // platform ignores all other system ui modes.
334
335 // We opt out of view controller based status bar visibility since we want
336 // to be able to modify this on the fly. The key used is
337 // UIViewControllerBasedStatusBarAppearance.
339 }
340 [[NSNotificationCenter defaultCenter]
341 postNotificationName:edgeToEdge ? FlutterViewControllerShowHomeIndicator
342 : FlutterViewControllerHideHomeIndicator
343 object:nil];
344}
345
346- (void)restoreSystemChromeSystemUIOverlays {
347 // Nothing to do on iOS.
348}
349
350- (void)setSystemChromeSystemUIOverlayStyle:(NSDictionary*)message {
351 NSString* brightness = message[@"statusBarBrightness"];
352 if (brightness == (id)[NSNull null]) {
353 return;
354 }
355
356 UIStatusBarStyle statusBarStyle;
357 if ([brightness isEqualToString:@"Brightness.dark"]) {
358 statusBarStyle = UIStatusBarStyleLightContent;
359 } else if ([brightness isEqualToString:@"Brightness.light"]) {
360 statusBarStyle = UIStatusBarStyleDarkContent;
361 } else {
362 return;
363 }
364
365 if (self.enableViewControllerBasedStatusBarAppearance) {
366 // This notification is respected by the iOS embedder.
367 [[NSNotificationCenter defaultCenter]
368 postNotificationName:@(kOverlayStyleUpdateNotificationName)
369 object:nil
370 userInfo:@{@(kOverlayStyleUpdateNotificationKey) : @(statusBarStyle)}];
371 } else {
373 }
374}
375
376- (void)popSystemNavigator:(BOOL)isAnimated {
377 // Apple's human user guidelines say not to terminate iOS applications. However, if the
378 // root view of the app is a navigation controller, it is instructed to back up a level
379 // in the navigation hierarchy.
380 // It's also possible in an Add2App scenario that the FlutterViewController was presented
381 // outside the context of a UINavigationController, and still wants to be popped.
382
383 FlutterViewController* engineViewController = [self.engine viewController];
384 UINavigationController* navigationController = [engineViewController navigationController];
385 if (navigationController) {
386 [navigationController popViewControllerAnimated:isAnimated];
387 } else {
388 UIViewController* rootViewController = nil;
389 UIApplication* flutterApplication = FlutterSharedApplication.application;
390 if (flutterApplication) {
391 rootViewController = flutterApplication.keyWindow.rootViewController;
392 } else {
393 if (@available(iOS 15.0, *)) {
394 rootViewController =
395 [engineViewController flutterWindowSceneIfViewLoaded].keyWindow.rootViewController;
396 } else {
397 [FlutterLogger logWarning:@"rootViewController is not available in application extension "
398 "prior to iOS 15.0."];
399 }
400 }
401
402 if (engineViewController != rootViewController) {
403 [engineViewController dismissViewControllerAnimated:isAnimated completion:nil];
404 }
405 }
406}
407
408- (NSDictionary*)getClipboardData:(NSString*)format {
409 UIPasteboard* pasteboard = [UIPasteboard generalPasteboard];
410 if (!format || [format isEqualToString:@(kTextPlainFormat)]) {
411 NSString* stringInPasteboard = pasteboard.string;
412 // The pasteboard may contain an item but it may not be a string (an image for instance).
413 return stringInPasteboard == nil ? nil : @{@"text" : stringInPasteboard};
414 }
415 return nil;
416}
417
418- (void)setClipboardData:(NSDictionary*)data {
419 UIPasteboard* pasteboard = [UIPasteboard generalPasteboard];
420 id copyText = data[@"text"];
421 if ([copyText isKindOfClass:[NSString class]]) {
422 pasteboard.string = copyText;
423 } else {
424 pasteboard.string = @"null";
425 }
426}
427
428- (NSDictionary*)clipboardHasStrings {
429 return @{@"value" : @([UIPasteboard generalPasteboard].hasStrings)};
430}
431
432- (BOOL)isLiveTextInputAvailable {
433 return [[self textField] canPerformAction:@selector(captureTextFromCamera:) withSender:nil];
434}
435
436- (void)showLookUpViewController:(NSString*)term {
437 UIViewController* engineViewController = [self.engine viewController];
438 UIReferenceLibraryViewController* referenceLibraryViewController =
439 [[UIReferenceLibraryViewController alloc] initWithTerm:term];
440 [engineViewController presentViewController:referenceLibraryViewController
441 animated:YES
442 completion:nil];
443}
444
445- (UITextField*)textField {
446 if (_textField == nil) {
447 _textField = [[UITextField alloc] init];
448 }
449 return _textField;
450}
451
452@end
void(^ FlutterResult)(id _Nullable result)
FLUTTER_DARWIN_EXPORT NSObject const * FlutterMethodNotImplemented
static void SetStatusBarStyleForSharedApplication(UIStatusBarStyle style)
static void SetStatusBarHiddenForSharedApplication(BOOL hidden)
FlutterEngine engine
Definition main.cc:84
G_BEGIN_DECLS G_MODULE_EXPORT FlValue * args
G_BEGIN_DECLS GBytes * message
#define FML_DCHECK(condition)
Definition logging.h:122
UIView< UITextInput > * textInputView()
BOOL showEditMenu:(ios(16.0) API_AVAILABLE)
FlutterTextInputPlugin * textInputPlugin
FlutterTextInputPlugin * _textInputPlugin
constexpr char kTextPlainFormat[]
Clipboard plain text format.
NSString *const kSearchURLPrefix
const char *const kOrientationUpdateNotificationKey
const char *const kOverlayStyleUpdateNotificationName
const char *const kOverlayStyleUpdateNotificationKey
const char *const kOrientationUpdateNotificationName
std::shared_ptr< const fml::Mapping > data
int BOOL