Skip to content

Latest commit

 

History

History
174 lines (133 loc) · 5.88 KB

README.md

File metadata and controls

174 lines (133 loc) · 5.88 KB

FlutterEmbedder

Flutter is an open-source UI software development kit created by Google. This project aims to use the Flutter "embedder" system to provide embedded UI applications targeting Nerves devices.

Similar Projects

This idea isn't completely original. Some notable similar projects include:

  • flutter-pi - A light-weight Flutter Engine Embedder for Raspberry Pi that runs without X.
  • go-flutter - Flutter on Windows, MacOS and Linux - based on Flutter Embedding, Go and GLFW.
  • Flutter from Scratch - A really nice introduction to the Flutter Embedder.

Getting Started

NOTE: this is still very much a work in progress project and document.

The below document assumes you use something akin to Poncho Projects when developing your Nerves application. This guide will take you through creating both the nerves and flutter application codebases, but adding flutter to an existing app should be equally as easy.

Dependencies and setup

The first steps will be to install Nerves and Flutter. You can find guides to both below:

NOTE: Currently this project is only compatible with Flutter 1.22.4.

flutter version 1.22.4

Once that's complete create a new folder for your project and then scafold two applications:

mkdir flutter-nerves-helloworld
cd flutter-nerves-helloworld
# Create a new Nerves Project
# currently, only rpi4 is supported.
mix nerves.new firmware --target=rpi4
# Create a new flutter application
flutter create ui

Wiring it up

The first step to getting up and running is to add flutter_embedder to your Elixir application's dependencies. Open mix.exs:

def deps do
  # ... ommited for clarity
  # Dependencies for specific targets
  # Add this line
  {:flutter_embedder, "~> 0.0", targets: :rpi4}
end

Then run a quick mix deps.get in that folder. This will fetch the flutter_embedder project.

Next, in that same file add a new option to the def project section:

def project do
  # ... ommited for clarity
  # add a new line to configure elixir to run the `flutter` compile step:
  compilers: compilers(Mix.target()),
  # Configure the flutter compiler to compile the app we created
  # in the previous step
  flutter: [
    cd: Path.expand("../ui", __DIR__)
  ]
end

# Create this function:
def compilers(:rpi4), do: [:flutter | Mix.compilers()]
def compilers(:host), do: Mix.compilers()

The final step is to start the Flutter embedder in your application. The easiest way to do this is by adding the child to your supervision tree. Open lib/application.ex:

def children(_target) do
  flutter_opts = [
    flutter_assets: Application.app_dir(:firmware, ["priv", "flutter_assets"])
  ]
  [
    # ... ommited for clarity
    {FlutterEmbedder, flutter_opts}
  ]
end

NOTE: By default Flutter uses Ariel font. These fonts require a license aggreement and are not supplied in this project. To boot flutter on your device, you will need this file in your firmware project:

f11c0317db527bdd80fa0afa04703441  rootfs_overlay/usr/share/fonts/truetype/msttcorefonts/Arial.ttf

Finally, to get the application up and running, follow the standard Nerves workflow:

mix firmware.burn

And when it boots, you should see the default flutter application.

Dart Hot Code Reloading with Visual Studio Code

One of the major selling points of Flutter is the ability to reload code instantly while developing. The Elixir Embedder supports this similarly to how go-flutter does it.

In your ui project folder, create a file called .vscode/launch.json if it's not there already:

{
  // Use IntelliSense to learn about possible attributes.
  // Hover to view descriptions of existing attributes.
  // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
  "version": "0.2.0",
  "configurations": [
    {
      "name": "Nerves Flutter Embedder",
      "request": "attach",
      "deviceId": "flutter-tester",
      "observatoryUri": "${command:dart.promptForVmService}",
      "type": "dart",
      "program": "lib/main.dart" // Dart-Code v3.3.0 required
    }
  ]
}

After that's available, from your firmware folder open a terminal and issue the command:

mix flutter.discover

It should return a result that looks something like:

Discovering devices via MDNS
=============================================================

  Found Flutter Observatory: 192.168.1.127
      tunnel:      ssh -L 46603:localhost:46603 192.168.1.127
      url:         http://localhost:46603/is1QgudddHQ=/

      launch.json: {"deviceId":"flutter-tester","name":"Nerves Flutter (192.168.1.127)","observatoryUri":"http://localhost:46603/is1QgudddHQ=/\n","program":"lib/main.dart","request":"attach","type":"dart"}
=============================================================

Now, copy the value in the tunnel section, execute it. This should open an SSH session to your device. Next copy the url section into your clipboard.

Finally back in the ui editor, press F5 and when it prompts you for a URL, paste the url from your clipboard. This will automatically connect to a debug session on the device. Every save you make to the Dart code will automatically sync over to the device.

NOTE: It must be noted that the port and path of this URL will change every time the Dart applicaiton is restarted, for example, reboot, crash, firmware upgrade etc. This means you will need to rerun this mix task every time one of those events happen.