IMG_0159.jpg

Automated Blind Conversion

motorizing IKEAs cheapest blinds with a backend connection to HomeKit

- SEE V2 HERE -


The Project

I really love smart home devices and automating trivial parts of my routine. One thing I’ve always wanted was motorized blinds that would close when I go to bed (to block the street light from pouring into my room) and open automatically in the morning and let me wake up to natural light! The current offerings for automatic blinds are pretty limited and very expensive. The cheapest ones that will connect to HomeKit are from IKEA, priced at a whopping $159.99 each. I think we can do it cheaper with a little bit of creativity using my existing IKEA blinds. This guide will walk through the mechanical assembly, wiring, electrical integration, and software setup to turn your standard IKEA blinds into HomeKit enabled motorized blinds!

You can check out the blog posts I made throughout the design process for more details on how they were developed:

What You’ll Need

For most of these items, you’ll need 1 per blind your looking to be motorized. The Arduino Nano should be able to control up to 3 blinds based on it’s available IO’s but theoretically you could control as many blinds as you want by scaling this setup, just make sure your power supply is up to the task (~1A per motor)! For everything else just multiply the quantity by the number of blinds you want to motorize.

Purchased Parts

You’ll also want an assortment of wires and connectors to create the wire harnesses and protoboard as well as access to a solder iron and solder. I use 2.54mm JST connectors for my projects. Depending on what you already have, you should only need to spend $40-$60 to assembly the motorized blinds.

Printed Parts

Files for these parts can be found on my thingiverse page here!

  • 1x Motorized Blind Bracket

  • 1x Drive Gear

  • 1x Pinion Gear

  • 1x Tube Insert

  • 1x Home Tab

  • 1x Board Housing (optional)

  • 1x Board Cover (optional)

There are mirrored versions of some files that let you mount the motor on different sides of the blinds. In my setup, I have one regular and one mirrored bracket.

Assembly

Drive Bracket Assembly

This page has a walkthrough for all the assembly steps and a video for assembling the bracket.

Assembly Instructions

Circuit Board & Housing

The control board needs to be assembled to meet your space constraints and number of blinds but I can give some general guidance on layout and assembly. This is a fairly straightforward circuit but the motor drivers have quite a few IO’s; I would recommend prototyping this on a breadboard to get familiar with the components before finalizing it on a protoboard.

IMG_9039.jpeg

I arranged my board with the arduino nano on top with the digital pin heavy side facing the motor drivers. I also positioned the motor connectors directly next to the motor outputs on the drivers to avoid running that wire set too far. The two endstop connectors are positioned in the top left.

I would strongly recommend bringing power into the board using a barrel jack connector. It makes powering way easier plus the Nano can be powered with 6-20V on it’s Vin pin so no need for an extra voltage regulator. I also installed headers in to the protoboard where the drivers and Nano plug in so I don’t have to solder them directly. This is really nice because if you ever need to swap these or need them for another project, you can just pop them in and out, no need for soldering! The final board with wires run is shown below. To help run the stepper motors more smoothly you should also insert a 100uF capacitor upstream of the drivers.

IMG_9041.jpeg

If you’re using this exact protoboard layout, you can download and print my housing to mount on a wall. The circuit board screws into the backplane with 4 M3x8 socket head screws. The cover then just slides over the lip around the base. See the exploded view below.

Untitled.png

Wiring Diagram

The wiring diagram for the protoboard can be found below. Note that limitations in the layout software forced me to show an A4988 driver instead of the TMC2208 and a conventional Arduino Nano Rev3 instead of a Nano 33 IoT. There is no difference in wiring the Nano board however there are some differences between the A4988 and TMC2208 wiring. The TMC2208 only has an MS1 and MS2 pin, no MS3 pin. Both should be pulled high to enable the proper micro steps. The TMC2208 also includes a CLK (clock) pin which can be kept disconnected. The PDN / UART connections can also be kept disconnected.

Software

Arduino Sketches

The following code will work on all wifi enabled Arduino boards. You’ll need two tabs in your sketch, one containing the main loop, and one “arduino_secrets.h” file for your wifi SSID and password (note: you should connect your Arduino to a 2.4GHz network). This sketch also requires that the WiFiNINA library and AccelStepper library are installed in your Arduino IDE.

When adapting the code for your setup make sure you update, IPAddress, closedPosition, blindSpeed, and blindAcceleration.

IPAddress is self explanatory, it’s the IP your assigning the Arduino to on your network. This allows HomeBridge to connect to the Arduino properly on boot up. Make sure you pick an IP address that isn’t already used by another device!

closedPosition sets the number of steps from the zero position that are needed to get to the fully closed position. For my window this was 52,000 but you should test to find the value for your specific window.

blindSpeed sets the speed in Steps/min that the motors will travel. The default value of 2000 should run the motors silently but you can definitely speed this up at the cost of increased noise

blindAcceleration sets the motor acceleration. I generally keep this equal to the speed value (meaning a 1 second ramp to max speed) but it can be adjusted at will.

The code for the main control loop is below:

    
#include <WiFiNINA.h>
#include "arduino_secrets.h"
#include "AccelStepper.h"

// Define motor controller connections and motor interface type (TMC2208)
#define dirPinA 2
#define stepPinA 3
#define enPinA 4

#define dirPinB 5
#define stepPinB 6
#define enPinB 7

#define TMC2208 1

// Define endstops
#define endstopA 8 
#define endstopB 9

//Open and Closed Positions
int closedPosition = 52000; //Closed position, in number of steps from home.  Needs to be measured manually
int openPosition = 0; //Open position (home)

//Define Movement Speeds (note higher speeds will be louder)
int blindSpeed = 2000;
int blindAcceleration = 2000;

//Define assigned IP address
IPAddress ip(192, 168, 1, 202);

// Create control variables
long homePosA=-1;  // Used to home motor A
long homePosB=-1;  // Used to home motor B
int completedHome = 0; //checks if a home was completed before allowing open and close actions
int target = openPosition; //target variable passed from HomeBridge
int state = 2; //0-moving, 2-idling

// Create motor instances in AccelStepper
AccelStepper stepperA = AccelStepper(TMC2208, stepPinA, dirPinA);
AccelStepper stepperB = AccelStepper(TMC2208, stepPinB, dirPinB);

// Create Server
char ssid[] = SECRET_SSID;             //  your network SSID (name) between the " "
char pass[] = SECRET_PASS;      // your network password between the " "
int keyIndex = 0;                 // your network key Index number (needed only for WEP)
int status = WL_IDLE_STATUS;      //connection status
WiFiServer server(80);            //server socket
WiFiClient client = server.available();

void setup() {
  
  //Begin serial connection
  Serial.begin(9600);  //Begin serial connection

  // Setup endstop inputs
  pinMode(endstopA, INPUT_PULLUP);
  pinMode(endstopB, INPUT_PULLUP);
  
  //Setup motor driver enable pins
  pinMode(enPinA, OUTPUT);
  pinMode(enPinB, OUTPUT);

  //Set movement speeds
  stepperA.setMaxSpeed(blindSpeed);
  stepperA.setAcceleration(blindAcceleration);
  stepperB.setMaxSpeed(blindSpeed);
  stepperB.setAcceleration(blindAcceleration);

  //enable both stepper drivers
  enableMotors();

  //Home blinds on start up
  homeBlinds();

  //enable wifi and start server
  WiFi.config(ip);
  enable_WiFi();
  connect_WiFi();
  server.begin();
  printWifiStatus();

}

void loop() {
  if (stepperA.currentPosition() == target && target < (0.1*closedPosition)){       //turn off motors if they're less than 10% open to save power
    disableMotors();
  }
  else{
    enableMotors();
  }
  
  moveOneStep();                                                                    //step both motors if a step is due

  client = server.available();

  if (client) {
    runServer();
  }
}

void enable_WiFi() {
  // check for the WiFi module:
  if (WiFi.status() == WL_NO_MODULE) {
    Serial.println("Communication with WiFi module failed!");
    // don't continue
    while (true);
  }

  String fv = WiFi.firmwareVersion();
  if (fv < "1.0.0") {
    Serial.println("Please upgrade the firmware");
  }
}

void moveOneStep() {
  stepperA.run();
  stepperB.run();
}

void connect_WiFi() {
  // attempt to connect to Wifi network:
  while (status != WL_CONNECTED) {
    Serial.print("Attempting to connect to SSID: ");
    Serial.println(ssid);
    // Connect to WPA/WPA2 network. Change this line if using open or WEP network:
    status = WiFi.begin(ssid, pass);
    // wait 10 seconds for connection:
    delay(10000);
  }
}

void printWifiStatus() {
  // print the SSID of the network you're attached to:
  Serial.print("SSID: ");
  Serial.println(WiFi.SSID());

  // print your board's IP address:
  IPAddress ip = WiFi.localIP();
  Serial.print("IP Address: ");
  Serial.println(ip);

  // print the received signal strength:
  long rssi = WiFi.RSSI();
  Serial.print("signal strength (RSSI):");
  Serial.print(rssi);
  Serial.println(" dBm");

  Serial.print("IP Address: http://");
  Serial.println(ip);
}

void homeBlinds() {
  // Home both motors if neither are home
  while (digitalRead(endstopA) == 1 && digitalRead(endstopB) == 1) {
    stepperA.moveTo(homePosA);
    stepperB.moveTo(homePosB);
    homePosA--;
    homePosB--;
    stepperA.run();
    stepperB.run();
  }

  // Home stepper B only if stepper B isn't home
  while (digitalRead(endstopA) != 1 && digitalRead(endstopB) == 1) {
    stepperB.moveTo(homePosB);
    homePosB--;
    stepperB.run();
  }

  // Home stepper A only if stepper A isn't home
  while (digitalRead(endstopA) == 1 && digitalRead(endstopB) != 1) {
    stepperA.moveTo(homePosA);
    homePosA--;
    stepperA.run();
  }

  //reset positions temporarily
  stepperA.setCurrentPosition(0);
  stepperB.setCurrentPosition(0);

  //back off endstops
  stepperA.moveTo(500);
  stepperB.moveTo(500);
  while (stepperA.currentPosition() != 500){
    stepperA.run();
    stepperB.run();
  } 

  //Set open position for both motors (0)
  stepperA.setCurrentPosition(openPosition);
  stepperB.setCurrentPosition(openPosition);  
  target = openPosition;
}

void setTarget(int scaledTarget) {
  target = map(scaledTarget, 100, 0, 0, closedPosition);  //map the scaled target from homebridge to 0 - closedPosition scale for motors
  stepperA.moveTo(target);                                //set new target for motors A and B
  stepperB.moveTo(target);
}

int getPosition() {
  return (map(stepperA.currentPosition(), 0, closedPosition, 100, 0));  //map the current position (in steps) to 0-100 scale for homebridge
}

int getState(){
  if (stepperA.currentPosition() < target) {
    return 0;                                     //if moving up, report opening
  } 
  if (stepperA.currentPosition() > target) {
    return 1;                                     //if moving down, report closing
  }
  else {
    return 2;                                     //otherwise report idle
  }
}


int enableMotors() {
  //pulls both motor driver enable pins low (enables current to motors)
  digitalWrite(enPinA, LOW);
  digitalWrite(enPinB, LOW);
}

int disableMotors() {
  //pulls both motor driver enable pins high (disables current to motors)
  digitalWrite(enPinA, HIGH); 
  digitalWrite(enPinB, HIGH);
}

void runServer() {
  if (client) {                             // if you get a client,
    String currentLine = "";                // make a String to hold incoming data from the client
    while (client.connected()) {            // loop while the client's connected
      if (client.available()) {             // if there's bytes to read from the client,
        char c = client.read();             // read a byte, then
        if (c == '\n') {                    // if the byte is a newline character
          if (currentLine.length() == 0) {
            //The HTTP response ends with another blank line:
            client.println();
            // break out of the while loop:
            break;
          }
          else {      // if you got a newline, then clear currentLine:
            currentLine = "";
          }
        }
        else if (c != '\r') {    // if you got anything else but a carriage return character,
          currentLine += c;      // add it to the end of the currentLine
        }

        if (currentLine.endsWith("/current_position")) {
          client.println("HTTP/1.1 200 OK");            //respond to request
          client.println("Content-type: text/plain");   //prepare response
          client.println();
          client.println(getPosition());                //give current motor position
        }
        if (currentLine.indexOf("set/") > 0) {
          String parameter = "";
          c = client.read();
          while (c != '\n') {
            Serial.write(c);
            parameter += c;
            c = client.read();
          }
          int scaledTarget = parameter.toInt();
          client.println("HTTP/1.1 200 OK");            //respond to request
          client.println("Content-type: text/plain");   //prepare repsonse
          client.println();
          client.println(parameter);
          setTarget(scaledTarget);                      //set new target position (0-100 scale)
          currentLine = "";                             // We overrode the newline check, so we have to reset ourselves
        }
        if (currentLine.endsWith("/current_state")) {
          client.println("HTTP/1.1 200 OK");            //respond to request
          client.println("Content-type: text/plain");   //prepare response
          client.println();
          client.println(getState());                   //report state (opening/closing/idle)
        }
      }
    }
    // close the connection:
    client.stop();
  }
}
    
  

The arduinosecrets.h file should be setup as shown below. Replace YOUR_SSID and YOUR_PASSWORD with the WiFi SSID and password for the network you’d like to connect to!

    
#define SECRET_SSID "YOUR_SSID"
#define SECRET_PASS "YOUR_PASSWORD"
    
  

HomeBridge Setup

I won’t be covering how to setup a HomeBridge server but there is great documentation here. Once you’ve setup the server you can add it to your Home app and install the http-minimal-blinds plugin. You should use this JSON configuration for the plugin to work with the Arduino sketch. The JSON is pasted in the Config menu of HomeBridge under accessories.

    
    "name": "Bedroom Blinds",
    "accessory": "MinimalisticHttpBlinds",
    "get_current_position_url": "http://IPAddress/get/current_position/",
    "set_target_position_url": "http://IPAddress/set/%position%",
    "get_current_state_url": "http://IPAddress/get/current_state/",
    "get_current_position_polling_millis": 5000,
    "get_current_state_polling_millis": 5000,
    "set_target_position_expected_response_code": 200
    
  

Make sure you replace IPAddress with, you guessed it, the IP address, the same one you assigned in the Arduino sketch. Once HomeBridge reboots and the configuration takes effect, the Arduino Nano should be able to connect to the home app. You should be able to confirm this in the system logs if no errors are printed.

Final Product

You can see the final product in action below!