/

Engineering

Nov 23, 2024

Nov 23, 2024

How to set up deep linking with Flutter for macOS apps

Learn how to set up deep linking for macOS apps using Flutter in this step-by-step guide. Create seamless navigation experiences, enabling users to access specific content directly within your app.

In today's world, the ability to open desktop applications with specific parameters is a staple feature that any modern application can easily adopt. Setting up this functionality with Flutter on macOS is quite easy without downloading a bloated package or plugin.

I am Nathan Courtney, Head of DevOps at Pieces.app. I focus on integrating system APIs with Flutter to leverage various features at the operating system level.

Whether your app is currently running or not, we are going to look into how we can utilize Flutter method channels to create a custom URL scheme or a deep link to launch your macOS application with certain arguments or interact when a user opens the link within their browser.


What is deep linking? 

A deep link allows your app to be opened from specific URLs, carrying parameters for personalized or contextual interactions. This feature is essential for smooth navigation and enhances user experience.


How to set up deep linking for macOS

First, let's start off with a simple Flutter template by creating a new Flutter project on macOS:

flutter create deep_linking

Navigate to the new directory you created, /deep_linking, and create a new file and class within our /deep_linking/lib directory.

Mine will be /deep_linking/lib/deep_link_handler.dart.

Let’s start by creating a simple class that will be the structure of our query parameters coming in from our URL. 

This doesn’t have to be done, but we can do this to provide more structure to our code if we expect links to arrive in a similar manner and to throw any errors if a link comes in without the proper structure.

class DeepLinkURLParameters {
  final String id;
  final String foo;
  final String bar;

  DeepLinkURLParameters({
    required this.id,
    required this.foo,
    required this.bar,
  });
}

Next, within the same file, let's set up our Flutter method channel, which will handle all the incoming URL parameters that come in from the Swift side.

class DeepLinkHandler {
  static const MethodChannel channel = MethodChannel('deep_linking/deep_link_handler');

  Function(DeepLinkURLParameters)? onReceiveParameters;

  DeepLinkHandler() {
    DeepLinkHandler.channel.setMethodCallHandler((call) async {
      switch (call.method.toString()) {
        case 'urlParams':
          DeepLinkURLParameters parameters = extractDeepLinkParameters(call.arguments);
          print(parameters.id);
          print(parameters.foo);
          print(parameters.bar);
          onReceiveParameters?.call(parameters);
        default:
          break;
      }
      return;
    });
  }

  DeepLinkURLParameters extractDeepLinkParameters(Map parameters) {
    String? id = parameters['id'];
    String? foo = parameters['foo'];
    String? bar = parameters['bar'];
    if (id == null || foo == null || bar == null) {
      throw Exception('Failed to get all deep-link parameters');
    }

    return DeepLinkURLParameters(
      id: id,
      foo: foo,
      bar: bar,
    );
  }
}

Our Flutter method channel will listen for messages coming in from the Swift side on the channel: "deep_linking/deep_link_handler." We will extract each parameter coming in from Swift as a map and print it out to our UI.

In your main.dart file, we can initialize our class to spin up our method channel and start listening to events coming in from the Swift side:

import 'deep_link_handler.dart';

void main() {
  runApp(const MyApp());
  WidgetsFlutterBinding.ensureInitialized();
  DeepLinkHandler();
}

We can also modify our application's home page to display our Query Parameters when they start coming in:

class MyApp extends StatelessWidget {
  const MyApp({super.key});

  // This widget is the root of your application.
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: 'Deep Link Demo',
      theme: ThemeData(
        colorScheme: ColorScheme.fromSeed(seedColor: Colors.deepPurple),
        useMaterial3: true,
      ),
      home: const MyHomePage(title: 'Deep Link Demo Home Page'),
    );
  }
}

class MyHomePage extends StatefulWidget {
  const MyHomePage({super.key, required this.title});

  final String title;

  @override
  State<MyHomePage> createState() => _MyHomePageState();
}

class _MyHomePageState extends State<MyHomePage> {
  DeepLinkHandler deepLinkHandler = DeepLinkHandler();
  DeepLinkURLParameters? parameters;

  @override
  void initState() {
    super.initState();
    deepLinkHandler.onReceiveParameters = (DeepLinkURLParameters params) {
      setState(() {
        parameters = params;
      });
    };
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        backgroundColor: Theme.of(context).colorScheme.inversePrimary,
        title: Text(widget.title),
      ),
      body: Center(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: <Widget>[
            if (parameters != null) ...[
              Text('ID: ${parameters!.id}'),
              Text('Foo: ${parameters!.foo}'),
              Text('Bar: ${parameters!.bar}'),
            ] else ...[
              Text('Waiting for deep link parameters...'),
            ],
          ],
        ),
      ),
    );
  }
}

That is it for the Dart side of things, as the rest of the implementation resides within our application's Swift code.

Let's go ahead and build our Flutter application once, then open it up in Xcode.

flutter build macos

Next, let's navigate to our Xcode Project file within /deep_linking/macos/Runner.xcworkspace and double-click it to open it within XCode.

First, we will want to navigate to our Info.plist to set the URL our app should respond to.

The easiest way to do this is to right click our Info.plist file and add the following XML code to the document:

   <key>CFBundleURLTypes</key>
    <array>
        <dict>
            <key>CFBundleTypeRole</key>
            <string>Viewer</string>
            <key>CFBundleURLName</key>
            <string>com.example.deepLinking</string>
            <key>CFBundleURLSchemes</key>
            <array>
                <string>deep-link-example</string>
            </array>
        </dict>
    </array>

Let’s break down what some of this does.

CFBundleURLTypes: This key holds an array of URL types the app can handle. Each type is defined by a dictionary, which specifies details about how the URL should be processed.

  1. Inside the dictionary:

    • CFBundleTypeRole: This field specifies the app's role when handling this URL type. Setting it to "Viewer" means the app opens the URL for viewing.

    • CFBundleURLName: This is a unique identifier for the URL type. It is set to "com.example.deepLinking", which can be any unique string, typically following a reverse-domain style.

    • CFBundleURLSchemes: This key holds an array of URL schemes that the app will recognize. In this case, it's set to ["deep-link-example"], meaning your app will respond to URLs that start with deep-link-example://.

With this configuration, macOS will open your app whenever a URL with the deep-link-example:// scheme is triggered, allowing you to capture parameters from the URL (like those after ?) in your app when launched.

Now that our app is set up to handle URLs that start with deep-link-example://, it's time to implement the Swift code to send those URL parameters up to our Dart runtime.

One quick note: (and this may not be the case for you) Xcode's inline error detection can sometimes get a little wonky when dealing with Flutter applications. I have found that if you build the Flutter app from within Xcode once, it will properly detect any errors you have within your Swift code. You can see you are having this issue by navigating to MainFlutterWindow.swift in Xcode and see if you get this same error "No such module 'FlutterMacos":

If you do see this, simply click the "Play" button at the top left to build the application. Your issue should be resolved after the fact. You can then hit the "Stop" button to end the application's running.

Next, let's create a new Swift file named DeepLinkHandler.swift that will handle incoming parameters while our application is running.

We can then start creating our class to set up our Flutter Method Channel on the Swift side, handle Apple Events for our deep linking, and structure our query parameters using a nice Dictionary that will conform to a Map in Dart.

import Foundation
import Cocoa
import CoreServices
import FlutterMacOS


class DeepLinkHandler: NSObject, FlutterPlugin {
    
    static let shared = DeepLinkHandler();
    
    static var channel: FlutterMethodChannel? = nil;
    
    public static func register(with registrar: FlutterPluginRegistrar) {
        DeepLinkHandler.channel = FlutterMethodChannel(
            name: "deep_linking/deep_link_handler",
            binaryMessenger: registrar.messenger)
        if DeepLinkHandler.channel !== nil {
            registrar.addMethodCallDelegate(shared, channel: DeepLinkHandler.channel!)
            print("Channel is Registered");
        } else {
            print("Channel Is Broken");
        }
    }
}

Currently, we are creating a function to register our method channel on “deep_linking/deep_link_handler and importing some other Swift APIs that we will use later for our Apple Events. 

An important step we do not want to miss is calling our public register() function within our MainFlutterWindow.swift to properly link our Flutter Method Channel with the FlutterViewController(). Doing so will finalize our connection to the Dart runtime and allow our DeepLinkHandler Dart class to start listening to events captured by our DeepLinkHandler Swift class.

Simply add the following line in your MainFlutterWindow.swift file:

DeepLinkHandler.register(with: flutterViewController.registrar(forPlugin: "DeepLinkHandler"))

Your entire file will look something like this: 

import Cocoa
import FlutterMacOS

class MainFlutterWindow: NSWindow {
  override func awakeFromNib() {
    let flutterViewController = FlutterViewController()
    let windowFrame = self.frame
    self.contentViewController = flutterViewController
    self.setFrame(windowFrame, display: true)

    RegisterGeneratedPlugins(registry: flutterViewController)
      
    DeepLinkHandler.register(with: flutterViewController.registrar(forPlugin: "DeepLinkHandler"))

    super.awakeFromNib()
  }
}

Let’s return to our DeepLinkHandler.swift file and create a function to handle incoming Apple Events and a public function that will parse the URL for query parameters and send them to the Dart runtime via the method channel we previously created. 

Let's start by creating a function to handle Apple Events. We will create handleDeepLinkQuery() after the fact.

@objc func handleAppleEvent(event: NSAppleEventDescriptor, replyEvent: NSAppleEventDescriptor) {
        
     guard let appleEventDescription = event.paramDescriptor(forKeyword: AEKeyword(keyDirectObject)) else {
         return
     }
        
     guard let appleEventURLString = appleEventDescription.stringValue else {
         return
     }
        
        
     let appleEventURL = URL(string: appleEventURLString)
     let params = appleEventURL!.query
        
     if (params != nil) {
         if (params?.components(separatedBy: "&") != nil) {
             handleDeepLinkQuery(params: params!)
         }
     }  
 }

This Swift code defines a function handleAppleEvent that is triggered when the app receives an Apple Event.

Here's how it works: 

  1. Extracting the Event Descriptor: The code attempts to retrieve the main parameter of the Apple Event (commonly the URL string for a deep link) by calling paramDescriptor(forKeyword:) with the keyDirectObject keyword.

  2. Converting to a String: If the event descriptor is successfully retrieved, it checks whether it contains a string value, which should be the URL. If either of these steps fails, the function returns early.

  3. Creating a URL and Extracting Query Parameters:

    • If the URL string is valid, the function initializes a URL object.

    • It then attempts to get the query part of the URL (everything after ?), which contains the parameters.

  4. Parsing and Handling Parameters:

    • If params (query string) is not nil, the function separates the parameters by & (common in query strings to separate key-value pairs).

    • If parameters exist, it calls handleDeepLinkQuery with params to process or handle the parameters further.

Next, let's create a public function handleDeepLinkQuery() to handle formatting our Query into a nice readable Dictionary that will conform to our Map in dart

   public func handleDeepLinkQuery(params: String) {
        let components = params.components(separatedBy: "&")
        
        var paramDict: [String: String] = [:]
        
        for index in 0...2 {
            paramDict[components[index].components(separatedBy: "=")[0]] = components[index].components(separatedBy: "=")[1]
        }
        
        if ((paramDict.count == 3) && (paramDict["id"] != nil) && (paramDict["foo"] != nil) && (paramDict["bar"] != nil)) {
            DeepLinkHandler.channel!.invokeMethod("urlParams", arguments: paramDict)
        }
        
        return
    }

Now, let's add some functions to register and unregister our class with the NSAppleEventManger. In doing so, our macOS app will now accept incoming events. specifically the kAEGetURL event, fired by the system through the given URL we set in the Info.plist file. This will be triggered when the application is opened via a URL scheme (such as deep-link-example://).

   func registerAppleEventHandler() {
        NSAppleEventManager.shared().setEventHandler(self, andSelector: #selector(self.handleAppleEvent(event:replyEvent:)), forEventClass: AEEventClass(kInternetEventClass), andEventID: AEEventID(kAEGetURL))
    }
    
    func unregisterAppleEventHandler() {
        NSAppleEventManager.shared().removeEventHandler(forEventClass: AEEventClass(kInternetEventClass), andEventID: AEEventID(kAEGetURL))
    }

Let’s break down each function: 

  1. registerAppleEventHandler():


    • This function sets up a handler for the kAEGetURL Apple Event, which is typically triggered when the app is opened via a custom URL scheme.

    • NSAppleEventManager.shared().setEventHandler(...) registers self (the current instance of the class) as the handler for this event.

    • The @selector(self.handleAppleEvent(event:replyEvent:)) specifies that the handleAppleEvent(event:replyEvent:) method should be called when the kAEGetURL event occurs.

    • The forEventClass: AEEventClass(kInternetEventClass) and andEventID: AEEventID(kAEGetURL) arguments specify that this handler should only be triggered for URL events (kAEGetURL).


  2. unregisterAppleEventHandler():

  • This function removes the event handler for the kAEGetURL event, effectively stopping the app from handling URL events until the handler is registered again.

We will call these functions on bootup and teardown of our application. 

Go ahead and add the following lines of swift in your AppDelegate.swift:

override func applicationDidFinishLaunching(_ notification: Notification) {
    DeepLinkHandler.shared.registerAppleEventHandler()
}
    
    
override func applicationWillTerminate(_ notification: Notification) {
    DeepLinkHandler.shared.unregisterAppleEventHandler()
}

Here’s how it all works: When a URL with the custom scheme is opened, macOS triggers an Apple Event that our Swift code captures, parses, and sends to Flutter over the Method Channel.

Let’s test this all out now!

Run your Flutter application, and attempt to paste the following URL within your browser. Chrome will be an excellent test:

deep-link-example://?id=123&foo=abc&bar=xyz

If everything is done correctly, you should see those query parameters printed out in the terminal and in the Flutter app's UI.


flutter: 123

flutter: abc

flutter: xyz


But we are not quite done yet! When you build out your production application, you will find that you will not receive any of these values from a cold start. 

In this scenario, your application is not currently running in the background. 

To fix that, we can add a snippet of code within our AppDelegate.swift file to handle this:

override func application(_ application: NSApplication, open urls: [URL]) {
     if let url = urls.first {
         let params = url.query
         if (params != nil) {
             if (params?.components(separatedBy: "&") != nil) {
                 DeepLinkHandler.shared.handleDeepLinkQuery(params: params!)
             }
         }
     }
 }

Now, when your app isn’t actively running, we can propagate those query params to the Dart runtime on boot! To test this out, you will need to build a release version of your application and then place it within your /Applications directory. You may also need to launch it at least once for everything to register properly. 

Now you’re all set and you should be able to flawlessly launch your desktop application through a deeplink, just like this.

Written by

Written by

SHARE

SHARE

How to set up deep linking with Flutter for macOS apps

Title

Title

our newsletter

Sign up for The Pieces Post

Check out our monthly newsletter for curated tips & tricks, product updates, industry insights and more.

our newsletter

Sign up for The Pieces Post

Check out our monthly newsletter for curated tips & tricks, product updates, industry insights and more.

our newsletter

Sign up for The Pieces Post

Check out our monthly newsletter for curated tips & tricks, product updates, industry insights and more.