I read an article by Tom Whitwell, detailing his process for creating a slow media player using e-paper and a Raspberry Pi 4. His project was in turn inspired by a 2018 article by Bryan Boyer about the same thing. Both of these were very simple, yet visually stunning and led me to create my own version of this project.
Both of the reference articles had pieces I liked and pieces I wanted to enhance about them. This particular project takes the strengths of their ideas and adds a few "ease of use" elements. The biggest of these are the ability to specify different displays and a built in web service to control file selection and playback. Once setup you shouldn't need CLI access to modify any parameters or change running files.
For detailed installation instructions read the Install document. An install script is also available in the setup
directory.
Display hardware is a big choice for this project. I used the Waveshare 7.5in E-ink display. Using the --epd
argument from above you can specify any display that works with the EPD abstraction library. This allows dynamic loading of different display types depending on what you've purchased. You can see an exhaustive list here. In general, most Waveshare displays are compatible.
Once the requirements are in place the program itself is contained in the vsmp.py
file. This file can take a few arguments, but doesn't need any to run normally:
--config
- path to a config file where any CLI arguments can be specified, useful for when running as a service--port
- the port the web server will run on, default is 5000--epd
- the e-ink display driver to use. Valid displays can be found here. waveshare_epd.epd7in5_V2 is the default.--debug
- when this flag is given the program will run in debug mode
Once started, either with the CLI or as a service, the program will start and the web service will be active. The display should also show a startup page that prompts you to continue setup. Load the web interface to continue setup at http://IP:5000. To monitor the progress of the program you can watch the log file using the command:
# from the vsmp-plus directory
tail -f tmp/log.log
Instead of passing in all the arguments on the command line you can also create a config file and pass in your values with the -c
option instead. There is a sample of this file in the setup/
directory. Any command line argument can be specified in the file.
# Example Config file
port = 8080
The program runs an embedded web server to control the program status and set additional parameters. The main page will show you the currently playing file and allow you to pause or resume the configured schedule. This is helpful if you wish to pause things and remove a USB stick to add more files without having to shut down the entire Raspberry Pi. When first run the player will be paused so you can change the settings to correct values. If running in directory mode you can also click the Next or Prev to cycle through videos in the directory. Clicking on the progress bar will seek the video to that percentage. This can be disabled on the setup page if unwanted.
The Player Setup page allows for the configuration of more specific parameters. This can be saved and will take effect without a program reboot. Settings do take about 1 min for the program to reload the config.
- Mode - File mode will play a single file whereas directory mode will play all the files in a given directory. This must be local to the Raspberry Pi.
- Path - The absolute path to either the file or the directory
- Update Time - how often to update the display, this is given as a cron expression
- Start time skip - the number of seconds into the video to start, if you want to skip into the video X amount
- End time skip - the number of seconds to cut off the end of the video, useful for skipping credit sequences
- Display - optionally show any combination of the title, timecode of the frame being displayed, or the device IP on the bottom of the display. Time in the form of HH:mm:SS. The device IP will automatically toggle itself on if the IP address of the system changes while the player is running. This helps prevents a "lost" player on the network due to DHCP.
- Allow Seeking - this will enable or disable seeking when clicking on the progress bar in the web interface. Useful to disable if this is happening on accident.
- Skip Blank Frames - when enabled frames that are all black (blank) will not be displayed. Useful for black screens at the start, end, or even middle of video files.
- Show Startup Screen - enables or disables the display of a startup screen when VSMP+ starts.
Once applied the given cron expression will be used to update the display starting at the start
frame of the video. The image will be displayed and then status information, specific to this video file, will be written to the database with the next frame to display. At each update time the database will be checked and the next frame will be displayed. Subsequent runs will continue to move forward by the increment
amount. If the video ends it will start over at the start
frame again. If reading from a directory it will start the next video. The log file for the program is stored in the tmp
directory, which is created the first time the program is run. Information related to the current player status and configuration is stored in a Redis database.
Once the program was up and running, one thing that was very hit/miss was what exactly the input parameters should be for my desired effect. Did I want the video take days, weeks, or months to display? What combination of increments and delays would get the effect I wanted? Using the Analyze menu item you can test parameters and see what happens with a given file, or set of files.
By default the analyze program loads the current settings. These can be tweaked without altering the main player that is running. Using the inputs the video is analyzed and some information is displayed regarding projected play times. Tweaking the configuration values you can find the optimum settings to get your desired play time. Each video will display separately, with a summary at the end. When looking at a whole directory the program will use the position of the currently playing file and analyze from this point forward.
The built in web server uses a few API endpoints to function. These can be utilized by other programs as well if you wish to automate the sign with other systems or scripts. The endpoints available are:
This returns the current configuration as a JSON object. Using a POST request you can update data, with settings like the file paths and cron expression being verified. Responses will include a success or failure of the update.
Example
curl http://localhost:5000/api/configuration
{
"allow_seek": false,
"display": ["timecode"],
"end": 300,
"increment": 50,
"mode": "dir",
"path": "/media/usb/Videos",
"start":100,
"update":"*/5 7-15 * * 1-5",
"show_startup": true,
"skip_blank": false
}
Initiate a player control action. Valid actions at this time are resume, pause, next, prev, and seek. Note that next and prev functions will return an error when not in diretory mode. Seeking requires an additional parameter in the POST body: {amount: percent}
where the percentage is a whole number 0-100.
Examples
# Pause Play
curl http://localhost:5000/api/action/pause
{
"action": "pause",
"message": "Action pause executed",
"success": true
}
# Seek
curl http://localhost:5000/api/control/seek -d '{"amount": 20}' -X POST
{
"action":"seek",
"data": ...same as /api/status...,
"message": "Seeking to 20.00 percent on next update",
"success":true
}
Returns the current status of the sign as a JSON object. This includes information about the currently playing video like it's title, file path, and percent complete. The pos
key value is the current frame being displayed.
Example
curl http://localhost:5000/api/status
{
"file": "/media/usb/Videos/Path.To.Video.mp4",
"info": {
"fps": 29.97,
"frame_count": 60725.0,
"runtime": 2326.192859526193,
"title":"Friendly Name of Video"
},
"player": {
"last_run": 1620242760.1,
"next_run": 1620242880.0,
"running": true
}
"name": "Video.File.Name.No.Ext",
"percent_complete": 20.5,
"pos": 12145.0
}
Returns the currently displayed image from the video file. This is a binary PNG file. Can be saved directly via something like curl
or embedded in another web application.
Takes the same parameters as the /api/configuration
POST method, however this will run the analyzer on the proposed configuration. The response includes a break down of each video analyzed.
Returns a list of directories and files within the given file path. Files are filtered to include only valid video files. Used by the web interface file browser. The user running the vsmp.py
file must have READ access to the directories to list them.
Example
curl http://localhost:5000/api/browse_files/media/usb/Videos
{
"dirs":[
"Dir 1",
"Dir 2"
],
"files": [
"Video.File.1.mp4",
"Video.File.2.mp4"
],
"path": "/media/usb/Videos",
"success": true
}
I mentioned two other versions of this type of project that I took inspiration from when creating this one. I tried to combine pieces of each that I liked while putting my own spin on things. Below is a quick summary of differences from other implementations. This may be overkill for some users, and if so checkout one of the other great VSMP projects.
- Using cron syntax for updating instead of a simple delay in seconds. This allows for more complex schedules, like not updating at night if you don't want to miss something.
- Added analyzer to help with figuring out video play times
- Use seconds instead of frames for
--start
value. More intuitive - Added
--end
value to skip end credits - Added the
-display
value so you can see where the title of the video and/or the timecode for the frame displayed - Added a built in web service for controlling the sign so more can be done without the CLI or restarting the service directly
- Use a Redis database to store information rather than a host of flat files. This should help limit reads/writes to the Raspberry PI SD card
- Added ability to jump to a specific time in the video (via web UI)
- Added blank frame (black screen) detection
- Added an abstraction library to load multiple types of e-ink displays
PRs accepted! If there a fix for any of the documentation or something is not quite clear, please point it out. I made this mostly for fun to get experience with libraries I wasn't very familiar with. Not really looking to enhance it too far from where it is but definitely want to keep it functional as dependencies change.