Curtain Control Node.
-
My First version of curtain control Node is ready, installed and tested. Some work is pending, which I'll complete on my weekend.
Edit: Code for the Node:Curtain Node
Curtain Control Node.Please share your feedback so that I can improvise.
-
What type of motor are you using?
-
Its a Geared DC Motor with Voltage Rating of 6-12V and 100 RPM. I am using a L239 to control the motor.
-
Looks good. I am curious to see the details... jealous for the speed your curtain is moving.
I have been working on a similar project last week. The mechanics and ´looks´ are the most difficult part for me.
For now I have used a ´tuned´ cheap & small stepper which makes it slooow but effective The rail and mounting is standard IKEA. The pro-mini has the jModule attached. Works like a charm (/snail)!
-
@AWI Thanks. Why dont you increase the pulley size (larger pulley) that will increase the speed.
I found driving stepper motor to be bit more complicated than driving a DC motor. I picked the Geared motor basis the calculation using this calculator.
Since without rails pulling the curtains on a metal pipe needed more torque. Thus I chose 100 RPM to be a suitable fit.
Will share the Schematics and PCB layout files soon. I have already added the sketch for reference. Since I was testing I did not put the circuit in any housing. But will give some shape soon.
Also will be adding 3 buttons for Manual Control.
-
DC Motor, very interesting!
Do you have end stops? The sketch doesn't open for some reason...
Cheers
-
@barduino I tried downloading the code, it works. Anyways I am pasting the code below.
For end stops I have not yet decided. I had some Ideas of using Hall effect sensor on both ends and a small piece of magnet in the string to detect ends or Maybe use stop switches. Not sure but will have to try both and see which works the best.
I am yet to clean the code I have some debugging serial prints in code which will remove once finalized.
#include <SPI.h> #include <MySensor.h> int motor_forward = 5; int motor_reverse = 6; int LDRinput = 1; //analog pin to which LDR is connected, here we set it to 0 so it means A0 int LDRValue = 0; //that�s a variable to store LDR values int light_sensitivity = 500; //This is the approx value of light surrounding your LDR int stopSwLeft = 4; int stopSwRight = 3; int stopSwNow = 2; int CHILD_CURTAIN_ID =1; int lastState; MySensor gw; MyMessage msg_S_COVER_U(CHILD_CURTAIN_ID, V_UP); MyMessage msg_S_COVER_D(CHILD_CURTAIN_ID, V_DOWN); MyMessage msg_S_COVER_S(CHILD_CURTAIN_ID, V_STOP); //MyMessage msg(CHILD_CURTAIN_ID,V_UP); // the setup routine runs once when you press reset: void setup() { Serial.begin(115200); // initialize the digital pin as an output for L239D. pinMode(motor_forward, OUTPUT); pinMode(motor_reverse, OUTPUT); // initialize the digital pin as an output for Stop Switches. pinMode(stopSwLeft, INPUT_PULLUP); pinMode(stopSwRight, INPUT_PULLUP); pinMode(stopSwNow, INPUT_PULLUP); gw.begin(incomingMessage, AUTO, true); // Send the sketch version information to the gateway and Controller gw.sendSketchInfo("Window_Curtain", "1.0"); gw.present(CHILD_CURTAIN_ID, S_COVER); } void loop(){ gw.process(); } void cover(int coverVal){ //int coverVal = gw.loadState(CHILD_CURTAIN_ID); Serial.print("Cover is : "); lastState = coverVal; switch (coverVal) { case 0: Serial.println("Opening"); while (lastState == coverVal) { m_left(); gw.process(); checkHWInnputs(); coverVal = gw.loadState(CHILD_CURTAIN_ID); } gw.send(msg_S_COVER_U.set(V_UP)); break; case 1: Serial.println("Closing"); while (lastState == coverVal) { m_right(); gw.process(); checkHWInnputs(); coverVal = gw.loadState(CHILD_CURTAIN_ID); } gw.send(msg_S_COVER_D.set(V_DOWN)); break; case 2: Serial.println("Idle"); while (lastState == coverVal) { m_stop(); gw.process(); checkHWInnputs(); coverVal = gw.loadState(CHILD_CURTAIN_ID); } gw.send(msg_S_COVER_S.set(V_STOP)); break; } return; } void incomingMessage(const MyMessage &message) { // We only expect one type of message from controller. But we better check anyway. Serial.println("recieved incomming message"); switch (message.type) { case V_UP: gw.saveState(CHILD_CURTAIN_ID, 0); Serial.print("Incoming change for ID_S_COVER:"); Serial.print(message.sensor); Serial.print(", New status: "); Serial.println("V_UP"); cover(gw.loadState(CHILD_CURTAIN_ID)); Serial.print("Done cover procedure"); //m_right(); break; case V_DOWN: gw.saveState(CHILD_CURTAIN_ID, 1); Serial.print("Incoming change for ID_S_COVER:"); Serial.print(message.sensor); Serial.print(", New status: "); Serial.println("V_DOWN"); cover(gw.loadState(CHILD_CURTAIN_ID)); Serial.print("Done cover procedure"); //m_left(); break; case V_STOP: gw.saveState(CHILD_CURTAIN_ID, 2); Serial.print("Incoming change for ID_S_COVER:"); Serial.print(message.sensor); Serial.print(", New status: "); Serial.println("V_STOP"); cover(gw.loadState(CHILD_CURTAIN_ID)); Serial.print("Done cover procedure"); //m_stop(); break; } Serial.print("exiting incoming message"); return; } // the loop routine runs over and over again forever: void m_right() { digitalWrite(motor_forward, HIGH); //terminal D1 will be HIGH digitalWrite(motor_reverse, LOW); //terminal D2 will be LOW Serial.print("Set For:High and Rev: Low"); exit; } void m_left() { digitalWrite(motor_forward, LOW); //terminal D1 will be LOW digitalWrite(motor_reverse, HIGH); //terminal D2 will be HIGH Serial.print("Set For:Low and Rev: High"); exit; } void m_stop() { digitalWrite(motor_forward, LOW); //terminal D1 will be LOW digitalWrite(motor_reverse, LOW); //terminal D2 will be HIGH Serial.print("Set For:Low and Rev: Low"); exit; } void checkHWInnputs() { if (digitalRead(stopSwNow) == LOW) { Serial.println("Detected stop button push. Stopping"); cover(2); Serial.println("Saved stop state to EEPROM"); } if (digitalRead(stopSwLeft) == LOW) { Serial.println("Detected stop button push. Stopping"); cover(1); Serial.println("Saved stop state to EEPROM"); } if (digitalRead(stopSwRight) == LOW) { Serial.println("Detected stop button push. Stopping"); cover(0); Serial.println("Saved stop state to EEPROM"); } }
-
Sweet!
My girl has put this on my todolist...
But still have to get started, this is some nice reference for my new porject!
-
@franzelare Great. Good Luck with the build.
I'll be posting my HW design and Code as part of Contest on openhardware.io
-
I have published my project with all the files.
https://www.openhardware.io/view/45/Curtain-Control-Node
-
I'm just curious, maybe its the arduino IDE version or something.
I'm using 1.6.4 and I get this:
However opening with a text editor is fine:
Cheers
-
@barduino Seems like the IDE. Open any example MySensors Sketch. Delete the contents from sketch and copy paste from my sketch. I guess that should work.
Are you able to compile any other sketch in Arduino?
-
@AWI Are you able to share the sketch you used with the UL2003 driver board? I have tried to combine the MySensors servo sketch with other arduino only sketches utilizing the UL2003, but without luck. Thanks.
-
@patchmaster I used the relay sketch and modified it to my needs. Sorry but I dont have any sketch for ULN2003.
-
@patchmaster This is my current curtain control sketch. I work with the development branch, so the syntax can be a little different.
The Accelstepper library gives you all the abilities to slow start/stop with numerous kinds of steppers.
My sketch includes a calibration function (stored in EEPROM) so the you don't need end stop detection (a stepper knows where it is)/* PROJECT: MY Sensors curtain controller PROGRAMMER: AWI DATE: march 11, 2016 FILE: AWI stepper1.ino LICENSE: Public domain Hardware: ATMega328p board w/ NRF24l01 and MySensors 2.0 (Development) Special: uses AccelStepper library Summary: Curtain control with stepper motor. Manual operation with 1 push button Calibration with manual button Remarks: Fixed node-id Change log: 20160312 - Cleanup */ // Enable debug prints to serial monitor #define MY_DEBUG #define MY_NODE_ID 13 // fixed node number // Enable and select radio type attached #define MY_RADIO_NRF24 //#define MY_RADIO_RFM69 #include <SPI.h> #include <MySensor.h> // stepper library #include <AccelStepper.h> // http://www.airspayce.com/mikem/arduino/AccelStepper/ #define HALFSTEP 8 // Stepper uses "Halfstep" mode // button library // used for: // - manual open close - single click: close/ stop/ open/ stop // - calibration - after long press: open calibarion- until single press - closed calibration - until single press (stop) #include <Button.h> // https://github.com/JChristensen/Button #define CHILD_ID 1 // Id of the sensor child #define SN "Curtain control 13" #define SV "1.0" #define buttonPin1 7 // Arduino pin connected to buttonPin1 #define buttonPin2 A0 // Arduino pin connected to buttonPin2 (fixed to ground) // Motor pin definitions #define motorPin1 3 // IN1 on the ULN2003 driver 1 #define motorPin2 4 // IN2 on the ULN2003 driver 1 #define motorPin3 5 // IN3 on the ULN2003 driver 1 #define motorPin4 6 // IN4 on the ULN2003 driver 1 const unsigned long heartbeatInterval = 1 * 3600UL * 1000UL ; // heartbeatinterval unsigned long heartbeatCounter = 0 ; // // helper routines to store and retrieve long in mysensors EEPROM union { // used to convert long to bytes for EEPROM storage long longInt; uint8_t LongByte[4]; } convLongInt ; void saveStateL(int EEposition, long StateL){ convLongInt.longInt = StateL ; for (int y = 0; y < 4 ; y++){ // convert to bytes saveState(EEposition + y , convLongInt.LongByte[y]) ; } Serial.print("State saved: "); Serial.println(StateL); } long loadStateL(int EEposition){ for (int y = 0; y < 4 ; y++){ // convert from bytes convLongInt.LongByte[y] = loadState(EEposition + y) ; } Serial.print("State read: "); Serial.println(convLongInt.longInt); return convLongInt.longInt ; } // Initialize with pin sequence IN1-IN3-IN2-IN4 for using the AccelStepper with 28BYJ-48 AccelStepper stepper1(HALFSTEP, motorPin1, motorPin3, motorPin2, motorPin4); // Initialize button active low, debounce and internal pull-up Button myBtn(buttonPin1, true, true, 40); // Initiate the button (pin, pull_up, invert, debounce_ms) MyMessage percentageMsg(CHILD_ID, V_PERCENTAGE); // used to send updates to controller const long maxRun = 4000000L ; // maximum runway long setPosition = 0 ; // remembers set position, need to be saved in EEPROM const int setPositionEE = 4 ; // eeprom location long openPosition = 0 ; // Position at open, need to be saved in EEPROM? const int openPositionEE = setPositionEE + 4 ; // eeprom location long closedPosition = 120000UL ; // Position at full close, need to be saved in EEPROM const int closedPositionEE = openPositionEE + 4 ; // eeprom location unsigned long idleTimer = millis() ; // return to idle timer unsigned long idleTime = 100000UL ; // return to idle after 100 secs unsigned long printTimer = millis() ; // print timer unsigned long printTime = 1000UL ; // print after 1 secs enum position_t {Open, Close, Idle, Running} ; position_t lastDirection = Open ; // lastDirection only for buttonpress position_t runStatus = Idle ; // indicates current status for running motor. used for status reporting to controller enum State_t {sIdle, sCalibrateOpen, sCalibrateClose} ; State_t State = sIdle ; void setup() { // setup buttons pinMode(buttonPin1, OUTPUT); stepper1.setMaxSpeed(2000.0); stepper1.setAcceleration(1000.0); //saveStateL(closedPositionEE, closedPosition) ; // INIT: save closed position in EEPROM closedPosition = loadStateL(closedPositionEE) ; // need to get last values from EEPROM and assume the current position is correct setPosition = loadStateL(setPositionEE) ; stepper1.setCurrentPosition(setPosition ); }//--(end setup )--- void presentation() { present(CHILD_ID, S_COVER, "Curtain"); // Window Cover sub-type, commands: V_UP, V_DOWN, V_STOP // Register the LED Dimmable Light with the gateway sendSketchInfo(SN, SV); } void loop() { unsigned int now = millis() ; // current time for loop // simple state machine for button press myBtn.read(); switch (State) { // Idle state, waiting for some action // - button press // - idleTimer case sIdle: if (myBtn.wasReleased()){ // idle Serial.println("Button release") ; // if running stop if (stepper1.isRunning()){ setPosition = stepper1.currentPosition(); stepper1.moveTo(setPosition) ; // move to current position (was already there..) } else if (lastDirection == Open) { stepper1.moveTo(closedPosition) ; lastDirection = Close ; } else { // lastDirection == Close stepper1.moveTo(openPosition) ; lastDirection = Open ; } } else if (myBtn.pressedFor(3000)){ // move to calibratete state with long press Serial.println("Button press long") ; idleTimer = now ; // return to idle after ... State = sCalibrateOpen ; stepper1.move(-maxRun) ; // let the stepper open with maximum } break ; // if not running and last action was open close ; else open // if longpress Calibrate open case sCalibrateOpen: // calibration going on if (myBtn.wasPressed()){ stepper1.setCurrentPosition(0 ); // set new 0 position ?? openPosition = setPosition = 0 ; State = sCalibrateClose ; // next is close calibarion stepper1.move(maxRun) ; // let the stepper close with maximum } else if (now > idleTimer + idleTime) { // timer expired -> abort calibration State = sIdle ; } break ; case sCalibrateClose: // calibrate closed position, end with keypress if (myBtn.wasPressed()) { closedPosition = setPosition = stepper1.currentPosition() ; saveStateL(closedPositionEE, closedPosition) ; // save closed position in EEPROM State = sIdle ; stepper1.moveTo(openPosition) ; // return to open after calibration } else if (now > idleTimer + idleTime) { // timer expired -> abort calibration State = sIdle ; } break ; default : break ; } // power off stepper if not running (no need to reenable)) if (!stepper1.isRunning()){ if (runStatus != Idle){ // there was a change in runningstatus, so report to controller setPosition = stepper1.currentPosition() ; // store in EEPROM and report ready to controller saveStateL(setPositionEE, setPosition) ; send( percentageMsg.set((100 * setPosition)/(closedPosition - openPosition))) ; runStatus = Idle ; } stepper1.disableOutputs(); } else { runStatus = Running ; } stepper1.run(); /* if (printTimer++ > now + printTime){ printTimer = now ; Serial.println(stepper1.currentPosition()); } */ } // This is called when a message is received void receive(const MyMessage &message) { // We only expect few types of messages from controller, check which switch (message.type) { case V_PERCENTAGE: // Curtain should be opened stepper1.moveTo(message.getInt() * (closedPosition - openPosition)/100); Serial.print("Message: "); Serial.print(message.sensor); Serial.print(" , value: % "); Serial.println( message.getInt()); Serial.print("Moving to: "); Serial.println(message.getInt() * (closedPosition - openPosition)/100); break ; case V_STATUS: // Curtain should be opened or closed full stepper1.moveTo((message.getInt() == HIGH)?openPosition:closedPosition); Serial.print("Message - valid: "); Serial.print(message.sensor); Serial.print(" , value: % "); break ; default : // not recognizable message Serial.print("Message - valid: "); Serial.print(message.sensor); Serial.print(", Unrecognized "); break ; } }
-
@AWI Thanks. I have been using the Accelstepper library, but this has already given me several other ideas.
I appreciate the help.
-
@AWI
I have been searching for something like this for a while, thank you for the code!But i have some problems compiling it
C:\Users\ToreR\Documents\Arduino\stepper_mysensor\stepper_mysensor.ino: In function 'void saveStateL(int, long int)': stepper_mysensor:74: error: 'saveState' was not declared in this scope saveState(EEposition + y , convLongInt.LongByte[y]) ; ^ C:\Users\ToreR\Documents\Arduino\stepper_mysensor\stepper_mysensor.ino: In function 'long int loadStateL(int)': stepper_mysensor:81: error: 'loadState' was not declared in this scope convLongInt.LongByte[y] = loadState(EEposition + y) ; ^ C:\Users\ToreR\Documents\Arduino\stepper_mysensor\stepper_mysensor.ino: In function 'void presentation()': stepper_mysensor:129: error: 'present' was not declared in this scope present(CHILD_ID, S_COVER, "Curtain"); // Window Cover sub-type, commands: V_UP, V_DOWN, V_STOP ^ stepper_mysensor:132: error: 'sendSketchInfo' was not declared in this scope sendSketchInfo(SN, SV); ^ C:\Users\ToreR\Documents\Arduino\stepper_mysensor\stepper_mysensor.ino: In function 'void loop()': stepper_mysensor:195: error: 'send' was not declared in this scope send( percentageMsg.set((100 * setPosition)/(closedPosition - openPosition))) ; ^ exit status 1 'saveState' was not declared in this scope```
-
@Tore-André-Rosander are you using the development branch? The errors seem to indicate that is the case.
-
@AWI Thank you, that helped
I now have it up and running.But how do i connect and use this with a controller (like Domoticz)?
I use an ESP gateway (from the dev branch) and the curtain node is connected and showing up in domoticz.
But how do i make domoticz send commands to the curtain node?All i get when adding the curtain node both as on/off button and blinds in domoticz is "unreconized" in the curtain serial monoitor.
-
@Tore-André-Rosander I have defined it as "Blinds Percentage" in Domoticz. This send the V_STATUS and V_PERCENTAGE messages where the node should respond to...
-
@AWI THanks again for your fast replys!
This is what i get in serial monitor when i use the "up" "down" buttons for blinds percentage in domoticz.
This is my very first MySensor setup so it might be some error with my setup, i will have a deeper look at it later tonight.
Starting sensor (RNNNA-, 2.0.0-beta) Radio init successful. State read: -1 State read: -1 send: 13-13-0-0 s=255,c=3,t=15,pt=0,l=2,sg=0,st=ok: send: 13-13-0-0 s=255,c=0,t=17,pt=0,l=10,sg=0,st=ok:2.0.0-beta send: 13-13-0-0 s=255,c=3,t=6,pt=1,l=1,sg=0,st=ok:0 read: 0-0-13 s=255,c=3,t=15,pt=0,l=2,sg=0: read: 0-0-13 s=255,c=3,t=6,pt=0,l=1,sg=0:M send: 13-13-0-0 s=1,c=0,t=5,pt=0,l=7,sg=0,st=ok:Curtain send: 13-13-0-0 s=255,c=3,t=11,pt=0,l=18,sg=0,st=ok:Curtain control 13 send: 13-13-0-0 s=255,c=3,t=12,pt=0,l=3,sg=0,st=ok:1.0 Init complete, id=13, parent=0, distance=1 read: 0-0-13 s=1,c=1,t=29,pt=0,l=0,sg=0: Message - valid: 1, Unrecognized read: 0-0-13 s=1,c=1,t=30,pt=0,l=0,sg=0: Message - valid: 1, Unrecognized read: 0-0-13 s=1,c=1,t=29,pt=0,l=0,sg=0: Message - valid: 1, Unrecognized read: 0-0-13 s=1,c=1,t=30,pt=0,l=0,sg=0: Message - valid: 1, Unrecognized read: 0-0-13 s=1,c=1,t=29,pt=0,l=0,sg=0: Message - valid: 1, Unrecognized read: 0-0-13 s=1,c=1,t=30,pt=0,l=0,sg=0: Message - valid: 1, Unrecognized read: 0-0-13 s=1,c=1,t=29,pt=0,l=0,sg=0: Message - valid: 1, Unrecognized read: 0-0-13 s=1,c=1,t=30,pt=0,l=0,sg=0: Message - valid: 1, Unrecognized read: 0-0-13 s=1,c=1,t=29,pt=0,l=0,sg=0: Message - valid: 1, Unrecognized read: 0-0-13 s=1,c=1,t=30,pt=0,l=0,sg=0: Message - valid: 1, Unrecognized```
-
@Tore-André-Rosander Nothing wrong with you interpretation. I have not written this sketch as a complete S_COVER implementation. Only the V_PERCENTAGE part (not V_UP/V_DOWN). If you define it in Domoticz as "Blinds percentage" (see my previous post) it will work. Sorry for the confusion.
-
@AWI Seems like my problem is with Domoticz itselfe.
I have the same problem as described in this post https://www.domoticz.com/forum/viewtopic.php?f=6&t=7748
-
@Tore-André-Rosander You can probably solve it by deleting the sensor in Domoticz and let the sketch present itself as S_DIMMER (and use another sensor id)
-
Maybe its because i have not calibrated it? I dont have any pushbuttons laying around, so im gonna make one with some wires. But i see that there is two pins defined as buttons. Should i use the buttonPin1 as VCC?
So it would look something like buttonPin1 -> buttonPin2 -> button -> gnd?
-
@Tore-André-Rosander You can ignore buttonpin2. Just connect your "button" to buttonpin1 and ground.
-
@AWI I'm curious where you got the rails for the curtain. I wasn't been able to find them, but stumbled on this topic.
-
@TheoL Ikea
-
@AWI Really interested in the mechanics
-
I'm currently working on a similar project, only I got started with 24V DC motors that I got from my work (left overs from a test project) These motors are very strong (way way over rated) and have a big gearbox what unfortunately makes the movement a bit slow.
I started off with 1 motor to move 2 curtains at the same time, but will change this to 2 motors for either side since the better half of me likes to control the curtains separately in case she only wants to close 1.
Next to the automation there will also be a manual control switch be hidden behind the curtains in case the node is down and I'm not at home to fix the issue...
all running on an ESP8266 (ESP-12e) unit to have it over WiFi and have enough ioCurrently the test setup is working fine and I'm ordering parts to start implementing this a bit bigger in the first room. the test setup I made with some old wood 1 meter of rail and a few old rugs as curtains...
I use a GT2 timing belt that is commently used for 3D printers and pully's are widely available as well as the belts in different lengths (I order them on rolls of 10 meters) and attached the timing belt to a carriage that is normally to pull the curtains with a rod.
the motor is mounted on a bracket with currently 2 end stop switches for open and close (but will be 1 switch for open and 1 for close in the final solution)
resulting in driving the curtain from 1 side
electronics are working fine now and need to be ported to a test print and later to a real PCB design
-
Hello.
To my project i use stepper motor JK42HS34-0404 and driver L298N.
This is a wiring diagram.
This is a program code that I use
#include <MySensor.h> #include <SPI.h> #include <AccelStepper.h> #define HALFSTEP 4 // Stepper uses "Halfstep" mode #define CURTAIN_CLOSED 1000 // value when closed #define CURTAIN_OPEN 0 // value when open #define CHILD_ID 1 // Id of the sensor child // MySensors definitions MySensor gw; // Initialize message MyMessage msg(CHILD_ID, V_TRIPPED); // Motor pin definitions #define motorPin1 3 // IN1 on the ULN2003 driver 1 #define motorPin2 4 // IN2 on the ULN2003 driver 1 #define motorPin3 5 // IN3 on the ULN2003 driver 1 #define motorPin4 6 // IN4 on the ULN2003 driver 1 // Initialize with pin sequence IN1-IN3-IN2-IN4 for using the AccelStepper with 28BYJ-48 AccelStepper stepper1(HALFSTEP, motorPin1, motorPin2, motorPin3, motorPin4); unsigned long current_position ; // remembers current position, need to be saved in EEPROM? void setup() { // MySensors gw.begin(); // fixed node 13 gw.sendSketchInfo("Curtain control 13", "1.0"); // Send the sketch version information to the gateway and Controller //pinMode(DIGITAL_INPUT_SENSOR, INPUT); // sets the motion sensor digital pin as input // Register all sensors to gw (they will be created as child devices) gw.present(CHILD_ID, S_COVER); // Window Cover sub-type, commands: V_UP, V_DOWN, V_STOP stepper1.setMaxSpeed(500.0); stepper1.setAcceleration(700.0); stepper1.moveTo(CURTAIN_OPEN); current_position = CURTAIN_OPEN ; stepper1.moveTo(CURTAIN_OPEN); }//--(end setup )--- void loop() { // if message = V_UP start moving until closed // if message = V_DOWN start moving back until open // if message = V_STOP stop moving // Test: Change direction when the stepper reaches the target position gw.process(); // check if message from controller if (stepper1.distanceToGo() == 0){ if (stepper1.currentPosition() == CURTAIN_OPEN){ stepper1.moveTo(CURTAIN_CLOSED); Serial.println("Curtain Open, now closing"); gw.request(CHILD_ID, V_UP, 0); // request new values from controller } else{ stepper1.moveTo(CURTAIN_OPEN); Serial.println("Curtain Closed, now opening"); gw.request(CHILD_ID, V_DOWN, 0); // request new values from controller } } stepper1.run(); } // This is called when a message is received void incomingMessage(const MyMessage &message) { // We only expect few types of messages from controller, check which stepper1.moveTo(message.getInt() * CURTAIN_CLOSED/100); Serial.print("Message - valid: "); Serial.print(message.sensor); Serial.print(" , value: % "); Serial.println(message.getInt()); switch (message.getInt()) { case 100: // Curtain should be opened Serial.print("Message - valid: "); Serial.print(message.sensor); Serial.print(", Message UP "); stepper1.moveTo(CURTAIN_OPEN); break; case 0: // Curtain should be closed Serial.print("Message - valid: "); Serial.print(message.sensor); Serial.print(", Message DOWN "); stepper1.moveTo(CURTAIN_CLOSED); break; case 50: // Curtain action should be stopped Serial.print("Message - valid: "); Serial.print(message.sensor); Serial.print(", Message STOP "); break; default: // not recognizable message Serial.print("Message - valid: "); Serial.print(message.sensor); Serial.print(", Unrecognized "); } }
When the program I have uploaded to my mysensors, the engine still turns, one left and one to the right and did not work the button blinds in Domoticz.
I cant turn off my stepper motor in Domoticz.
Stepper motor on MySensors in Domoticz – 01:25
— Roman KubatPlease help me.
-
I write a program on library Stepper not AccelStepper.
Now is working.Program code:
// Enable debug prints to serial monitor #define MY_DEBUG // Enable and select radio type attached #define MY_RADIO_NRF24 //#define MY_RADIO_RFM69 // Enable repeater functionality for this node #define MY_REPEATER_FEATURE #include <SPI.h> #include <MySensors.h> #include <Stepper.h> #define CURTAIN_CLOSED 1000 // wartosc gdy kurtyna zamknieta #define CURTAIN_OPEN -1000 // wartosc gdy kurtyna otwarta #define CHILD_ID 1 // definicje MySensors MyMessage message(CHILD_ID, S_COVER); int in1Pin = 3; int in2Pin = 4; int in3Pin = 5; int in4Pin = 6; int lastState; // liczba kroków na jeden obrót #define stepsPerRevolution 200 //ustawienie szybkości silnika int motorSpeed = 40; Stepper myStepper(stepsPerRevolution, in1Pin, in2Pin, in3Pin, in4Pin); void setup() { // ustawienie pinów jako wyjście pinMode(in1Pin, OUTPUT); pinMode(in2Pin, OUTPUT); pinMode(in3Pin, OUTPUT); pinMode(in4Pin, OUTPUT); // ustawienie szybkosci silnika myStepper.setSpeed(motorSpeed); } void presentation() { // Wyslanie informacji o wersji programu sendSketchInfo("Program kurtyna", "1.0"); // Register all sensors to gw (they will be created as child devices) present(CHILD_ID, S_COVER); // Window Cover sub-type, commands: V_UP, V_DOWN, V_STOP } void receive(const MyMessage &message) { // if message = V_UP start moving until closed if (message.type==V_UP) { myStepper.step(CURTAIN_OPEN); // Store state in eeprom saveState(message.sensor, message.getBool()); request(CHILD_ID, V_UP, 0); // request new values from controller } if (message.type==V_DOWN) { myStepper.step(CURTAIN_CLOSED); // Store state in eeprom saveState(message.sensor, message.getBool()); request(CHILD_ID, V_DOWN, 0); // request new values from controller } }
I use MySensors Library 2.0
Youtube film
Silnik krokowy z L298N - sterowanie z Domoticz – 00:40
— Roman Kubat
-
your wiring diagram is not accessable:
Error (403)
It seems you don't belong here! You should probably sign in. Check out our Help Center and forums for help, or head back to home.i use a DC motor so can't help with stepper motors, but can you share an image of how you setup your rails?
-
Now the wiring diagram should be visible.
-
I write program with AccelStepper library
/ Enable debug prints to serial monitor #define MY_DEBUG // Enable and select radio type attached #define MY_RADIO_NRF24 //#define MY_RADIO_RFM69 // Enable repeater functionality for this node #define MY_REPEATER_FEATURE #include <SPI.h> #include <MySensors.h> #include <AccelStepper.h> //import biblioteki AccelStepper #define HALFSTEP 8 #define CURTAIN_CLOSED 2000 // wartosc gdy kurtyna zamknieta #define CURTAIN_OPEN 0 // wartosc gdy kurtyna otwarta #define CHILD_ID 1 // definicje MySensors MyMessage message(CHILD_ID, S_COVER); // Definicja pinow silnika #define IN1 3 // IN1 - zielony #define IN2 4 // IN2 - czarny #define IN3 5 // IN3 - niebieski #define IN4 6 // IN4 - czerwony AccelStepper stepper1(HALFSTEP, IN1, IN2, IN3, IN4); void setup() { stepper1.setMaxSpeed(200.0); stepper1.setAcceleration(1000.0); stepper1.setSpeed(200); stepper1.moveTo(CURTAIN_OPEN); } void presentation() { // Wyslanie informacji o wersji programu sendSketchInfo("Program kurtyna", "1.0"); // Register all sensors to gw (they will be created as child devices) present(CHILD_ID, S_COVER); // Window Cover sub-type, commands: V_UP, V_DOWN, V_STOP } void loop() { stepper1.run(); //Start } void receive(const MyMessage &message) { // if message = V_UP start moving until closed if (message.type==V_UP) { if (stepper1.distanceToGo() == 0){ if (stepper1.currentPosition() == CURTAIN_OPEN){ stepper1.moveTo(CURTAIN_CLOSED); // Store state in eeprom saveState(message.sensor, message.getBool()); request(CHILD_ID, V_UP, 0); // request new values from controller } } } if (message.type==V_DOWN) { stepper1.moveTo(CURTAIN_OPEN); // Store state in eeprom saveState(message.sensor, message.getBool()); request(CHILD_ID, V_DOWN, 0); // request new values from controller } if (message.type==V_STOP) { stepper1.setCurrentPosition(0); // Store state in eeprom saveState(message.sensor, message.getBool()); request(CHILD_ID, V_STOP, 0); // request new values from controller } }
I use MySensors library 2.0
Now i working.
-
Hello,
I trying to adopt the code for 28BYJ-48 stepper motor with ULN2003 driver board for roller blinds control. Code is working, but when the stepper motor is always on - it gets hot. Because of it also its consumes more energy. So thats why I want to turn on ULN2003 board only when new action is started, and after that it should be shuted down again. I could do it, because ULN2003 has On/Off jumper:
Question #1: I trying to use relay for that On/Off. Could I do it without relay and control On/Off jumper directly from arduino? Does relay is the best solution?
Question #2: I suck at programming, but I trying to add theese lines to my code to control the relay SIL05-1A72-71D, which controls the ULN2003 driver board:
int powerPin = 7; //before SETUP
pinMode(powerPin, OUTPUT); // In SETUP
digitalWrite(powerPin, LOW); // In SETUP
digitalWrite(powerPin, HIGH); // In void receive function
delay(CURTAIN_CLOSED); // In void receive function
digitalWrite(powerPin, LOW); // In void receive functiondigitalWrite(powerPin, HIGH); - I think is in right position, but I don't know how to turn off relay when action is done. Please advise for coding, because I not good in it. Thank You! Below is the code:
// Enable debug prints to serial monitor #define MY_DEBUG // Enable and select radio type attached #define MY_NODE_ID 10 #define MY_RADIO_NRF24 //#define MY_RADIO_RFM69 // Enable repeater functionality for this node //#define MY_REPEATER_FEATURE #include <SPI.h> #include <MySensors.h> #include <AccelStepper.h> //import biblioteki AccelStepper #define HALFSTEP 8 #define CURTAIN_CLOSED 10000 // wartosc gdy kurtyna zamknieta #define CURTAIN_OPEN 0 // wartosc gdy kurtyna otwarta #define CHILD_ID 1 int powerPin = 7; // definicje MySensors MyMessage message(CHILD_ID, S_COVER); // Definicja pinow silnika #define IN1 3 // IN1 - zielony #define IN2 4 // IN2 - czarny #define IN3 5 // IN3 - niebieski #define IN4 6 // IN4 - czerwony AccelStepper stepper1(HALFSTEP, IN1, IN3, IN2, IN4); void setup() { stepper1.setMaxSpeed(1000.0); stepper1.setAcceleration(100.0); stepper1.setSpeed(200); stepper1.moveTo(CURTAIN_OPEN); pinMode(powerPin, OUTPUT); digitalWrite(powerPin, LOW); } void presentation() { // Wyslanie informacji o wersji programu sendSketchInfo("Roller blinds", "1.0"); // Register all sensors to gw (they will be created as child devices) present(CHILD_ID, S_COVER); // Window Cover sub-type, commands: V_UP, V_DOWN, V_STOP } void loop() { stepper1.run(); //Start } void receive(const MyMessage &message) { // if message = V_UP start moving until closed if (message.type==V_UP) { digitalWrite(powerPin, HIGH); if (stepper1.distanceToGo() == 0){ if (stepper1.currentPosition() == CURTAIN_OPEN){ stepper1.moveTo(CURTAIN_CLOSED); // Store state in eeprom saveState(message.sensor, message.getBool()); request(CHILD_ID, V_UP, 0); // request new values from controller } } } if (message.type==V_DOWN) { digitalWrite(powerPin, HIGH); stepper1.moveTo(CURTAIN_OPEN); // Store state in eeprom saveState(message.sensor, message.getBool()); request(CHILD_ID, V_DOWN, 0); // request new values from controller } if (message.type==V_STOP) { digitalWrite(powerPin, HIGH); stepper1.setCurrentPosition(0); // Store state in eeprom saveState(message.sensor, message.getBool()); request(CHILD_ID, V_STOP, 0); // request new values from controller } // delay(CURTAIN_CLOSED); // digitalWrite(powerPin, LOW); }
-
@jacikaas said in Curtain Control Node.:
void receive(const MyMessage &message)
{
// if message = V_UP start moving until closed
if (message.type==V_UP) {
digitalWrite(powerPin, HIGH);
if (stepper1.distanceToGo() == 0){
if (stepper1.currentPosition() == CURTAIN_OPEN){
stepper1.moveTo(CURTAIN_CLOSED);
// Store state in eeprom
saveState(message.sensor, message.getBool());
request(CHILD_ID, V_UP, 0); // request new values from controller
}
}
}
if (message.type==V_DOWN) {
digitalWrite(powerPin, HIGH);
stepper1.moveTo(CURTAIN_OPEN);
// Store state in eeprom
saveState(message.sensor, message.getBool());
request(CHILD_ID, V_DOWN, 0); // request new values from controller
}
if (message.type==V_STOP) {
digitalWrite(powerPin, HIGH);
stepper1.setCurrentPosition(0);
// Store state in eeprom
saveState(message.sensor, message.getBool());
request(CHILD_ID, V_STOP, 0); // request new values from controller
}
// delay(CURTAIN_CLOSED);
// digitalWrite(powerPin, LOW);
}You can enable and disable outputs and try. In the following function:
void receive(const MyMessage &message)
//add the following at the begining of the function stepper1.enableOutputs () //and add the following at the end in the same function. stepper1.disableOutputs ()
I hope this helps haven't tried it myself.
-
@suresh-mali Thank You Suresh for Your answer!
I disconnect relay and delete it lines from the code.I add lines to code like You said:
void receive(const MyMessage &message) { stepper1.enableOutputs (); // rest of code stepper1.disableOutputs (); }
It looks like it have to work because it is simple solution, but it doesn't... On driver board ULN2003, when motor is not rotating, A Led is always on. I think if command with disableOutputs would be activated, then the LED should not be on?
-
@jacikaas Ok.
//Remove this line from loop function stepper1.run(); //Change this line stepper1.moveTo(CURTAIN_CLOSED); //to stepper1.runToPosition(CURTAIN_CLOSED);
I hope this works. Also make similar changes in block which closes curtains.
Refernce article: https://www.pjrc.com/teensy/td_libs_AccelStepper.html
-
@suresh-mali
I now reading about that funktions in http://www.airspayce.com/mikem/arduino/AccelStepper/classAccelStepper.html#a344f58fef8cc34ac5aa75ba4b665d21cDid what You said, but getting error: no matching function for call to 'AccelStepper::runToPosition(int)'
I paste here all code, but I think I did everything without mistakes:// Enable debug prints to serial monitor #define MY_DEBUG // Enable and select radio type attached #define MY_NODE_ID 10 #define MY_RADIO_NRF24 //#define MY_RADIO_RFM69 // Enable repeater functionality for this node //#define MY_REPEATER_FEATURE #include <SPI.h> #include <MySensors.h> #include <AccelStepper.h> #define HALFSTEP 8 #define CURTAIN_CLOSED 10000 #define CURTAIN_OPEN 0 #define CHILD_ID 1 // int powerPin = 7; // definicje MySensors MyMessage message(CHILD_ID, S_COVER); // Definicja pinow silnika #define IN1 3 // IN1 #define IN2 4 // IN2 #define IN3 5 // IN3 #define IN4 6 // IN4 AccelStepper stepper1(HALFSTEP, IN1, IN3, IN2, IN4); void setup() { stepper1.setMaxSpeed(1000.0); stepper1.setAcceleration(100.0); stepper1.setSpeed(200); stepper1.runToPosition(CURTAIN_CLOSED); } void presentation() { sendSketchInfo("Roller blinds", "1.0"); present(CHILD_ID, S_COVER); // Window Cover sub-type, commands: V_UP, V_DOWN, V_STOP } void loop() { //stepper1.run(); //Start } void receive(const MyMessage &message) { stepper1.enableOutputs (); // if message = V_UP start moving until closed if (message.type==V_UP) { if (stepper1.distanceToGo() == 0){ if (stepper1.currentPosition() == CURTAIN_OPEN){ stepper1.runToPosition(CURTAIN_CLOSED); // Store state in eeprom saveState(message.sensor, message.getBool()); request(CHILD_ID, V_UP, 0); // request new values from controller } } } if (message.type==V_DOWN) { stepper1.moveTo(CURTAIN_OPEN); // Store state in eeprom saveState(message.sensor, message.getBool()); request(CHILD_ID, V_DOWN, 0); // request new values from controller } if (message.type==V_STOP) { stepper1.setCurrentPosition(0); // Store state in eeprom saveState(message.sensor, message.getBool()); request(CHILD_ID, V_STOP, 0); // request new values from controller } stepper1.disableOutputs (); }
-
@jacikaas Sorry to revive an old thread, but did you get this resolved and if so how?
I just 3D printed my first motorised roller blind mounts and have tested with arduino stepper only.
Next is to go mysensors with it, but which way is best? Standard arduino stepper or accelstepper?
-
Okey. This is an old thread but I hope it's ok I keep it alive.
I just printed som parts for my blinds.
I use a stepper motor (28BYJ-48 with ULN2003 Driver board) and this sketch found in this thread.
The sketch is awesome.
To my quiestion.
I want to run 2 blinds at the same time from ONE node. Synchronized or maybe individual.
Is this possible?
There isn't any digital pins over for this when using button(s).
I think i saw someone using analog pin connected to the ULN2003 someware.
Is it possible to run 2 motors on one board?
Really hope somone can answer som of my question.
What modifications is needed in the sketch if this is possible.
This is the sketch I use./* PROJECT: MY Sensors curtain controller PROGRAMMER: AWI DATE: march 11, 2016 FILE: AWI stepper1.ino LICENSE: Public domain Hardware: ATMega328p board w/ NRF24l01 and MySensors 2.0 (Development) Special: uses AccelStepper library Summary: Curtain control with stepper motor. Manual operation with 1 push button Calibration with manual button Remarks: Fixed node-id Change log: 20160312 - Cleanup */ // Enable debug prints to serial monitor #define MY_DEBUG #define MY_NODE_ID 13 // fixed node number // Enable and select radio type attached #define MY_RADIO_RF24 //#define MY_RADIO_RFM69 #include <SPI.h> #include <MySensors.h> // stepper library #include <AccelStepper.h> // http://www.airspayce.com/mikem/arduino/AccelStepper/ #define HALFSTEP 8 // Stepper uses "Halfstep" mode // button library // used for: // - manual open close - single click: close/ stop/ open/ stop // - calibration - after long press: open calibarion- until single press - closed calibration - until single press (stop) #include <Button.h> // https://github.com/JChristensen/Button #define CHILD_ID 1 // Id of the sensor child #define SN "Curtain control 13" #define SV "1.0" #define buttonPin1 7 // Arduino pin connected to buttonPin1 #define buttonPin2 A0 // Arduino pin connected to buttonPin2 (fixed to ground) // Motor pin definitions #define motorPin1 3 // IN1 on the ULN2003 driver 1 #define motorPin2 4 // IN2 on the ULN2003 driver 1 #define motorPin3 5 // IN3 on the ULN2003 driver 1 #define motorPin4 6 // IN4 on the ULN2003 driver 1 const unsigned long heartbeatInterval = 1 * 3600UL * 1000UL ; // heartbeatinterval unsigned long heartbeatCounter = 0 ; // // helper routines to store and retrieve long in mysensors EEPROM union { // used to convert long to bytes for EEPROM storage long longInt; uint8_t LongByte[4]; } convLongInt ; void saveStateL(int EEposition, long StateL){ convLongInt.longInt = StateL ; for (int y = 0; y < 4 ; y++){ // convert to bytes saveState(EEposition + y , convLongInt.LongByte[y]) ; } Serial.print("State saved: "); Serial.println(StateL); } long loadStateL(int EEposition){ for (int y = 0; y < 4 ; y++){ // convert from bytes convLongInt.LongByte[y] = loadState(EEposition + y) ; } Serial.print("State read: "); Serial.println(convLongInt.longInt); return convLongInt.longInt ; } // Initialize with pin sequence IN1-IN3-IN2-IN4 for using the AccelStepper with 28BYJ-48 AccelStepper stepper1(HALFSTEP, motorPin1, motorPin3, motorPin2, motorPin4); // Initialize button active low, debounce and internal pull-up Button myBtn(buttonPin1, true, true, 40); // Initiate the button (pin, pull_up, invert, debounce_ms) MyMessage percentageMsg(CHILD_ID, V_PERCENTAGE); // used to send updates to controller const long maxRun = 4000000L ; // maximum runway long setPosition = 0 ; // remembers set position, need to be saved in EEPROM const int setPositionEE = 4 ; // eeprom location long openPosition = 0 ; // Position at open, need to be saved in EEPROM? const int openPositionEE = setPositionEE + 4 ; // eeprom location long closedPosition = 120000UL ; // Position at full close, need to be saved in EEPROM const int closedPositionEE = openPositionEE + 4 ; // eeprom location unsigned long idleTimer = millis() ; // return to idle timer unsigned long idleTime = 100000UL ; // return to idle after 100 secs unsigned long printTimer = millis() ; // print timer unsigned long printTime = 1000UL ; // print after 1 secs enum position_t {Open, Close, Idle, Running} ; position_t lastDirection = Open ; // lastDirection only for buttonpress position_t runStatus = Idle ; // indicates current status for running motor. used for status reporting to controller enum State_t {sIdle, sCalibrateOpen, sCalibrateClose} ; State_t State = sIdle ; void setup() { // setup buttons pinMode(buttonPin1, OUTPUT); stepper1.setMaxSpeed(2000.0); stepper1.setAcceleration(1000.0); //saveStateL(closedPositionEE, closedPosition) ; // INIT: save closed position in EEPROM closedPosition = loadStateL(closedPositionEE) ; // need to get last values from EEPROM and assume the current position is correct setPosition = loadStateL(setPositionEE) ; stepper1.setCurrentPosition(setPosition ); }//--(end setup )--- void presentation() { present(CHILD_ID, S_COVER, "Curtain"); // Window Cover sub-type, commands: V_UP, V_DOWN, V_STOP // Register the LED Dimmable Light with the gateway sendSketchInfo(SN, SV); } void loop() { unsigned int now = millis() ; // current time for loop // simple state machine for button press myBtn.read(); switch (State) { // Idle state, waiting for some action // - button press // - idleTimer case sIdle: if (myBtn.wasReleased()){ // idle Serial.println("Button release") ; // if running stop if (stepper1.isRunning()){ setPosition = stepper1.currentPosition(); stepper1.moveTo(setPosition) ; // move to current position (was already there..) } else if (lastDirection == Open) { stepper1.moveTo(closedPosition) ; lastDirection = Close ; } else { // lastDirection == Close stepper1.moveTo(openPosition) ; lastDirection = Open ; } } else if (myBtn.pressedFor(3000)){ // move to calibratete state with long press Serial.println("Button press long") ; idleTimer = now ; // return to idle after ... State = sCalibrateOpen ; stepper1.move(-maxRun) ; // let the stepper open with maximum } break ; // if not running and last action was open close ; else open // if longpress Calibrate open case sCalibrateOpen: // calibration going on if (myBtn.wasPressed()){ stepper1.setCurrentPosition(0 ); // set new 0 position ?? openPosition = setPosition = 0 ; State = sCalibrateClose ; // next is close calibarion stepper1.move(maxRun) ; // let the stepper close with maximum } else if (now > idleTimer + idleTime) { // timer expired -> abort calibration State = sIdle ; } break ; case sCalibrateClose: // calibrate closed position, end with keypress if (myBtn.wasPressed()) { closedPosition = setPosition = stepper1.currentPosition() ; saveStateL(closedPositionEE, closedPosition) ; // save closed position in EEPROM State = sIdle ; stepper1.moveTo(openPosition) ; // return to open after calibration } else if (now > idleTimer + idleTime) { // timer expired -> abort calibration State = sIdle ; } break ; default : break ; } // power off stepper if not running (no need to reenable)) if (!stepper1.isRunning()){ if (runStatus != Idle){ // there was a change in runningstatus, so report to controller setPosition = stepper1.currentPosition() ; // store in EEPROM and report ready to controller saveStateL(setPositionEE, setPosition) ; send( percentageMsg.set((100 * setPosition)/(closedPosition - openPosition))) ; runStatus = Idle ; } stepper1.disableOutputs(); } else { runStatus = Running ; } stepper1.run(); /* if (printTimer++ > now + printTime){ printTimer = now ; Serial.println(stepper1.currentPosition()); } */ } // This is called when a message is received void receive(const MyMessage &message) { // We only expect few types of messages from controller, check which switch (message.type) { case V_PERCENTAGE: // Curtain should be opened stepper1.moveTo(message.getInt() * (closedPosition - openPosition)/100); Serial.print("Message: "); Serial.print(message.sensor); Serial.print(" , value: % "); Serial.println( message.getInt()); Serial.print("Moving to: "); Serial.println(message.getInt() * (closedPosition - openPosition)/100); break ; case V_STATUS: // Curtain should be opened or closed full stepper1.moveTo((message.getInt() == HIGH)?openPosition:closedPosition); Serial.print("Message - valid: "); Serial.print(message.sensor); Serial.print(" , value: % "); break ; default : // not recognizable message Serial.print("Message - valid: "); Serial.print(message.sensor); Serial.print(", Unrecognized "); break ; } }
-
You can use an I/O expander like an MCP23017 which is a 16 port expander or a 8 port expander. They're cheap. Keep in mind they consume 1 mAmp. I'm currently working on a product to overcome that power issue for a battery powered Node. Will try to post about it in a while
-
@xydix
Hello @xydix do you have the curtain working with the sketch above?
I don't get the button working on pin 7.
Witch button library do you use?
I have the library witch is in the sketch, i only must change Bbutton.h in LC_Button.h.
Is that maby my problem.
I hope u can help me.
Greatz Ton
-
@barduino said in Curtain Control Node.:
DC Motor, very interesting!
Do you have end stops? The sketch doesn't open for some reason...
Cheers
I have been working on a similar project last week. The mechanics and ´looks´ are the most difficult part for me.
-
@ton-rijnaard
Hi. This was a problem for me to. The reason is the library changed name and updates was made.
I use the library version 0.9.
I have like zero skills in arduino. If you use the old library the button will work with the sketch i posted a moth ago.
Looke here: https://github.com/JChristensen/JC_Button/releases
Maybe some of the other older releases will work. Don't remember why o choose V0.9
Look at 2.0.0, "This is a major release that is not backwards-compatible with previous releases."
-
@xydix
To answer my own question the stepper runs fine from the analog pins.
Next step is to find some intersted in the same thing and see if we could get the sketch adjusted to run two steppers.
I will try when i get some time but I guess I will have problem to clone the calibration and store-to-eerprom-part.
-
@AWI Sorry for butting in to an old thread, but automating my blinds is top priority!
I've copied your sketch and have everything setup - however, I have eratic behaviour and a few questions:
void setup() { // setup buttons pinMode(buttonPin1, OUTPUT); stepper1.setMaxSpeed(2000.0); stepper1.setAcceleration(1000.0); //saveStateL(closedPositionEE, closedPosition) ; // INIT: save closed position in EEPROM closedPosition = loadStateL(closedPositionEE) ; // need to get last values from EEPROM and assume the current position is correct setPosition = loadStateL(setPositionEE) ; stepper1.setCurrentPosition(setPosition ); }//--(end setup )---
-
Should the pinMode on buttonPin1 actually be an INPUT?
-
Can you please describe how to go about calibrating the system?
-
Does the //saveStateL(closedPositionEE, closedPosition);
need uncommenting for one boot of the arduino and the blind closed - then the sketch re-uploading with it commented again?
Your help is greatly appreciated as I have a twitching motor and feel I'm almost there!
-
-
@adds666 What i can tell it was a long time since @AWI was active here.
I use this sketch with my first bild since a couple of days.
For me it works like a charm. But in my case I control the motor from analog pins just to test and it seems OK.
No problem with the button.
The only thing I don't like, is the calibration.
I my case, using this on a blind, when calibrating, first it goes down, then i press the button, the blind stops at once and change it's direction and goes up. But when i reach the top level and want to stop it and save the state, it stops slow and pass the point i pressed the button and then reverse to the right spot. Problem is, the blind can't go further because it is in the top position.
If it would be the other way around it wouldn't be a problem.
Maybe i have the motor one the "wrong" side of the blind but when it is down the "downkey" in my HA is greyed out.
When it i up the "upkey" it greyed out so that seems right.What is not working in your case?
What motor do you use? I use 28BYJ-48 with ULN2003 and had to setstepper1.setMaxSpeed(2000.0);
tostepper1.setMaxSpeed(1000.0);
because the motor "slipped" at a speed of 2000.
Like i wrote earlier in the thread I tried to figure out how to "dublicate" the sketch and ute it for two motors.
I really don't know where to start to get this done. It would be awesome to use one arduino with two blinds.
-
Hi @xydix , glad to hear you have the sketch working properly. Do you have the pinmode as OUTPUT or did you change to input?
And do you still have the saveStateL(closedPositionEE, closedPosition); line commented out?
If I change the pinMode to input, I get the system powered up and connected to the mysensors gateway fine - serial print repeats 'Button Release'. If I short press the button, the system does nothing. If I hold the button for 3000ms serial print reads 'Button Long Press' and the motor turns - however If I let go of the button, the motor stop and system returns to its 'button release' state.
I'm not able to get the system in to calibration mode.
-
@adds666
I am still in "testing mode".
I had some problems with the node, it stoped receiving messages but did always work with the button.
Yesterday i built a new node and uploaded the sketch and it seems to work better.
Here is the sketch i use, I changed the pin for the button due to missing digital pins on my pcb.
And I am back on using digital pins for the stepper driver for the same reason.
I greyed out buttonpin 2. I don't think it is needed.
I also greyed out the heartbeat-part as I had problems. Just for testing. Don't know if it is suitable when using Home Assistant./* PROJECT: MY Sensors curtain controller PROGRAMMER: AWI DATE: march 11, 2016 FILE: AWI stepper1.ino LICENSE: Public domain Hardware: ATMega328p board w/ NRF24l01 and MySensors 2.0 (Development) Special: uses AccelStepper library Summary: Curtain control with stepper motor. Manual operation with 1 push button Calibration with manual button Remarks: Fixed node-id Change log: 20160312 - Cleanup */ // Enable debug prints to serial monitor #define MY_DEBUG #define MY_NODE_ID 8 // fixed node number // Enable and select radio type attached #define MY_RADIO_RF24 #define MY_RF24_PA_LEVEL RF24_PA_HIGH //#define MY_RADIO_RFM69 #include <SPI.h> #include <MySensors.h> // stepper library #include <AccelStepper.h> // http://www.airspayce.com/mikem/arduino/AccelStepper/ #define HALFSTEP 8 // Stepper uses "Halfstep" mode // button library // used for: // - manual open close - single click: close/ stop/ open/ stop // - calibration - after long press: open calibarion- until single press - closed calibration - until single press (stop) #include <Button.h> // https://github.com/JChristensen/Button - Använd gammal version. testat med V0.9 och det funkar #define CHILD_ID 1 // Id of the sensor child #define SN "Curtain control" #define SV "1.0" #define buttonPin1 A0 // Arduino pin connected to buttonPin1 //#define buttonPin2 A0 // Arduino pin connected to buttonPin2 (fixed to ground) // Motor pin definitions #define motorPin1 2 // IN1 on the ULN2003 driver 1 #define motorPin2 3 // IN2 on the ULN2003 driver 1 #define motorPin3 5 // IN3 on the ULN2003 driver 1 #define motorPin4 6 // IN4 on the ULN2003 driver 1 //const unsigned long heartbeatInterval = 1 * 3600UL * 1000UL ; // heartbeatinterval //unsigned long heartbeatCounter = 0 ; // // helper routines to store and retrieve long in mysensors EEPROM union { // used to convert long to bytes for EEPROM storage long longInt; uint8_t LongByte[4]; } convLongInt ; void saveStateL(int EEposition, long StateL){ convLongInt.longInt = StateL ; for (int y = 0; y < 4 ; y++){ // convert to bytes saveState(EEposition + y , convLongInt.LongByte[y]) ; } Serial.print("State saved: "); Serial.println(StateL); } long loadStateL(int EEposition){ for (int y = 0; y < 4 ; y++){ // convert from bytes convLongInt.LongByte[y] = loadState(EEposition + y) ; } Serial.print("State read: "); Serial.println(convLongInt.longInt); return convLongInt.longInt ; } // Initialize with pin sequence IN1-IN3-IN2-IN4 for using the AccelStepper with 28BYJ-48 AccelStepper stepper1(HALFSTEP, motorPin1, motorPin3, motorPin2, motorPin4); // Initialize button active low, debounce and internal pull-up Button myBtn(buttonPin1, true, true, 40); // Initiate the button (pin, pull_up, invert, debounce_ms) MyMessage percentageMsg(CHILD_ID, V_PERCENTAGE); // used to send updates to controller const long maxRun = 4000000L ; // maximum runway long setPosition = 0 ; // remembers set position, need to be saved in EEPROM const int setPositionEE = 4 ; // eeprom location long openPosition = 0 ; // Position at open, need to be saved in EEPROM? const int openPositionEE = setPositionEE + 4 ; // eeprom location long closedPosition = 120000UL ; // Position at full close, need to be saved in EEPROM const int closedPositionEE = openPositionEE + 4 ; // eeprom location unsigned long idleTimer = millis() ; // return to idle timer unsigned long idleTime = 100000UL ; // return to idle after 100 secs unsigned long printTimer = millis() ; // print timer unsigned long printTime = 1000UL ; // print after 1 secs enum position_t {Open, Close, Idle, Running} ; position_t lastDirection = Open ; // lastDirection only for buttonpress position_t runStatus = Idle ; // indicates current status for running motor. used for status reporting to controller enum State_t {sIdle, sCalibrateOpen, sCalibrateClose} ; State_t State = sIdle ; void setup() { // setup buttons pinMode(buttonPin1, OUTPUT); stepper1.setMaxSpeed(1000.0); stepper1.setAcceleration(1000.0); //saveStateL(closedPositionEE, closedPosition) ; // INIT: save closed position in EEPROM closedPosition = loadStateL(closedPositionEE) ; // need to get last values from EEPROM and assume the current position is correct setPosition = loadStateL(setPositionEE) ; stepper1.setCurrentPosition(setPosition ); }//--(end setup )--- void presentation() { present(CHILD_ID, S_COVER, "Curtain"); // Window Cover sub-type, commands: V_UP, V_DOWN, V_STOP // Register the LED Dimmable Light with the gateway sendSketchInfo(SN, SV); } void loop() { unsigned int now = millis() ; // current time for loop // simple state machine for button press myBtn.read(); switch (State) { // Idle state, waiting for some action // - button press // - idleTimer case sIdle: if (myBtn.wasReleased()){ // idle Serial.println("Button release") ; // if running stop if (stepper1.isRunning()){ setPosition = stepper1.currentPosition(); stepper1.moveTo(setPosition) ; // move to current position (was already there..) } else if (lastDirection == Open) { stepper1.moveTo(closedPosition) ; lastDirection = Close ; } else { // lastDirection == Close stepper1.moveTo(openPosition) ; lastDirection = Open ; } } else if (myBtn.pressedFor(3000)){ // move to calibratete state with long press Serial.println("Button press long") ; idleTimer = now ; // return to idle after ... State = sCalibrateOpen ; stepper1.move(-maxRun) ; // let the stepper open with maximum } break ; // if not running and last action was open close ; else open // if longpress Calibrate open case sCalibrateOpen: // calibration going on if (myBtn.wasPressed()){ stepper1.setCurrentPosition(0 ); // set new 0 position ?? openPosition = setPosition = 0 ; State = sCalibrateClose ; // next is close calibarion stepper1.move(maxRun) ; // let the stepper close with maximum } else if (now > idleTimer + idleTime) { // timer expired -> abort calibration State = sIdle ; } break ; case sCalibrateClose: // calibrate closed position, end with keypress if (myBtn.wasPressed()) { closedPosition = setPosition = stepper1.currentPosition() ; saveStateL(closedPositionEE, closedPosition) ; // save closed position in EEPROM State = sIdle ; stepper1.moveTo(openPosition) ; // return to open after calibration } else if (now > idleTimer + idleTime) { // timer expired -> abort calibration State = sIdle ; } break ; default : break ; } // power off stepper if not running (no need to reenable)) if (!stepper1.isRunning()){ if (runStatus != Idle){ // there was a change in runningstatus, so report to controller setPosition = stepper1.currentPosition() ; // store in EEPROM and report ready to controller saveStateL(setPositionEE, setPosition) ; send( percentageMsg.set((100 * setPosition)/(closedPosition - openPosition))) ; runStatus = Idle ; } stepper1.disableOutputs(); } else { runStatus = Running ; } stepper1.run(); /* if (printTimer++ > now + printTime){ printTimer = now ; Serial.println(stepper1.currentPosition()); } */ } // This is called when a message is received void receive(const MyMessage &message) { // We only expect few types of messages from controller, check which switch (message.type) { case V_PERCENTAGE: // Curtain should be opened stepper1.moveTo(message.getInt() * (closedPosition - openPosition)/100); Serial.print("Message: "); Serial.print(message.sensor); Serial.print(" , value: % "); Serial.println( message.getInt()); Serial.print("Moving to: "); Serial.println(message.getInt() * (closedPosition - openPosition)/100); break ; case V_STATUS: // Curtain should be opened or closed full stepper1.moveTo((message.getInt() == HIGH)?openPosition:closedPosition); Serial.print("Message - valid: "); Serial.print(message.sensor); Serial.print(" , value: % "); break ; default : // not recognizable message Serial.print("Message - valid: "); Serial.print(message.sensor); Serial.print(", Unrecognized "); break ; } }
Suggested Topics
-
One for All
Domoticz • • fets