Turn your OBEGRÄNSAD LED Wall Lamp into a live drawing canvas
⚠ Disclaimer: Use this code and instructions at your own risk! Improper use may damage the device. Contribute: Have suggestions or improvements? Feel free to submit a PR or open an issue. 😊
- Persist your drawing
- Rotate image
- Live Drawing
- OTA Update
- Wifi Control
- Web-GUI
- Load an image
- Switch plugin by pressing the button
- Schedule Plugins to switch after "n" seconds
- Plugins
- Draw
- Game of life
- Breakout
- Snake
- Stars
- Lines
- Circle
- Clock
- Big Clock
- Weather
- Rain
- Animation with the "Animation Creator in Web UI"
- Firework
- DDP
- Pong Clock
Area.mp4
You can control the lamp with a supplied web GUI. You can get the IP via serial output or you can search it in your router settings.
First of all. This software was written for the ESP32 Dev Board, but it should work with any other Arduino board as well. You just need to remove the WiFi, OTA and web server related code.
The ESP32 I used:
Verified to work with TTGO LoRa32 V2.1 (T3_V1.6.1). Note: On esp8266 per pixel brightness only works when storage and global brightness (analogWrite) are disabled.
I'm sorry to say this, but you'll have to pry open the back of your Lamp, as IKEA didn't install regular screws here. I lifted the back with a screwdriver between the screws and pried it open with a second object, but you can also drill out the rivets to avoid breaking the backpanel.
After you open the back, you will see 4 identical plates. These are each equipped with 64 Leds in 4 fields. We are only interested in the lowest one. Here you will find 6 connectors at the bottom edge, to which we connect our board. Above is a microcontroller. You have to remove it, because it contains the standard programs.
-
Prerequisites
- Install Visual Studio Code.
- Install the PlatformIO IDE extension from the VS Code Extensions Marketplace.
-
Clone the Project
- Download the project from GitHub and open it in VS Code. PlatformIO will automatically load dependencies.
git clone git@github.com:ph1p/ikea-led-obegraensad.git
cd ikea-led-obegraensad
code .
-
Connect ESP32
- Connect your ESP32 via USB.
- Check the COM port in the PlatformIO Devices tab.
-
Prepare the Project
- Perform a
PlatformIO: Clean
(Recycle bin icon at the bottom right). - Add a
secrets.h
file to theinclude
directory. Modify passwords and save the file. Go in the next section for WiFi instructions.
- Perform a
#pragma once
#define WIFI_HOSTNAME ""
#ifdef ESP8266
#define WIFI_SSID ""
#define WIFI_PASSWORD ""
#endif
#define OTA_USERNAME ""
#define OTA_PASSWORD ""
- Set variables inside
include/constants.h
.
-
Build the Project
- Click the
PlatformIO Build
icon (bottom right corner). - Check the log for missing libraries.
- Use the Libraries icon in PlatformIO to install required libraries.
- Repeat
Clean
andBuild
until the build succeeds.
- Click the
-
Upload to ESP32
- Click
PlatformIO Upload
(bottom right) to upload the firmware to the ESP32.
- Click
Note: The WiFi manager only works on ESP32. For ESP8266, WIFI_SSID
and WIFI_PASSWORD
need to be provided in secrets.h
.
This project uses tzapu's WiFiManager. After booting up, the device will try
to connect to known access points. If no known access point is available, the device will create a network called
Ikea Display Setup WiFi
. Connect to this network on any device. A captive portal will pop up and will take you
through the configuration process. After a successful connection, the device will reboot and is ready to go.
The name of the created network can be changed by modifying WIFI_MANAGER_SSID
in include/constants.h
.
Connect them like this and remember to set them in include/constants.h
according to your board.
LCD | ESP32 | TTGO LoRa32 | NodeMCUv2 | Lolin D32 (Pro) |
---|---|---|---|---|
GND | GND | GND | GND | GND |
VCC | 5V | 5V | VIN | USB |
EN (PIN_ENABLE) | GPIO26 | IO22 | GPIO16 D0 | GPIO26 |
IN (PIN_DATA) | GPIO27 | IO23 | GPIO13 D7 | GPIO27 |
CLK (PIN_CLOCK) | GPIO14 | IO02 | GPIO14 D5 | GPIO14 |
CLA (PIN_LATCH) | GPIO12 | IO15 | GPIO0 D3 | GPIO12 |
BUTTON one end | GPIO16 | IO21 | GPIO2 D4 | GPIO25 |
BUTTON other end | GND | GND | GND | GND |
Thanks to RBEGamer who is showing in this issue how to use the original button wiring. With this solution you won't need the "BUTTON one end" and "BUTTON other end" soldering from the table above.
Get current values and the (fixed) metadata, like number of rows and columns and a list of available plugins.
GET http://your-server/api/info
To set an active plugin by ID, make an HTTP PATCH request to the following endpoint, passing the parameter as a query string:
PATCH http://your-server/api/plugin
e.g. curl -X PATCH "http://your-server/api/plugin?id=7"
id
(required): The ID of the plugin to set as active.
- Success:
200 OK
with the message "Plugin Set". - Not Found:
404 Not Found
with the message "Plugin not found".
To set the brightness of the LED display, make an HTTP GET request to the following endpoint, passing the parameter as a query string:
PATCH http://your-server/api/brightness
e.g. curl -X PATCH "http://your-server/api/brightness?value=100"
value
(required): The brightness value (0..255).
- Success:
200 OK
with the message "Ok". - Invalid Value:
404 Not Found
with the message "Invalid Brightness Value".
To get the current displayed data as an byte-array, each byte representing the brightness value. Be aware that the global brightness value gets applied AFTER these values, so if you set the global brightness to 16, you will still get values of 255 this way.
GET http://your-server/api/data
It is possible to switch between plugins automatically. You can define your schedule in the Web UI or just send an API call.
### set your plugins and duration in seconds
curl -X POST http://x.x.x.x/api/schedule -d 'schedule=[{"pluginId":10,"duration":2},{"pluginId":8,"duration": 5}'
### clear the schedule
curl http://x.x.x.x/api/schedule/clear
### start the schedule
curl http://x.x.x.x/api/schedule/start
### stop the schedule
curl http://x.x.x.x/api/schedule/stop
You can set the panel to DDP using the button or via the web interface. This Protocol uses UDP and listens on Port 4048.
Info: The DDP Header is 10 Bytes!
In the repository you will find an ddp.py
as an example.
Change the plugin to DDP
and run following command with your IP:
./ddp.py --ip x.x.x.x --pixel 2 2 200 --pixel 0 0 255
The LED Display service provides HTTP API to display messages and graphs on a 16x16 LED display. This functionality can be accessed through HTTP calls to the service endpoint.
To display a message on the LED display, users can make an HTTP GET request to the following endpoint:
http://your-server/api/message
text
(optional): The text message to be displayed on the LED display.graph
(optional): A comma-separated list of integers representing a graph. The values should be in the range of 0 to 15 and will be visualized as a graph on the LED display.miny
(optional): scaling for lower end of the graph, defaults to 0maxy
(optional): scaling for upper end of the graph, defaults to 15repeat
(optional): The number of times the message should be repeated. If not provided, the default is 1. Set this value to -1 to repeat infinitely. While messages ar pending for display an indicator led in the upper left corner will flash.id
(optional): A unique identifier for the message. This can be used for later removal or modification of the message.delay
(optional): The number of ms of delay between every scroll move. Default is 50 ms.
GET http://your-server/api/message?text=Hello&graph=8,5,2,1,0,0,1,4,7,10,13,14,15,15,14,11&repeat=3&id=1&delay=60
This example will display the message "Hello" on the LED display with a corresponding graph, repeat it three times, and assign it the identifier 1, waits 60ms while scrolling.
To remove a message from the display, users can make an HTTP GET request to the following endpoint:
http://your-server/api/removemessage
id
(required): The unique identifier of the message to be removed.
GET http://your-server/api/removemessage?id=1
This example will remove the message with the identifier 1 from the LED display.
-
src
contains the arduino code.- Run it with platform io
- You can uncomment the OTA lines in
platformio.ini
if you want. Replace the IP with your device IP.
-
frontend
contains the web code.- First run
pnpm install
- Set your device IP inside the
.env
file - Start the server with
pnpm dev
- Build it with
pnpm build
. This command creates thewebgui.cpp
for you.
- First run
-
Build frontend using
Docker
- From the root of the repo, run
docker compose run node
- From the root of the repo, run
- Start by creating a new C++ file for your plugin. For example, let's call it plugins/MyPlugin.(cpp/h).
plugins/MyPlugin.h
#pragma once
#include "PluginManager.h"
class MyPlugin : public Plugin {
public:
MyPlugin();
~MyPlugin() override;
void setup() override;
void loop() override;
const char* getName() const override;
void teardown() override; // optional
void websocketHook(DynamicJsonDocument &request) override; // optional
};
plugins/MyPlugin.cpp
#include "plugins/MyPlugin.h"
MyPlugin::MyPlugin() {
// Constructor logic, if needed
}
void MyPlugin::setup() {
// Setup logic for your plugin
}
void MyPlugin::loop() {
// Loop logic for your plugin
}
const char* MyPlugin::getName() const {
return "MyPlugin"; // name in GUI
}
void MyPlugin::teardown() {
// code if plugin gets deactivated
}
void MyPlugin::websocketHook(DynamicJsonDocument &request) {
// handle websocket requests
}
- Add your plugin to the
main.cpp
.
#include "plugins/MyPlugin.h"
pluginManager.addPlugin(new MyPlugin());
- Check all soldering points, especially VCC
- Check if the board gets enough power