No Description
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

states.cpp 22KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681
  1. #include <Arduino.h>
  2. #include "config.h"
  3. #include "config_pins.h"
  4. #include "data.h"
  5. #include "common.h"
  6. #include "lcd.h"
  7. #include "steppers.h"
  8. #include "statemachine.h"
  9. #include "states.h"
  10. //#define DISABLE_HOMING_STATE
  11. #define ENABLE_MOVEMENT_TEST_STATE
  12. #define ENABLE_INPUT_TEST_STATE
  13. // --------------------------------------
  14. int preset_selection = 0, dispense_off = 0, dispense_count = 0, dispense_index = 0;
  15. float move_pos_x = 0.0, move_pos_y = 0.0, move_pos_z = 0.0, move_pos_e = 0.0;
  16. State sm_ask_homing = State();
  17. State sm_do_homing = State(&sm_ask_homing);
  18. StateMenu sm_menu = StateMenu(&sm_do_homing, false);
  19. StateMenu sm_auto = StateMenu(&sm_menu);
  20. StateDynamicMenu sm_presets = StateDynamicMenu(&sm_auto);
  21. StateValue<int> sm_disp_off = StateValue<int>(&sm_presets, dispense_off, 1, 1);
  22. StateValue<int> sm_disp_cnt = StateValue<int>(&sm_disp_off, dispense_count, 1, 1);
  23. State sm_disp_go = State(&sm_disp_cnt);
  24. State sm_disp_run_z_travel = State(&sm_disp_go);
  25. State sm_disp_run_xy_travel = State(&sm_disp_run_z_travel);
  26. State sm_disp_run_z_disp = State(&sm_disp_run_xy_travel);
  27. State sm_disp_run = State(&sm_disp_run_z_disp);
  28. State sm_disp_done = State(&sm_disp_run);
  29. State sm_new_preset = State(&sm_auto);
  30. State sm_wiz_move = State(&sm_new_preset);
  31. StateValues<uint8_t, 2> sm_wiz_count = StateValues<uint8_t, 2>(&sm_wiz_move);
  32. StateValues<float, 2> sm_wiz_first_pos = StateValues<float, 2>(&sm_wiz_count);
  33. StateValues<float, 2> sm_wiz_last_pos = StateValues<float, 2>(&sm_wiz_first_pos);
  34. // TODO move z above container
  35. // TODO move z inside bottom container
  36. // TODO move z inside top container
  37. // TODO move to extruder depth
  38. State sm_new_preset_done = State(&sm_wiz_last_pos);
  39. StateDynamicMenu sm_mod_preset = StateDynamicMenu(&sm_auto);
  40. // TODO modify
  41. StateDynamicMenu sm_del_preset = StateDynamicMenu(&sm_auto);
  42. // TODO delete
  43. StateMenu sm_move = StateMenu(&sm_menu);
  44. StateValue<float> sm_move_x = StateValue<float>(&sm_move, move_pos_x, X_AXIS_MIN, X_AXIS_MAX);
  45. StateValue<float> sm_move_y = StateValue<float>(&sm_move, move_pos_y, Y_AXIS_MIN, Y_AXIS_MAX);
  46. StateValue<float> sm_move_z = StateValue<float>(&sm_move, move_pos_z, Z_AXIS_MIN, Z_AXIS_MAX);
  47. StateValue<float> sm_move_e = StateValue<float>(&sm_move, move_pos_e, E_AXIS_MIN, E_AXIS_MAX);
  48. State sm_disable_steppers = State(&sm_menu);
  49. StateMenu sm_config = StateMenu(&sm_menu);
  50. State sm_conf_load = State(&sm_config);
  51. State sm_conf_save = State(&sm_config);
  52. State sm_conf_clear = State(&sm_config);
  53. StateValue<float> sm_conf_speed_xy = StateValue<float>(&sm_config, data_options()->speed_x, 1.0, XY_MAX_SPEED);
  54. StateValue<float> sm_conf_speed_z = StateValue<float>(&sm_config, data_options()->speed_z, 1.0, Z_MAX_SPEED);
  55. StateValue<float> sm_conf_speed_e = StateValue<float>(&sm_config, data_options()->speed_e, 1.0, E_MAX_SPEED);
  56. StateValue<float> sm_conf_accel_xy = StateValue<float>(&sm_config, data_options()->accel_x, 1.0, XY_MAX_ACCEL);
  57. StateValue<float> sm_conf_accel_z = StateValue<float>(&sm_config, data_options()->accel_z, 1.0, Z_MAX_ACCEL);
  58. StateValue<float> sm_conf_accel_e = StateValue<float>(&sm_config, data_options()->accel_e, 1.0, E_MAX_ACCEL);
  59. State sm_error = State(NULL);
  60. #if defined(ENABLE_MOVEMENT_TEST_STATE) || defined(ENABLE_INPUT_TEST_STATE)
  61. StateMenu sm_debug = StateMenu(&sm_menu);
  62. #endif
  63. #ifdef ENABLE_MOVEMENT_TEST_STATE
  64. StateMenu sm_movement_test = StateMenu(&sm_debug);
  65. State sm_move_xy_test = State(&sm_movement_test);
  66. State sm_move_x_test = State(&sm_movement_test);
  67. State sm_move_y_test = State(&sm_movement_test);
  68. State sm_move_z_test = State(&sm_movement_test);
  69. State sm_move_e_test = State(&sm_movement_test);
  70. #endif // ENABLE_MOVEMENT_TEST_STATE
  71. #ifdef ENABLE_INPUT_TEST_STATE
  72. State sm_input_test = State(&sm_debug);
  73. #endif // ENABLE_INPUT_TEST_STATE
  74. // --------------------------------------
  75. State *current_state = NULL;
  76. String strbuf;
  77. struct data_config_preset wizard_preset = { .count_x = 1, .count_y = 1 };
  78. float wizard_first_x = 0.0, wizard_first_y = 0.0;
  79. float wizard_last_x = 0.0, wizard_last_y = 0.0;
  80. unsigned long last_state_change = 0, last_input = 0;
  81. void states_init(void) {
  82. // ----------------------------------
  83. sm_ask_homing.setTitle("Ask for homing");
  84. sm_ask_homing.setHeading("Homing Required!");
  85. sm_ask_homing.setText("Click to home all four axes.");
  86. // ----------------------------------
  87. sm_do_homing.setTitle("Home all axes");
  88. sm_do_homing.setHeading("Homing...");
  89. sm_do_homing.onEnter([]() {
  90. #ifndef DISABLE_HOMING_STATE
  91. steppers_start_homing();
  92. #endif // DISABLE_HOMING_STATE
  93. });
  94. sm_do_homing.whenIn([](StateMachineInput smi) {
  95. #ifndef DISABLE_HOMING_STATE
  96. if (smi.all_motors_done) {
  97. if (steppers_homed()) {
  98. #endif // DISABLE_HOMING_STATE
  99. async_beep(HOMING_BEEP_TIME, HOMING_BEEP_FREQ);
  100. states_go_to(&sm_menu);
  101. #ifndef DISABLE_HOMING_STATE
  102. }
  103. }
  104. #endif // DISABLE_HOMING_STATE
  105. static char last_axis = ' ';
  106. char axis = steppers_homing_axis();
  107. if ((axis != ' ') && (axis != last_axis)) {
  108. last_axis = axis;
  109. strbuf = String(F("Currently homing ")) + axis + String(F(" axis"));
  110. sm_do_homing.setText(strbuf.c_str());
  111. sm_do_homing.updateText();
  112. }
  113. });
  114. // ----------------------------------
  115. sm_menu.setTitle("Main Menu");
  116. sm_menu.addChild(&sm_do_homing, 1);
  117. // ----------------------------------
  118. sm_move.setTitle("Move Axis");
  119. sm_move_x.setTitle("X Axis");
  120. sm_move_x.onEnter([]() {
  121. move_pos_x = steppers_pos_x();
  122. });
  123. sm_move_x.onLiveUpdate([](float v) {
  124. steppers_move_x(v);
  125. });
  126. sm_move_y.setTitle("Y Axis");
  127. sm_move_y.onEnter([]() {
  128. move_pos_y = steppers_pos_y();
  129. });
  130. sm_move_y.onLiveUpdate([](float v) {
  131. steppers_move_y(v);
  132. });
  133. sm_move_z.setTitle("Z Axis");
  134. sm_move_z.onEnter([]() {
  135. move_pos_z = steppers_pos_z();
  136. });
  137. sm_move_z.onLiveUpdate([](float v) {
  138. steppers_move_z(v);
  139. });
  140. sm_move_e.setTitle("E Axis");
  141. sm_move_e.onEnter([]() {
  142. move_pos_e = steppers_pos_e();
  143. });
  144. sm_move_e.onLiveUpdate([](float v) {
  145. steppers_move_e(v);
  146. });
  147. sm_disable_steppers.setTitle("Disable Steppers");
  148. sm_disable_steppers.onEnter([]() {
  149. steppers_kill();
  150. states_go_to(&sm_ask_homing);
  151. });
  152. // ----------------------------------
  153. sm_auto.setTitle("Filling Menu");
  154. auto preset_name_func = [](int i) {
  155. strbuf = String(i + 1);
  156. strbuf += String(F(" ("));
  157. strbuf += String(data_preset(i)->count_x);
  158. strbuf += String(F("*"));
  159. strbuf += String(data_preset(i)->count_y);
  160. strbuf += String(F(" / "));
  161. strbuf += String(data_preset(i)->distance_x);
  162. strbuf += String(F("*"));
  163. strbuf += String(data_preset(i)->distance_y);
  164. strbuf += String(F(")"));
  165. return strbuf.c_str();
  166. };
  167. sm_presets.setTitle("Use Preset");
  168. sm_presets.setPrefix("Preset ");
  169. sm_presets.dataCount([]() {
  170. return (int)data_preset_count();
  171. });
  172. sm_presets.dataGet(preset_name_func);
  173. sm_presets.dataCall([](int i) {
  174. preset_selection = i;
  175. states_go_to(&sm_disp_off);
  176. });
  177. sm_disp_off.setTitle("Offset");
  178. sm_disp_off.setText("Where to start dispensing?");
  179. sm_disp_off.onEnter([]() {
  180. dispense_off = 1;
  181. sm_disp_off.setMax(data_preset(preset_selection)->count_x * data_preset(preset_selection)->count_y);
  182. });
  183. sm_disp_cnt.setTitle("Count");
  184. sm_disp_cnt.setText("How many to fill?");
  185. sm_disp_cnt.onEnter([]() {
  186. dispense_count = data_preset(preset_selection)->count_x * data_preset(preset_selection)->count_y - (dispense_off - 1);
  187. sm_disp_cnt.setMax(dispense_count);
  188. });
  189. sm_disp_go.setTitle("Confirm");
  190. sm_disp_go.onEnter([]() {
  191. dispense_off -= 1;
  192. strbuf = String(F("Filling from ")) + String(dispense_off + 1);
  193. strbuf += String(F(" to ")) + String(dispense_off + dispense_count);
  194. strbuf += String(F("!\nIs this ok?"));
  195. sm_disp_go.setText(strbuf.c_str());
  196. dispense_index = dispense_off;
  197. // TODO allow yes/no choice?
  198. });
  199. sm_disp_run_z_travel.setTitle("Z Travel");
  200. sm_disp_run_z_travel.onEnter([]() {
  201. // TODO move z to travel height
  202. });
  203. sm_disp_run_z_travel.whenIn([](StateMachineInput smi) {
  204. // TODO update status screen
  205. if (smi.all_motors_done) {
  206. states_go_to(&sm_disp_run_xy_travel);
  207. }
  208. });
  209. sm_disp_run_xy_travel.setTitle("XY Movement");
  210. sm_disp_run_xy_travel.onEnter([]() {
  211. int y = dispense_index / data_preset(preset_selection)->count_x;
  212. int x = dispense_index % data_preset(preset_selection)->count_x;
  213. Serial.print(F("xy_travel: index="));
  214. Serial.print(dispense_index);
  215. Serial.print(F(" x="));
  216. Serial.print(x);
  217. Serial.print(F(" y="));
  218. Serial.println(y);
  219. float x_pos = data_preset(preset_selection)->offset_x + x * data_preset(preset_selection)->distance_x;
  220. float y_pos = data_preset(preset_selection)->offset_y + y * data_preset(preset_selection)->distance_y;
  221. steppers_move_x(x_pos);
  222. steppers_move_y(y_pos);
  223. });
  224. sm_disp_run_xy_travel.whenIn([](StateMachineInput smi) {
  225. // TODO update status screen
  226. if (smi.all_motors_done) {
  227. states_go_to(&sm_disp_run_z_disp);
  228. }
  229. });
  230. sm_disp_run_z_disp.setTitle("Z Dispense");
  231. sm_disp_run_z_disp.onEnter([]() {
  232. // TODO move z to dispension height
  233. });
  234. sm_disp_run_z_disp.whenIn([](StateMachineInput smi) {
  235. // TODO update status screen
  236. if (smi.all_motors_done) {
  237. states_go_to(&sm_disp_run);
  238. }
  239. });
  240. sm_disp_run.setTitle("Dispensing");
  241. sm_disp_run.onEnter([]() {
  242. // TODO start dispensing with e and moving z up
  243. });
  244. sm_disp_run.whenIn([](StateMachineInput smi) {
  245. // TODO update status screen
  246. if (smi.all_motors_done) {
  247. dispense_index++;
  248. if (dispense_index >= (dispense_count + dispense_off)) {
  249. states_go_to(&sm_disp_done);
  250. } else {
  251. states_go_to(&sm_disp_run_z_travel);
  252. }
  253. }
  254. });
  255. sm_disp_done.setTitle("Finished");
  256. sm_disp_done.onEnter([]() {
  257. // TODO show end screen, "stats"
  258. });
  259. sm_disp_done.whenIn([](StateMachineInput smi) {
  260. if (smi.click) {
  261. states_go_to(&sm_auto);
  262. }
  263. });
  264. sm_new_preset.setTitle("Add new Preset");
  265. sm_new_preset.setHeading("Preset Wizard");
  266. sm_new_preset.setText("Moving axes into initial positions. Click to continue.");
  267. sm_wiz_move.setTitle("Initial Movement");
  268. sm_wiz_move.onEnter([]() {
  269. steppers_start_homing();
  270. });
  271. sm_wiz_move.whenIn([](StateMachineInput smi) {
  272. #ifndef DISABLE_HOMING_STATE
  273. if (smi.all_motors_done) {
  274. if (steppers_homed()) {
  275. #endif // DISABLE_HOMING_STATE
  276. states_go_to(&sm_wiz_count);
  277. #ifndef DISABLE_HOMING_STATE
  278. }
  279. }
  280. #endif // DISABLE_HOMING_STATE
  281. static char last_axis = ' ';
  282. char axis = steppers_homing_axis();
  283. if ((axis != ' ') && (axis != last_axis)) {
  284. last_axis = axis;
  285. strbuf = String(F("Currently homing ")) + axis + String(F(" axis"));
  286. sm_wiz_move.setText(strbuf.c_str());
  287. sm_wiz_move.updateText();
  288. }
  289. });
  290. sm_wiz_count.setTitle("Container Count");
  291. sm_wiz_count.setData(0, "X: ", &wizard_preset.count_x, 1, 50);
  292. sm_wiz_count.setData(1, "Y: ", &wizard_preset.count_y, 1, 100);
  293. sm_wiz_first_pos.setTitle("First Container Position");
  294. sm_wiz_first_pos.setData(0, "X: ", &wizard_first_x, X_AXIS_MIN, X_AXIS_MAX);
  295. sm_wiz_first_pos.setData(1, "Y: ", &wizard_first_y, Y_AXIS_MIN, Y_AXIS_MAX);
  296. sm_wiz_first_pos.onEnter([]() {
  297. wizard_first_x = steppers_current_pos(0);
  298. wizard_first_y = steppers_current_pos(1);
  299. });
  300. sm_wiz_first_pos.onLiveUpdate([](size_t i, float v) {
  301. if (i == 0) {
  302. steppers_move_x(v);
  303. } else {
  304. steppers_move_y(v);
  305. }
  306. });
  307. sm_wiz_first_pos.onUpdate([](size_t i, float v) {
  308. if (i == 0) {
  309. wizard_preset.offset_x = v;
  310. } else {
  311. wizard_preset.offset_y = v;
  312. }
  313. });
  314. sm_wiz_last_pos.setTitle("Last Container Position");
  315. sm_wiz_last_pos.setData(0, "X: ", &wizard_last_x, X_AXIS_MIN, X_AXIS_MAX);
  316. sm_wiz_last_pos.setData(1, "Y: ", &wizard_last_y, Y_AXIS_MIN, Y_AXIS_MAX);
  317. sm_wiz_last_pos.onEnter([]() {
  318. wizard_last_x = steppers_current_pos(0);
  319. wizard_last_y = steppers_current_pos(1);
  320. });
  321. sm_wiz_last_pos.onLiveUpdate([](size_t i, float v) {
  322. if (i == 0) {
  323. steppers_move_x(v);
  324. } else {
  325. steppers_move_y(v);
  326. }
  327. });
  328. sm_wiz_last_pos.onUpdate([](size_t i, float v) {
  329. if (i == 0) {
  330. wizard_preset.distance_x = (v - wizard_preset.offset_x) / wizard_preset.count_x;
  331. } else {
  332. wizard_preset.distance_y = (v - wizard_preset.offset_y) / wizard_preset.count_y;
  333. }
  334. });
  335. sm_new_preset_done.setTitle("Preset Wizard Done");
  336. sm_new_preset_done.setHeading("Preset Wizard Done");
  337. sm_new_preset_done.onEnter([]() {
  338. if (!data_preset_add(wizard_preset)) {
  339. strbuf = String(data_eeprom_error()) + F(" Error adding preset!");
  340. sm_new_preset_done.setText(strbuf.c_str());
  341. } else {
  342. strbuf = F("Preset ");
  343. strbuf += String(data_preset_count());
  344. strbuf += F(" has been added! Don't forget to store config to EEPROM to keep it.");
  345. sm_new_preset_done.setText(strbuf.c_str());
  346. }
  347. });
  348. sm_new_preset_done.whenIn([](StateMachineInput smi) {
  349. if (smi.click) {
  350. states_go_to(&sm_auto);
  351. }
  352. });
  353. // TODO
  354. sm_mod_preset.setTitle("Modify Preset");
  355. sm_mod_preset.setPrefix("Preset ");
  356. sm_mod_preset.dataCount([]() {
  357. return (int)data_preset_count();
  358. });
  359. sm_mod_preset.dataGet(preset_name_func);
  360. sm_del_preset.setTitle("Delete Preset");
  361. sm_del_preset.setPrefix("Preset ");
  362. sm_del_preset.dataCount([]() {
  363. return (int)data_preset_count();
  364. });
  365. sm_del_preset.dataGet(preset_name_func);
  366. sm_del_preset.dataCall([](int i) {
  367. data_preset_remove(i);
  368. states_go_to(&sm_auto);
  369. });
  370. // ----------------------------------
  371. sm_config.setTitle("Configuration");
  372. sm_conf_load.setTitle("Load from EEPROM");
  373. sm_conf_load.setHeading("Load from EEPROM");
  374. sm_conf_load.onEnter([]() {
  375. if (!data_eeprom_read()) {
  376. strbuf = String(data_eeprom_error()) + " Error reading configuration!";
  377. sm_conf_load.setText(strbuf.c_str());
  378. } else {
  379. sm_conf_load.setText("Configuration read successful!");
  380. steppers_set_speed_x(data_options()->speed_x);
  381. steppers_set_accel_x(data_options()->accel_x);
  382. steppers_set_speed_y(data_options()->speed_y);
  383. steppers_set_accel_y(data_options()->accel_y);
  384. steppers_set_speed_z(data_options()->speed_z);
  385. steppers_set_accel_z(data_options()->accel_z);
  386. steppers_set_speed_e(data_options()->speed_e);
  387. steppers_set_accel_e(data_options()->accel_e);
  388. }
  389. });
  390. sm_conf_save.setTitle("Store to EEPROM");
  391. sm_conf_save.setHeading("Store to EEPROM");
  392. sm_conf_save.setText("Configuration written to EEPROM!");
  393. sm_conf_save.onEnter([]() {
  394. data_eeprom_write();
  395. });
  396. sm_conf_clear.setTitle("Clear EEPROM & RAM");
  397. sm_conf_clear.setHeading("Clear EEPROM & RAM");
  398. sm_conf_clear.setText("EEPROM and RAM Configuration cleared!");
  399. sm_conf_clear.onEnter([]() {
  400. data_clear();
  401. data_eeprom_write();
  402. // reapply default conf values
  403. steppers_set_speed_x(data_options()->speed_x);
  404. steppers_set_speed_y(data_options()->speed_y);
  405. steppers_set_speed_z(data_options()->speed_z);
  406. steppers_set_speed_e(data_options()->speed_e);
  407. steppers_set_accel_x(data_options()->accel_x);
  408. steppers_set_accel_y(data_options()->accel_y);
  409. steppers_set_accel_z(data_options()->accel_z);
  410. steppers_set_accel_e(data_options()->accel_e);
  411. });
  412. sm_conf_speed_xy.setTitle("XY Speed");
  413. sm_conf_speed_xy.onUpdate([](float v) {
  414. steppers_set_speed_x(v);
  415. steppers_set_speed_y(v);
  416. });
  417. sm_conf_speed_z.setTitle("Z Speed");
  418. sm_conf_speed_z.onUpdate([](float v) {
  419. steppers_set_speed_z(v);
  420. });
  421. sm_conf_speed_e.setTitle("E Speed");
  422. sm_conf_speed_e.onUpdate([](float v) {
  423. steppers_set_speed_e(v);
  424. });
  425. sm_conf_accel_xy.setTitle("XY Acceleration");
  426. sm_conf_accel_xy.onUpdate([](float v) {
  427. steppers_set_accel_x(v);
  428. steppers_set_accel_y(v);
  429. });
  430. sm_conf_accel_z.setTitle("Z Acceleration");
  431. sm_conf_accel_z.onUpdate([](float v) {
  432. steppers_set_accel_z(v);
  433. });
  434. sm_conf_accel_e.setTitle("E Acceleration");
  435. sm_conf_accel_e.onUpdate([](float v) {
  436. steppers_set_accel_e(v);
  437. });
  438. sm_error.setChild(&sm_ask_homing);
  439. sm_error.setTitle("Axis Error");
  440. sm_error.setHeading("Axis Error");
  441. sm_error.setText("An unknown error occured!");
  442. #if defined(ENABLE_MOVEMENT_TEST_STATE) || defined(ENABLE_INPUT_TEST_STATE)
  443. sm_debug.setTitle("Debug Menu");
  444. #endif
  445. #ifdef ENABLE_MOVEMENT_TEST_STATE
  446. sm_movement_test.setTitle("Movement Test");
  447. sm_move_xy_test.setTitle("XY Axes");
  448. sm_move_xy_test.setHeading("XY Move Test");
  449. sm_move_xy_test.setText("Moving axis through full range with selected speed.");
  450. sm_move_xy_test.whenIn([](StateMachineInput smi) {
  451. static bool sx = false, sy = false;
  452. if (smi.motor_done[X_AXIS]) {
  453. sx = !sx;
  454. steppers_move_x(sx ? (X_AXIS_MAX - X_AXIS_EPSILON) : (X_AXIS_MIN + X_AXIS_EPSILON));
  455. }
  456. if (smi.motor_done[Y_AXIS]) {
  457. sy = !sy;
  458. steppers_move_y(sy ? (Y_AXIS_MAX - Y_AXIS_EPSILON) : (Y_AXIS_MIN + Y_AXIS_EPSILON));
  459. }
  460. if (smi.click) {
  461. states_go_to(&sm_movement_test);
  462. }
  463. });
  464. sm_move_x_test.setTitle("X Axis");
  465. sm_move_x_test.setHeading("X Move Test");
  466. sm_move_x_test.setText("Moving axis through full range with selected speed.");
  467. sm_move_x_test.whenIn([](StateMachineInput smi) {
  468. static bool s = false;
  469. if (smi.motor_done[X_AXIS]) {
  470. s = !s;
  471. steppers_move_x(s ? (X_AXIS_MAX - X_AXIS_EPSILON) : (X_AXIS_MIN + X_AXIS_EPSILON));
  472. }
  473. if (smi.click) {
  474. states_go_to(&sm_movement_test);
  475. }
  476. });
  477. sm_move_y_test.setTitle("Y Axis");
  478. sm_move_y_test.setHeading("Y Move Test");
  479. sm_move_y_test.setText("Moving axis through full range with selected speed.");
  480. sm_move_y_test.whenIn([](StateMachineInput smi) {
  481. static bool s = false;
  482. if (smi.motor_done[Y_AXIS]) {
  483. s = !s;
  484. steppers_move_y(s ? (Y_AXIS_MAX - Y_AXIS_EPSILON) : (Y_AXIS_MIN + Y_AXIS_EPSILON));
  485. }
  486. if (smi.click) {
  487. states_go_to(&sm_movement_test);
  488. }
  489. });
  490. sm_move_z_test.setTitle("Z Axis");
  491. sm_move_z_test.setHeading("Z Move Test");
  492. sm_move_z_test.setText("Moving axis through full range with selected speed.");
  493. sm_move_z_test.whenIn([](StateMachineInput smi) {
  494. static bool s = false;
  495. if (smi.motor_done[Z_AXIS]) {
  496. s = !s;
  497. steppers_move_z(s ? (Z_AXIS_MAX - Z_AXIS_EPSILON) : (Z_AXIS_MIN + Z_AXIS_EPSILON));
  498. }
  499. if (smi.click) {
  500. states_go_to(&sm_movement_test);
  501. }
  502. });
  503. sm_move_e_test.setTitle("E Axis");
  504. sm_move_e_test.setHeading("E Move Test");
  505. sm_move_e_test.setText("Moving axis through full range with selected speed.");
  506. sm_move_e_test.whenIn([](StateMachineInput smi) {
  507. static bool s = false;
  508. if (smi.motor_done[E_AXIS]) {
  509. s = !s;
  510. steppers_move_e(s ? (E_AXIS_MAX - E_AXIS_EPSILON) : (E_AXIS_MIN + E_AXIS_EPSILON));
  511. }
  512. if (smi.click) {
  513. states_go_to(&sm_movement_test);
  514. }
  515. });
  516. #endif // ENABLE_MOVEMENT_TEST_STATE
  517. #ifdef ENABLE_INPUT_TEST_STATE
  518. sm_input_test.setTitle("Input Test");
  519. sm_input_test.setHeading("Endstop Test");
  520. sm_input_test.whenIn([](StateMachineInput smi) {
  521. String s = "Endstops: ";
  522. for (int i = 0; i < AXIS_COUNT; i++) {
  523. if (steppers_home_switch(i)) {
  524. s += '1';
  525. } else {
  526. s += '0';
  527. }
  528. if (i < 3) {
  529. s += ' ';
  530. }
  531. }
  532. if (s != strbuf) {
  533. strbuf = s;
  534. sm_input_test.setText(strbuf.c_str());
  535. sm_input_test.updateText();
  536. }
  537. });
  538. #endif // ENABLE_INPUT_TEST_STATE
  539. // ----------------------------------
  540. states_go_to(&sm_ask_homing);
  541. }
  542. void states_run(StateMachineInput smi) {
  543. if (smi.click || smi.kill || (smi.encoder != 0)) {
  544. last_input = millis();
  545. }
  546. if (smi.click) {
  547. async_beep(ENCODER_CLICK_BEEP_TIME, ENCODER_CLICK_BEEP_FREQ);
  548. }
  549. if (smi.kill) {
  550. steppers_kill();
  551. Serial.println(F("Kill button pressed!"));
  552. blocking_beep(KILL_BEEP_TIME, KILL_BEEP_FREQ, KILL_BEEP_REPEAT - 1);
  553. states_go_to(&sm_ask_homing);
  554. return;
  555. }
  556. #if (IDLE_TIMEOUT > 0)
  557. unsigned long last_time = max(last_input, last_state_change);
  558. if ((millis() - last_time) > IDLE_TIMEOUT) {
  559. if (current_state != &sm_ask_homing) {
  560. Serial.println(F("Idle timeout, disabling motors"));
  561. steppers_kill();
  562. states_go_to(&sm_ask_homing);
  563. return;
  564. }
  565. }
  566. #endif // (IDLE_TIMEOUT > 0)
  567. if (current_state != NULL) {
  568. current_state->inState(smi);
  569. }
  570. }
  571. State *states_get(void) {
  572. return current_state;
  573. }
  574. void states_go_to(State *state) {
  575. last_state_change = millis();
  576. current_state = state;
  577. if (current_state != NULL) {
  578. current_state->enterState();
  579. }
  580. }