- Last update:

Creating GPX overlay videos on Linux

A tutorial about adding GPX data on top of videos (speed, heart rate, etc)

🕒 5 min read

Category: Linux

Tags: video, linux, gps, gpx, kdenlive

Just because I found this article so good but a bit too long, I'm creating my own.

Up until now, I was relying on the infamous Virb Edit program from Garmin to create videos with GPX data on top. But this software is only available on Mac and Windows, and also outdated and unmaintained, it does not support HEVC videos. So I started looking for another solution that would work on Linux and be ideally dead simple, with no GUI. Also, I was very tired of having to export the GPX data on top of an existing video. Wouldn't it be great to be able to export a video with a transparent background, that I can then put on top of any other footage? GoPro videos, Insta360 videos, any! That would preserve the quality of the video (no need to re-encode it in Virb and then in the final montage software).

I found salvation in gopro-dashboard-overlay! It does all of that! No GUI, can export just a transparent video with the GPX data only but also on top of an existing footage, supports as many formats and codecs as does FFMPEG, and of course works on Linux!

So here is my TL;DR tutorial on how to use it:

1. Install the project

git clone git@github.com:time4tea/gopro-dashboard-overlay.git
cd gopro-dashboard-overlay
python3 -m venv .env
source .env/bin/activate
sudo apt install pkg-config libcairo2-dev
pip3 install -r requirements.txt
pip3 install gopro-overlay
sudo apt install fonts-roboto

2. Setup

mkdir ~/.gopro-graphics/
vim ~/.gopro-graphics/ffmpeg-profiles.json

Then:

{
  "vp9": {
    "input": [],
    "output": ["-vcodec", "vp9", "-pix_fmt", "yuva420p", "-r", "5"]
  },
  "mp4": {
    "input": [],
    "output": ["-vcodec", "libx264", "-r", "25"]
  },
  "png": {
    "input": [],
    "output": ["-vcodec", "png"]
  }
}

Then:

vim ~/Documents/my-layout-1920x1080.xml

Then:

<layout>
    <composite x="20" y="20" name="date_and_time">
        <component type="datetime" x="0" y="0" format="%H:%M:%S" size="32" align="left"/>
        <!-- <component type="text" x="0" y="36" size="32">Distance (km): </component>
        <component type="metric" x="215" y="36" metric="odo" units="km" size="32" dp="2" /> -->
    </composite>

    <composite x="20" y="976" name="big_kph">
        <!-- 1080 - 20 (margin) - 64 (altitude) - 20 (margin) = y 976 -->
        <component type="metric" x="0" y="-160" metric="speed" units="speed" dp="0" size="160" />
        <component type="metric_unit" x="0" y="-176" metric="speed" units="speed" size="16">km/h</component>
    </composite>

    <composite x="20" y="1060" name="altitude">
        <component type="icon" x="0" y="-64" file="mountain.png" size="64"/>
        <component type="metric_unit" x="70" y="-64" metric="alt" units="alt" size="16">Altitude ({:~C})</component>
        <component type="metric" x="70" y="-46" metric="alt" units="alt" dp="1" size="46" />
    </composite>

    <composite x="270" y="1060" name="gradient">
        <component type="icon" x="0" y="-64" file="slope-triangle.png" size="64"/>
        <component type="text" x="70" y="-64" size="16">Slope (%)</component>
        <component type="metric" x="70" y="-46" metric="gradient" dp="1" size="46" />
    </composite>

    <component type="chart" name="gradient_chart" x="450" y="996"/>

    <composite x="1900" y="980" name="temperature">
        <component type="icon" x="-64" y="-64" file="thermometer.png" size="64"/>
        <component type="metric" x="-70" y="-64" dp="0" size="64" align="right" metric="temp" units="temp"/>
    </composite>

    <composite x="1900" y="1060" name="heartbeat">
        <component type="icon" x="-64" y="-64" file="heartbeat.png" size="64"/>
        <component type="metric" x="-70" y="-64" metric="hr" dp="0" size="64" align="right"/>
    </composite>

    <component type="moving_map" name="moving_map" x="1644" y="20" size="256" zoom="16" corner_radius="35"/>
    <component type="journey_map" name="journey_map" x="1644" y="296" size="256" corner_radius="35"/>
</layout>

Then:

vim ~/Documents/my-layout-portrait-1080x1920.xml

Then:

<layout>
    <composite x="1060" y="1816" name="big_kph">
        <!-- 1920 - 20 (margin) - 64 (altitude) - 20 (margin) = y 1816 -->
        <component type="metric" x="0" y="-160" metric="speed" units="speed" dp="0" size="160" align="right" />
        <component type="metric_unit" x="0" y="-192" metric="speed" units="speed" size="32" align="right">km/h</component>
    </composite>

    <composite x="1060" y="1900" name="heartbeat">
        <component type="icon" x="-64" y="-64" file="heartbeat.png" size="64"/>
        <component type="metric" x="-70" y="-64" metric="hr" dp="0" size="64" align="right"/>
    </composite>
</layout>

Then:

vim ~/Documents/my-layout-4k-3840x2160-ski.xml

Then:

<layout>
    <!-- LAYOUT SIZE = 3840x2160 -->
    <!-- <composite x="40" y="40" name="date_and_time">
        <component type="datetime" x="0" y="0" format="%H:%M:%S" size="64" align="left"/>
    </composite> -->

    <composite x="40" y="1952" name="big_kph">
        <!-- 2160 - 40 (margin) - 128 (altitude) - 40 (margin) = y 1952 -->
        <component type="metric_unit" outline_width="4" x="0" y="-352" metric="speed" units="speed" size="32">km/h</component>
        <component type="metric" outline_width="4" x="0" y="-320" metric="speed" units="speed" dp="0" size="320" />
    </composite>

    <composite x="40" y="2120" name="altitude">
        <!-- 2160 - 40 (margin) = y 2120 -->
        <component type="icon" x="0" y="-128" file="mountain.png" size="128"/>
        <component type="metric_unit" outline_width="4" x="140" y="-128" metric="alt" units="alt" size="32">Altitude ({:~C})</component>
        <component type="metric" outline_width="4" x="140" y="-92" metric="alt" units="alt" dp="1" size="92" />
    </composite>

    <composite x="540" y="2120" name="gradient">
        <component type="icon" x="0" y="-128" file="slope-triangle.png" size="128"/>
        <component type="text" outline_width="4" x="140" y="-128" size="32">Slope (%)</component>
        <component type="metric" outline_width="4" x="140" y="-92" metric="gradient" dp="0" size="92" />
    </composite>

    <composite x="3800" y="2120" name="heartbeat">
        <component type="icon" x="-128" y="-128" file="heartbeat.png" size="128"/>
        <component type="metric" outline_width="4" x="-140" y="-128" metric="hr" dp="0" size="128" align="right"/>
    </composite>
</layout>

3. Generate a video

A transparent video with the data only

.env/bin/gopro-dashboard.py --use-gpx-only --gpx ~/Downloads/some-ride.gpx --profile vp9 --layout-xml ~/Documents/my-layout-1920x1080.xml --overlay-size 1920x1080 --units-speed kph --units-altitude meter --units-distance km --units-temperature degC --gps-speed-max 120 --gps-speed-max-units kph ~/Downloads/output.webm

It's going to take some hours. Encoding with VP9 in a .webm container is actually slower than with PNG in .mov container, but the file size is like 100 times smaller. Another way to speed up the processing is to further reduce the final file framerate (30 by default, here in the XML we set it to 5).

Although Kdenlive works fine with VP9 files with a framerate set to 5, Davinci Resolve 19 does not. In this case, exporting in .mov, using the default built-in mov profile:

.env/bin/gopro-dashboard.py --use-gpx-only --gpx ~/Downloads/some-ski-ride.gpx  --profile mov --layout-xml ~/Documents/my-layout-4k-3840x2160-ski --overlay-size 3840x2160 --units-speed kph --units-altitude meter --units-distance km --units-temperature degC --gps-speed-max 120 --gps-speed-max-units kph ~/Downloads/output.mov

That's the kind of result you can expect:

A screenshot of the video
A screenshot of the video, in case you can't play it

And the final result, merged with GoPro footage:

All that's left now, is merge this video with an actual footage, sync it, and voilĂ ! I recommend using Kdenlive on Linux..

The final video right away (footage + overlay)

Make sure the input video has the correct mtime (modified time). Check with stat file.mp4 or ls -la file.mp4. If it's a Insta360 video, and the filename matches YYYYMMDD_HHMMSS_sss.mp4, you can update it with this script: https://github.com/rpellerin/dotfiles/blob/master/scripts/updateModifyTimeInsta360File.py

.env/bin/gopro-dashboard.py --use-gpx-only --gpx ~/Downloads/some-ride.gpx  --profile mp4 --layout-xml ~/Documents/my-layout-portrait.xml --overlay-size 1080x1920 --units-speed kph --units-altitude meter --units-distance km --units-temperature degC --gps-speed-max 120 --gps-speed-max-units kph --video-time-start file-modified some-video.mp4  ~/Downloads/output.mp4

That's it!