FlipFlat.cpp 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328
  1. // FlipFlat based on the Altinak Flip-Flat commands. Makes it possible to use it with ASCOM and INDI drivers
  2. // Enable DEBUG in FlipFlat.h to see logging.
  3. // Due to driver imitations in de indi driver, when using ESP32 Serial can't be used for FlipFlat communication.
  4. // Use Serial 2 (pins 16 and 17) instead. Standard Serial is used for debug messages
  5. #include <FlipFlat.h>
  6. int deviceId = FLIP_FLAT; // set this to FLAT_MAN if you want to remove or not use the motor handling
  7. int motorStatus = STOPPED;
  8. int lightStatus = OFF;
  9. int coverStatus = CLOSED;
  10. int motorDirection = NONE;
  11. float currentAngle = 0.0;
  12. int brightness = 0;
  13. EasyButton brightness_plus(BRIGHT_PLUS);
  14. EasyButton brightness_min(BRIGHT_MIN);
  15. EasyButton cover_control(COVER_FF);
  16. AccelStepper stepper(AccelStepper::DRIVER, STEP_PIN, DIR_PIN);
  17. void setup()
  18. {
  19. setupSerial();
  20. LOG("Debug mode");
  21. LOG("Starting FlipFlat");
  22. LOG("Setup PWM pin");
  23. pinMode(PWM, OUTPUT);
  24. LOG("Setup brightness buttons");
  25. brightness_plus.begin();
  26. brightness_plus.onPressed(increaseBrightness);
  27. brightness_min.begin();
  28. brightness_min.onPressed(decreaseBrightness);
  29. LOG("Setup open-close button");
  30. cover_control.begin();
  31. cover_control.onPressed(handleCoverSwitch);
  32. LOG("Initializing stepper");
  33. stepper.setMaxSpeed(1000);
  34. stepper.setAcceleration(50);
  35. stepper.setSpeed(200);
  36. stepper.moveTo(STEPS);
  37. }
  38. void loop()
  39. {
  40. brightness_plus.read();
  41. brightness_min.read();
  42. cover_control.read();
  43. if (FF_SERIAL.available() > 0) // all incoming communications are fixed length at 6 bytes including the \r
  44. {
  45. LOG("Command received");
  46. parseCommand();
  47. }
  48. }
  49. void increaseBrightness()
  50. {
  51. LOG("Brightness+ is pressed");
  52. brightness++;
  53. setBrightness(brightness);
  54. }
  55. void decreaseBrightness()
  56. {
  57. LOG("Brightness- is pressed");
  58. brightness--;
  59. setBrightness(brightness);
  60. }
  61. void setBrightness(int newBrightness)
  62. {
  63. LOG1("Setting brightness to ", newBrightness);
  64. analogWrite(PWM, newBrightness);
  65. }
  66. void parseCommand()
  67. {
  68. LOG("Parse command");
  69. char *cmd;
  70. char *data;
  71. char str[20];
  72. memset(str, 0, 20);
  73. FF_SERIAL.readBytesUntil('\n', str, 20);
  74. cmd = str + 1;
  75. data = str + 2;
  76. // useful for debugging to make sure your commands came through and are parsed correctly.
  77. LOG1("cmd = ", cmd);
  78. LOG1("data = ", data);
  79. processCommand(cmd, data);
  80. while (FF_SERIAL.available() > 0)
  81. FF_SERIAL.read();
  82. }
  83. void processCommand(const char *cmd, const char *data)
  84. {
  85. switch (*cmd)
  86. {
  87. /*
  88. Ping device
  89. Request: >POOO\r
  90. Return : *PiiOOO\n
  91. ii = DEVICE_ID
  92. */
  93. case 'P':
  94. LOG("Ping received");
  95. FF_SERIAL << F("*P") << deviceId << F("OOO\n");
  96. break;
  97. /*
  98. Open shutter
  99. Request: >OOOO\r
  100. Return : *OiiOOO\n
  101. ii = deviceId
  102. This command is only supported on the Flip-Flat!
  103. */
  104. case 'O':
  105. LOG("Open received");
  106. FF_SERIAL << F("*O") << deviceId << F("OOO\n");
  107. openFlipFlat();
  108. break;
  109. /*
  110. Close shutter
  111. Request: >COOO\r
  112. Return : *CiiOOO\n
  113. ii = deviceId
  114. This command is only supported on the Flip-Flat!
  115. */
  116. case 'C':
  117. LOG("Close received");
  118. FF_SERIAL << F("*C") << deviceId << F("OOO\n");
  119. closeFlipFlat();
  120. break;
  121. /*
  122. Turn light on
  123. Request: >LOOO\r
  124. Return : *LiiOOO\n
  125. ii = deviceId
  126. */
  127. case 'L':
  128. LOG("Turn on received");
  129. FF_SERIAL << F("*L") << deviceId << F("OOO\n");
  130. lightStatus = ON;
  131. setBrightness(brightness);
  132. break;
  133. /*
  134. Turn light off
  135. Request: >DOOO\r
  136. Return : *DiiOOO\n
  137. id = deviceId
  138. */
  139. case 'D':
  140. LOG("Turn off received");
  141. FF_SERIAL << F("*D") << deviceId << F("OOO\n");
  142. lightStatus = OFF;
  143. setBrightness(0);
  144. break;
  145. /*
  146. Set brightness
  147. Request: >Bxxx\r
  148. xxx = brightness value from 000-255
  149. Return : *Bidyyy\n
  150. id = deviceId
  151. yyy = value that brightness was set from 000-255
  152. */
  153. case 'B':
  154. LOG("Set brightness received");
  155. FF_SERIAL << F("*B") << deviceId << _WIDTHZ(brightness, 3) << F("\n");
  156. brightness = atoi(data);
  157. if (lightStatus == ON)
  158. setBrightness(brightness);
  159. break;
  160. /*
  161. Get brightness
  162. Request: >J000\r
  163. Return : *Jiiyyy\n
  164. id = deviceId
  165. yyy = current brightness value from 000-255
  166. */
  167. case 'J':
  168. LOG("Get brightness received");
  169. FF_SERIAL << F("*J") << deviceId << _WIDTHZ(brightness, 3) << F("\n");
  170. break;
  171. /*
  172. Get device status:
  173. Request: >S000\r
  174. Return : *SiiMLC\n
  175. ii = deviceId
  176. M = motor status( 0 stopped, 1 running)
  177. L = light status( 0 off, 1 on)
  178. C = Cover Status( 0 moving, 1 closed, 2 open)
  179. */
  180. case 'S':
  181. LOG("Get status recived");
  182. FF_SERIAL << F("*S") << deviceId << motorStatus << lightStatus << coverStatus << F("\n");
  183. break;
  184. /*
  185. Get firmware version
  186. Request: >V000\r
  187. Return : *Vii001\n
  188. ii = deviceId
  189. */
  190. case 'V': // get firmware version
  191. LOG("Get firmware version received");
  192. FF_SERIAL << F("*V") << deviceId << F("001") << F("\n");
  193. break;
  194. }
  195. }
  196. void openFlipFlat()
  197. {
  198. if (coverStatus != OPEN)
  199. {
  200. LOG("Turn light off");
  201. setBrightness(0);
  202. motorDirection = OPENING;
  203. rotateMotor(270.0);
  204. }
  205. }
  206. void closeFlipFlat()
  207. {
  208. LOG("Closing FlipFlat");
  209. if (coverStatus != CLOSED)
  210. {
  211. motorDirection = CLOSING;
  212. rotateMotor(270.0);
  213. }
  214. else
  215. {
  216. LOG("FlipFlap already closed");
  217. }
  218. }
  219. void rotateMotor(float newAngle)
  220. {
  221. if (motorDirection == OPENING)
  222. {
  223. motorStatus = RUNNING;
  224. coverStatus = NEITHER_OPEN_NOR_CLOSED;
  225. stepper.move(STEPS * newAngle / 360.0);
  226. stepper.runToPosition();
  227. coverStatus = OPEN;
  228. }
  229. else
  230. {
  231. if (motorDirection == CLOSING)
  232. {
  233. motorStatus = RUNNING;
  234. coverStatus = NEITHER_OPEN_NOR_CLOSED;
  235. stepper.move(-STEPS * newAngle / 360.0);
  236. stepper.runToPosition();
  237. coverStatus = CLOSED;
  238. }
  239. }
  240. motorStatus = STOPPED;
  241. motorDirection = NONE;
  242. }
  243. void setupSerial()
  244. {
  245. #ifdef ESP32
  246. FF_SERIAL.begin(9600, SERIAL_8N1, 16, 17);
  247. LOG_SERIAL.begin(9600);
  248. #endif
  249. #ifdef ARDUINO_AVR_LEONARDO
  250. FF_SERIAL.begin(9600);
  251. LOG_SERIAL.begin(9600);
  252. #endif
  253. LOG("Logging serial up and running");
  254. LOG("FlipFlat Serial up and running");
  255. }
  256. void handleCoverSwitch(){
  257. switch (coverStatus){
  258. case CLOSED:
  259. LOG("Opening cover");
  260. openFlipFlat();
  261. break;
  262. case OPEN:
  263. LOG("Closing cover");
  264. closeFlipFlat();
  265. break;
  266. default:
  267. LOG("Cover not open or closed");
  268. }
  269. }