Raspberry Pi: Waveshare e-paper dashboard

Instructions on setting up a Raspberry Pi Zero WH with a Waveshare ePaper 7.5 Inch HAT. The screen will display:

  • Date and time
  • Weather icon with high and low temperature
  • Google Calendar or Outlook Calendar entries

Here it is in action

In a picture frame Generated image
epaper dashboard

Shopping list

E-Paper Display

The most important component is the Waveshare display, which is a 7.5 inch e-paper HAT with SKU: 13504 and UPC: 614961951068. A quick search will also show similar displays available, with a single additional color. As tempting as they may be, the problem with those displays is the refresh rate, in part due to the way the third color is ‘pushed’ to the surface when displaying a color. While the black and white display isn’t very fast, the colored ones are much, much slower and are only suitable for frequently-refreshing dashboards.

E-Paper Display

Raspberry Pi

Although any Raspberry Pi can be used, the best one to get here is the Raspberry Pi Zero W - it’s thinner and more portable. Since it’s a HAT (Hardware Attached on Top), you can save some time by buying it with the GPIO presoldered. Of course you’ll also need a microSD card.

Raspberry Pi Zero WH microSDHC card

Picture frame

You’ll need a 6”x4” picture frame to hold everything together. This is the best size just larger than the e-paper display. The back needs to be made of cheap material so that it can be cut out for the e-paper display’s connection mechanism.

Picture frame

Instructions on setting up a Raspberry Pi Zero WH with a Waveshare ePaper 7.5 Inch HAT. Date/Time, Weather, Google/Outlook Calendar

Setup the PI

Prepare the Pi

I’ve got a separate post for this, prepare the Raspberry Pi with WiFi and SSH. Once the Pi is set up, and you can access it, come back here.

Connect the display

Turn the Pi off, then put the HAT on top of the Pi’s GPIO pins.

Connect the ribbon from the epaper display to the extension. To do this you will need to lift the black latch at the back of the connector, insert the ribbon slowly, then push the latch down. Now turn the Pi back on.

Setup dependencies

The display as well as the code that talks to the display has quite a few dependencies. Here we’re installing a few fonts, some Python dependencies.

sudo apt install git ttf-wqy-zenhei ttf-wqy-microhei python3-pip python-imaging libopenjp2-7-dev libjpeg8-dev inkscape figlet wiringpi python3
sudo pip3 install astral spidev RPi.GPIO Pillow  # Pillow took multiple attempts to install as it's always missing dependencies
sudo pip3 install --upgrade google-api-python-client google-auth-httplib2 google-auth-oauthlib msal

Enable SPI

SPI is how the Raspberry Pi will send data over the ribbon to the e-paper display. You will need to enable it and reboot the device.

sudo sed -i s/#dtparam=spi=on/dtparam=spi=on/ /boot/config.txt  #This enables SPI
sudo reboot

Get the BCM2835 library

The BCM2835 library provides access to the GPIO pins and it needs to be manually installed.

wget http://www.airspayce.com/mikem/bcm2835/bcm2835-1.58.tar.gz
sudo tar zxvf bcm2835-1.58.tar.gz
cd bcm2835-1.58/
sudo ./configure
sudo make
sudo make check
sudo make install

Using this application

You can now use the code from this repo

Clone it

Clone this repository to your Raspberry Pi

git clone --recursive [email protected]:mendhak/waveshare-epaper-display.git

Waveshare version

Modify the env.sh file and set the version of your Waveshare 7.5” e-Paper Module (newer ones are version 2)


Climacell API key

The Climacell API will be used to get the weather conditions. Modify the env.sh file and put your Climacell API key in there.

export CLIMACELL_APIKEY=xxxxxx

Location information for Weather

Modify the env.sh file and update with the latitude and longitude of your location. As needed, change the temperature format (CELSIUS or FARENHEIT).

export WEATHER_LATITUDE=51.3656

Google Calendar setup

The script will by default get its info from your primary Google Calendar. If you need to pick a specific calendar you will need its ID. To get its ID, open up Google Calendar and go to the settings for your preferred calendar. Under the ‘Integrate Calendar’ section you will see a Calendar ID which looks like [email protected]. Set that value in env.sh

export GOOGLE_CALENDAR_ID=[email protected]

Google Calendar token

In order to display Google Calendar events, we’ll need an OAuth Token. However Google have not made the process of obtaining one simple at all. The Oauth process needs to complete once manually in order to allow the Python code to then continuously query Google Calendar for information.

Go to the Python Quickstart page and enable Google Calendar API. When presented, download or copy the credentials.json file and add it to this directory.

Next, on the Raspberry Pi, run:

python3 screen-calendar-get.py

The script will prompt you to visit a URL in your browser and then wait. Copy the URL, open it in a browser on another PC, and you will go through the login process. When the OAuth workflow tries to redirect back (and fails), copy the URL it was trying to go to (eg: http://localhost:8080/...). Open up a brand new SSH session to the Raspberry Pi, and curl that URL.

curl "http://localhost:8080/..." 

On the first SSH session, you should now see the auth flow complete, and a new token.pickle file appears. The Python script should now be able to run in the future without prompting required.

Outlook Calendar setup

You can use Outlook Calendar instead of Google Calendar. The setup is very simple, just run this script which will give instructions on how to login:

python3 outlook_util.py

Login with the Microsoft account you want to get the calendar from, and accept the consent screen. After a moment, the script will then display a set of Calendar IDs and some sample events from those Calendars. Copy the ID of the calendar you want, and add it to env.sh like so:


Note that if you set an Outlook Calendar ID, the Google Calendar will be ignored.

Run it

Run ./run.sh which should query Climacell, Google/Outlook Calendar. It will then create a png, then display the png on screen. After a few runs, if everything is working well, you should then make this a cron job.

* * * * * cd /home/pi/waveshare-epaper-display && bash run.sh > run.log 2>&1

Putting it in a picture frame

The picture frame I got had a cheap backing. Using a box cutter (Stanley knife) I was able to remove a square portion from the bottom. This allowed me to put the e-paper display inside the picture frame while its connector hung outside.

The ribbon from the connector loops upwards and over to the picture frame’s stand. The Raspberry Pi Zero WH is light enough that it could be taped right to the stand.

The only bit of wire in the whole setup is the USB to power the Raspberry Pi.

Cutout for e-paper connector Topdown view or Raspberry Pi attached to picture frame stand
Picture frame details

How it works

Everything starts with the screen-template.svg which holds the labels and layout for the final image to be produced. SVGs are simply XML files which are understood by renderers. Being text files makes them easy to work with from dynamic scripts.

API Calls

The first part of run.sh calls on the screen-weather.get.py script which queries Climacell API, gets the weather info and substitutes icons and temperatures in the SVG. It also sets the date and time. The SVG is then written out to screen-output-weather.svg. The API response is stored in

The last API call is to Google Calendar, the upcoming 2 calendar entries are written to the same SVG.

Due to API rate limits, you will see various .pickle files which store the Google/Outlook Calendar and Dark Sky API responses for a few hours. This means that any new entries in your target calendar won’t show up immediately. Similarly weather info will be up to a few hours delayed.

Image conversion and display

The image is converted from the intermediate SVG to PNG, and then the display.py renders it to screen using the e-Paper libraries. It’s slow, it takes about 30 seconds to write to screen.

It’s possible to use the C libraries to make this process faster, but it requires writing and compiling the display binary yourself. It could further be sped up by converting the PNG to a 1-bit BMP so that there’s less data to send over the wire. The C way would take about 6-8 seconds.

The reason for sticking with the Python way is that I’ve got a v1 Waveshare display, while most users have a v2 Waveshare display, and it’s easier to cater to both this way. Curse of the early adopter!

Learn more: Waveshare documentation and sample code

Waveshare have a user manual which you can get to from their Wiki

The Waveshare demo repo is here. Assuming all dependencies are installed, these demos should work.

git clone https://github.com/waveshare/e-Paper
cd e-Paper

This is the best place to start for troubleshooting - try to make sure the examples given in their repo works for you.

Readme for the C demo

Readme for the Python demo