Welcome to virtual_platform, a package that simplifies the cross-platform development process by providing a virtual platform than can be changed at runtime, together with declarative virtual platform utils.

NB: If you are using also physical_platform, you can opt to only use omni_platform instead. omni_platform integrates virtual_platform and physical_platform, without adding any extra functionality. This way you only need to declare one package in your pubspec.yaml.

Warning: use this package only when targeting a single design language

This package assumes that you use only one design language (e.g. only MaterialApp is defined in your widget tree), however, some parts of the code might require better-fitting widgets (e.g., the date picker and the bottom sheets should look like proper iOS ones if the virtual platform is iOS).

On the other hand, if your intent is to use multiple design languages (e.g. MaterialApp should be used in Android devices, CupertinoApp on iOS devices, etc.), then you should use another package: design_language. A virtual platform can still be used in combination with design_language if the package design_language does not provide all necessary utils.

Motivation

Standard Flutter does not offer a straightforward approach for rendering platform-specific widgets on a different platform. For example, let's say that the widget tree is mainly composed of Material widgets, however the date picker and the bottom sheets should look like proper iOS ones. One would usually need an Android smartphone and an iPhone to test this.

However, what if you don't have an iOS device? Or what if constantly switching between physical devices ends up being very time consuming and/or annoying? Wouldn't you prefer to to test all these different layouts directly from your main development station? Since all platforms can render widgets from any design language — e.g., Cupertino widgets can be rendered on any platform — why not take full advantage of that?

This package was made to cover all the scenarios above: by introducing a virtual platform, which can be changed at runtime, we can force all the virtual-platform-specific widgets to be rebuilt when needed.

This package — virtual_platform — does not come with any prebuilt, ready-to-use widgets: it is up to the developer to determine the exact widget, per virtual platform.

Getting started

You need to add virtual_platform to your dependencies.

dependencies:
  virtual_platform: ^latest # replace latest with version number

Next, you have to import package:virtual_platform/virtual_platform.dart.

Usage

The arguments of all instruments are functions, i.e., VirtualPlatformDispatcher and matchVirtualPlatform are of type Widget Function(BuildContext context, Widget? child) and T Function(), respectively. If the goal is to only select a widget or a value — a function expression is what needs to be passed, e.g., (_, __) => Text('text') or () => 'value', respectively.

Virtual platforms and virtual platform groups are summarized at the end of this README.

VirtualPlatform and its notifier

You need to instantiate VirtualPlatform.notifier before you use VirtualPlatformDispatcher or anything that relies on the virtual platform. A good place would be inside your main function.

void main() {
  VirtualPlatform.notifier = VirtualPlatformNotifier(VirtualPlatform.fromPhysicalPlatform);
  // the virtual platform is reset upon every restart
  runApp(VirtualPlatformDispatcher(
    other: (_, __) => const MyApp(), // this will get selected by all devices but iOS devices
    ios: (_, __) => const MyIosApp(), // this will get selected by iOS devices
  ));
}

If the goal is to force a specific virtual platform on startup for all platforms:

void main() {
we  VirtualPlatform.notifier = VirtualPlatformNotifier(VirtualPlatforms.ios);
  // the virtual platform is reset upon every restart
  runApp(VirtualPlatformDispatcher(
    other: (_, __) => const MyApp(),
    ios: (_, __) => const MyIosApp(), // this will get selected by all devices
  ));
}

The previous virtual platform can also be loaded from persistance storage.


// in a previous session:
sharedPreferences.setString('virtual_platform', VirtualPlatform.current.toString());

// when the app starts:
void main() async {
  final sharedPreferences = await SharedPreferences.getInstance();
  final previous = sharedPreferences.getString('virtual_platform');
  final previousVirtualPlatform = VirtualPlatform.fromString(previous);
  VirtualPlatform.notifier = VirtualPlatformNotifier(previousVirtualPlatform);
  runApp(VirtualPlatformDispatcher(
    other: (_, __) => const MyApp(),
    ios: (_, __) => const MyIosApp(),
  ));
}

How to change the virtual platform

You can change the platform by accessing VirtualPlatform.notifier's setter chosenPlatform, e.g.:

ElevatedButton(
    onPressed: () => VirtualPlatform.notifier.chosenPlatform =
        VirtualPlatforms.linux,
    child: const Text('set linux'),
),
ElevatedButton(
    onPressed: () => VirtualPlatform.notifier.chosenPlatform =
        VirtualPlatforms.ios,
    child: const Text('set ios'),
),
...

How to reset the virtual platform:

ElevatedButton(
    onPressed: () => VirtualPlatform.notifier.chosenPlatform =
        VirtualPlatform.fromPhysicalPlatform,
    child: const Text('reset virtual platform'),
),

VirtualPlatforms

VirtualPlatforms is a class containing all possible VirtualPlatform values as static fields. The constructors of VirtualPlatforms and VirtualPlatform are both private.

N.B.: The default and current virtual platform are accessed through VirtualPlatform (and not through VirtualPlatforms):

  • VirtualPlatform.fromPhysicalPlatform
  • VirtualPlatform.current (once the notifier is set)

VirtualPlatformDispatcher

VirtualPlatformDispatcher is a widget builder dispatcher that should be used if there's a dependency only on the virtual platform.

VirtualPlatformDispatcher in action

The following snippets should be self-explanatory. If they are not, consult the platform table at the end of this README.

Each defined platform requires a nullable TransitionBuilder, with exception to the other chain, whose TransitionBuilder is non-nullable.

VirtualPlatformDispatcher(
  desktopSystems: (_, __) => Text('desktop'),
  other: (_, __) => Text('other'),
);
// 'other' on web, all mobile platforms and fuchsia
// 'desktop' will be displayed on macOS, Windows and Linux
VirtualPlatformDispatcher(
  desktopSystems: (_, __) => Text('desktop'),
  linux: (_, __) => Text('linux'),
  other: (_, __) => Text('other'),
);
// 'other' on web, all mobile platforms and fuchsia
// 'desktop' will be displayed only on macOS and Windows
// 'linux' will be displayed on Linux
VirtualPlatformDispatcher(
  appleSystems: (_, __) => Text('apple'),
  other: (_, __) => Text('other'),
);
// 'apple' will be displayed on macOS and iOS
VirtualPlatformDispatcher(
  appleSystems: (_, __) => Text('apple'),
  ios: (_, __) => Text('ios'),
  other: (_, __) => Text('other'),
);
// 'apple' will be displayed only on macOS

Other chain and safety measures

The following code is pretty safe:

VirtualPlatformDispatcher(
  other: (_, __) => Text('other'),
  mobileSystems: (_, __) => Text('mobile'),
  appleSystems: (_, __) => Text('apple'),
  ios: (_, __) => Text('ios'),
  desktopSystems: (_, __) => Text('desktop'),
  linux: (_, __) => Text('linux'),
  web: (_, __) => Text('web'),
);

However, not the same might be said about the following two widgets:

// [1] At the top of the widget tree:
VirtualPlatformDispatcher(
  other: (_, __) => MaterialApp(...), // NB: do not use the context, since it doesn't exist yet.
  ios: (_, __) => CupertinoApp(...), // NB: do not use the context, since it doesn't exist yet.
  macos: (_, __) => MacosApp(...), // NB: do not use the context, since it doesn't exist yet.
);

// [2] More down in the widget tree:
VirtualPlatformDispatcher(
  other: (_, __) => ListTile(...), // will crash if virtual platform is iOS
  macos: (_, __) => MacosListTile(...),
);

Specifying other does not automatically bring safety, i.e., a crash might occur even if the other chain is specified. For example, let us assume the virtual platform is VirtualPlatforms.ios and the widget tree consists of Cupertino widgets (1 in the code snippet above). If the matched chain is other and this chain builds a widget using the Material design (2 in the code snippet above), this widget should be wrapped in a Material before getting added to the widget tree. If not, it might result in a crash. The following snippet is a possible fix:

// Safe variant of [2]:
VirtualPlatformDispatcher(
  other: (_, __) => ListTile(...),
  ios: (_, __) => Material(child: ListTile(...)),
  macos: (_, __) => MacosListTile(...),
);

Also, if support for an additional design language is added or dropped, the static analysis will not show you any errors or warnings if you have been using this package for design-language dispatching.

For these reasons, one should prefer the package design_language over virtual_platform when doing cross-design-language development

Context and child provision

One might need the context in some cases. The first parameter of all builders is a BuildContext context. You can also specify a Widget? child widget, so that you do not have to repeat it in all builders.

Example:

return MaterialApp(
    title: 'Flutter Demo',
    theme: ThemeData(
        primarySwatch: Colors.blue,
    ),
    home: Scaffold(
        appBar: AppBar(title: const Text('example')),
        body: VirtualPlatformDispatcher(
            child: const CommonSubtree(),
            mobileSystems: (context, child) => SpecialWidget(
                child,
                onPressed(() {do something with context}),
            ),
            other: (context, child) => OtherSpecialWidget(
                child,
                onPressed(() {do something with context}),
            ),
        );
    ),
);

matchVirtualPlatform

matchVirtualPlatform is a declarative pattern to invoke the right function for the matching physical platform.

You can use this in case you need to invoke some virtual-platform-specific functions, e.g.:

  • showCupertinoModalPopup for iOS
  • showModalBottomSheet for all other cases

Same example, in code:

// inside some callback
matchVirtualPlatform(
  other: () => showModalBottomSheet(...),
  ios: () => showCupertinoModalPopup(...),
);

Avoid using this function for building widgets as VirtualPlatformDispatcher should be used for this use-case.

Platforms and platform groups


Platform appleSystems mobileSystems desktopSystems
android
ios
linux
macos
windows
web
fuchsia

Priority order: from left to right.

Libraries

virtual_platform
A package that simplifies the cross-platform development process by providing a virtual platform than can be changed at runtime.