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.

statemachine.cpp 12KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544
  1. #include <Arduino.h>
  2. #include "config.h"
  3. #include "config_pins.h"
  4. #include "lcd.h"
  5. #include "states.h"
  6. State::State(State *_parent) : parent(_parent), child(NULL), title("no title") {
  7. if (_parent != NULL) {
  8. _parent->setChild(this);
  9. }
  10. }
  11. // --------------------------------------
  12. StateText::StateText(State *_parent) : State(_parent) {
  13. heading = "no heading";
  14. text = "text missing";
  15. onEnterFunc = []() { };
  16. whenInFunc = [](StateMachineInput smi) {
  17. State *s = states_get();
  18. if (smi.click && (s != NULL)) {
  19. if (s->getChild() != NULL) {
  20. states_go_to(s->getChild());
  21. } else if (s->getParent() != NULL) {
  22. states_go_to(s->getParent());
  23. }
  24. }
  25. };
  26. }
  27. void StateText::setHeading(const char *_heading) {
  28. heading = _heading;
  29. }
  30. void StateText::setText(const char *_text) {
  31. text = _text;
  32. }
  33. void StateText::onEnter(EnterFuncPtr func) {
  34. onEnterFunc = func;
  35. }
  36. void StateText::whenIn(InFuncPtr func) {
  37. whenInFunc = func;
  38. }
  39. void StateText::updateText(void) {
  40. lcd_clear();
  41. if (heading != NULL) {
  42. lcd_set_heading(heading);
  43. } else if (getTitle() != NULL) {
  44. lcd_set_heading(getTitle());
  45. }
  46. if (text != NULL) {
  47. lcd_set_text(text);
  48. }
  49. }
  50. void StateText::enterState(void) {
  51. if (onEnterFunc != NULL) {
  52. onEnterFunc();
  53. }
  54. updateText();
  55. }
  56. void StateText::inState(struct StateMachineInput smi) {
  57. if (whenInFunc != NULL) {
  58. whenInFunc(smi);
  59. }
  60. }
  61. // --------------------------------------
  62. StateMenu::StateMenu(State *_parent, bool _show_parent) : State(_parent) {
  63. show_parent = _show_parent;
  64. menuPos = 0;
  65. menuOff = 0;
  66. }
  67. void StateMenu::setChild(State *_child) {
  68. children.push_back(_child);
  69. }
  70. void StateMenu::addChild(State *_child, int pos) {
  71. if (pos < 0) {
  72. setChild(_child);
  73. } else {
  74. array_insert_at_pos(&children, _child, pos);
  75. }
  76. }
  77. void StateMenu::enterState(void) {
  78. display();
  79. }
  80. void StateMenu::display(void) {
  81. lcd_clear();
  82. lcd_set_heading(getTitle());
  83. int size = children.size();
  84. if (show_parent) {
  85. size++;
  86. }
  87. for (int i = menuOff; (i < menuOff + lcd_text_lines()) && (i < size); i++) {
  88. String s;
  89. if (i == menuPos) {
  90. s = F("> ");
  91. } else {
  92. s = F(" ");
  93. }
  94. if (i == children.size()) {
  95. s += getParent()->getTitle();
  96. } else {
  97. s += children.at(i)->getTitle();
  98. }
  99. lcd_set_menu_text(i - menuOff, s.c_str());
  100. }
  101. }
  102. void StateMenu::inState(struct StateMachineInput smi) {
  103. int size = children.size();
  104. if (show_parent) {
  105. size++;
  106. }
  107. if (smi.encoder != 0) {
  108. menuPos -= smi.encoder;
  109. while (menuPos < 0) {
  110. menuPos += size;
  111. }
  112. while (menuPos >= size) {
  113. menuPos -= size;
  114. }
  115. while (menuPos < menuOff) {
  116. menuOff--;
  117. }
  118. while (menuPos >= (menuOff + lcd_text_lines())) {
  119. menuOff++;
  120. }
  121. display();
  122. }
  123. if (smi.click) {
  124. if (menuPos == children.size()) {
  125. menuPos = 0;
  126. menuOff = 0;
  127. states_go_to(getParent());
  128. } else {
  129. states_go_to(children.at(menuPos));
  130. }
  131. }
  132. }
  133. // --------------------------------------
  134. StateDynamicMenu::StateDynamicMenu(State *_parent) : State(_parent) {
  135. menuPos = 0;
  136. menuOff = 0;
  137. prefix = "";
  138. countFunc = NULL;
  139. getFunc = NULL;
  140. callFunc = NULL;
  141. }
  142. void StateDynamicMenu::dataCount(CountFuncPtr count) {
  143. countFunc = count;
  144. }
  145. void StateDynamicMenu::dataGet(GetFuncPtr get) {
  146. getFunc = get;
  147. }
  148. void StateDynamicMenu::dataCall(CallFuncPtr call) {
  149. callFunc = call;
  150. }
  151. void StateDynamicMenu::display(void) {
  152. lcd_clear();
  153. lcd_set_heading(getTitle());
  154. for (int i = menuOff; (i < menuOff + lcd_text_lines()) && (i < count + 1); i++) {
  155. String s;
  156. if (i == menuPos) {
  157. s = F("> ");
  158. } else {
  159. s = F(" ");
  160. }
  161. if (i == count) {
  162. s += getParent()->getTitle();
  163. } else {
  164. s += contents.at(i);
  165. }
  166. lcd_set_menu_text(i - menuOff, s.c_str());
  167. }
  168. }
  169. void StateDynamicMenu::setPrefix(String pre) {
  170. prefix = pre;
  171. }
  172. void StateDynamicMenu::enterState(void) {
  173. // cache all entries on entering state
  174. if (countFunc != NULL) {
  175. count = countFunc();
  176. } else {
  177. count = 0;
  178. }
  179. contents.clear();
  180. for (int i = 0; i < count; i++) {
  181. if (getFunc != NULL) {
  182. contents.push_back(prefix + String(getFunc(i)));
  183. } else {
  184. contents.push_back(prefix + String(i + 1));
  185. }
  186. }
  187. display();
  188. }
  189. void StateDynamicMenu::inState(StateMachineInput smi) {
  190. if (smi.encoder != 0) {
  191. menuPos -= smi.encoder;
  192. while (menuPos < 0) {
  193. menuPos += count + 1;
  194. }
  195. while (menuPos >= count + 1) {
  196. menuPos -= count + 1;
  197. }
  198. while (menuPos < menuOff) {
  199. menuOff--;
  200. }
  201. while (menuPos >= (menuOff + lcd_text_lines())) {
  202. menuOff++;
  203. }
  204. display();
  205. }
  206. if (smi.click) {
  207. if (menuPos == count) {
  208. menuPos = 0;
  209. menuOff = 0;
  210. states_go_to(getParent());
  211. } else {
  212. if (callFunc != NULL) {
  213. callFunc(menuPos);
  214. }
  215. }
  216. }
  217. }
  218. // --------------------------------------
  219. template <typename T>
  220. StateValue<T>::StateValue(State *_parent, T &_value, T _min, T _max) : State(_parent), value(_value) {
  221. min = _min;
  222. max = _max;
  223. heading = NULL;
  224. text = NULL;
  225. onEnterFunc = NULL;
  226. updateFunc = NULL;
  227. updateLiveFunc = NULL;
  228. }
  229. template <typename T>
  230. void StateValue<T>::setHeading(const char *_heading) {
  231. heading = _heading;
  232. }
  233. template <typename T>
  234. void StateValue<T>::setText(const char *_text) {
  235. text = _text;
  236. }
  237. template <typename T>
  238. void StateValue<T>::onEnter(EnterFuncPtr func) {
  239. onEnterFunc = func;
  240. }
  241. template <typename T>
  242. void StateValue<T>::onUpdate(UpdateFuncPtr func) {
  243. updateFunc = func;
  244. }
  245. template <typename T>
  246. void StateValue<T>::onLiveUpdate(UpdateFuncPtr func) {
  247. updateLiveFunc = func;
  248. }
  249. template <typename T>
  250. void StateValue<T>::display(void) {
  251. lcd_clear();
  252. if (heading == NULL) {
  253. lcd_set_heading(getTitle());
  254. } else {
  255. lcd_set_heading(heading);
  256. }
  257. String s = String(min) + F(" .. ") + String(value) + F(" .. ") + String(max);
  258. if (text != NULL) {
  259. s = text + String(F("\n")) + s;
  260. }
  261. lcd_set_text(s.c_str());
  262. }
  263. template <typename T>
  264. void StateValue<T>::enterState(void) {
  265. if (onEnterFunc != NULL) {
  266. onEnterFunc();
  267. }
  268. display();
  269. }
  270. template <typename T>
  271. void StateValue<T>::inState(StateMachineInput smi) {
  272. if (smi.encoder != 0) {
  273. float vf = smi.encoder;
  274. vf *= 1.0 + ((float)smi.rpm / ENCODER_RPM_VALUE_FACTOR);
  275. int v = vf;
  276. value -= v;
  277. if (value < min) {
  278. value = min;
  279. }
  280. if (value > max) {
  281. value = max;
  282. }
  283. if (updateLiveFunc != NULL) {
  284. updateLiveFunc(value);
  285. }
  286. display();
  287. }
  288. if (smi.click) {
  289. if (updateFunc != NULL) {
  290. updateFunc(value);
  291. }
  292. states_go_to(getParent());
  293. }
  294. }
  295. template class StateValue<int>;
  296. template class StateValue<float>;
  297. // --------------------------------------
  298. template <typename T, size_t N>
  299. StateValues<T, N>::StateValues(State *_parent) : State(_parent) {
  300. heading = NULL;
  301. onEnterFunc = NULL;
  302. updateFunc = NULL;
  303. updateLiveFunc = NULL;
  304. pos = 0;
  305. editing = false;
  306. }
  307. template <typename T, size_t N>
  308. void StateValues<T, N>::setData(size_t index, const char *name, T *value, T min, T max) {
  309. if (index >= N) {
  310. return;
  311. }
  312. values[index] = value;
  313. mins[index] = min;
  314. maxs[index] = max;
  315. texts[index] = name;
  316. }
  317. template <typename T, size_t N>
  318. void StateValues<T, N>::setHeading(const char *_heading) {
  319. heading = _heading;
  320. }
  321. template <typename T, size_t N>
  322. void StateValues<T, N>::onEnter(EnterFuncPtr func) {
  323. onEnterFunc = func;
  324. }
  325. template <typename T, size_t N>
  326. void StateValues<T, N>::onUpdate(UpdateFuncPtr func) {
  327. updateFunc = func;
  328. }
  329. template <typename T, size_t N>
  330. void StateValues<T, N>::onLiveUpdate(UpdateFuncPtr func) {
  331. updateLiveFunc = func;
  332. }
  333. template <typename T, size_t N>
  334. void StateValues<T, N>::display(void) {
  335. lcd_clear();
  336. if (heading == NULL) {
  337. lcd_set_heading(getTitle());
  338. } else {
  339. lcd_set_heading(heading);
  340. }
  341. for (size_t i = 0; i < (N + 1); i++) {
  342. String s;
  343. if (i == pos) {
  344. if (editing) {
  345. s = F("# ");
  346. } else {
  347. s = F("> ");
  348. }
  349. } else {
  350. s = F(" ");
  351. }
  352. if (i < N) {
  353. s += texts[i] + String(*(values[i])) + F(" (") + String(mins[i]) + F("/") + String(maxs[i]) + F(")");
  354. } else {
  355. if (getChild() != NULL) {
  356. s += F("Continue");
  357. } else {
  358. s += F("Done");
  359. }
  360. }
  361. lcd_set_menu_text(i, s.c_str());
  362. }
  363. }
  364. template <typename T, size_t N>
  365. void StateValues<T, N>::enterState(void) {
  366. pos = 0;
  367. if (onEnterFunc != NULL) {
  368. onEnterFunc();
  369. }
  370. display();
  371. }
  372. template <typename T, size_t N>
  373. void StateValues<T, N>::inState(StateMachineInput smi) {
  374. if (editing) {
  375. if (smi.encoder != 0) {
  376. float vf = smi.encoder;
  377. vf *= 1.0 + ((float)smi.rpm / ENCODER_RPM_VALUE_FACTOR);
  378. int v = vf;
  379. *(values[pos]) -= v;
  380. if (*(values[pos]) < mins[pos]) {
  381. *(values[pos]) = mins[pos];
  382. }
  383. if (*(values[pos]) > maxs[pos]) {
  384. *(values[pos]) = maxs[pos];
  385. }
  386. if (updateLiveFunc != NULL) {
  387. updateLiveFunc(pos, *(values[pos]));
  388. }
  389. display();
  390. }
  391. if (smi.click) {
  392. editing = false;
  393. display();
  394. }
  395. } else {
  396. if (smi.encoder != 0) {
  397. int tmp = pos;
  398. tmp -= smi.encoder;
  399. while (tmp < 0) {
  400. tmp += N + 1;
  401. }
  402. while (tmp >= (N + 1)) {
  403. tmp -= N + 1;
  404. }
  405. pos = tmp;
  406. display();
  407. }
  408. if (smi.click) {
  409. if (pos < N) {
  410. editing = true;
  411. display();
  412. } else {
  413. if (updateFunc != NULL) {
  414. for (size_t i = 0; i < N; i++) {
  415. updateFunc(i, *(values[i]));
  416. }
  417. }
  418. if (getChild() != NULL) {
  419. states_go_to(getChild());
  420. } else if (getParent() != NULL) {
  421. states_go_to(getParent());
  422. }
  423. }
  424. }
  425. }
  426. }
  427. template class StateValues<uint8_t, 2>;
  428. template class StateValues<float, 2>;
  429. // --------------------------------------
  430. template <typename T, size_t N>
  431. void array_print(Array<T, N> *arr) {
  432. Serial.print(F("Array length: "));
  433. Serial.print(arr->size());
  434. Serial.println(F(" contents:"));
  435. for (int i = 0; i < arr->size(); i++) {
  436. Serial.print(i);
  437. Serial.print(F(": "));
  438. Serial.println(arr->at(i)->getTitle());
  439. }
  440. }
  441. template <typename T, size_t N>
  442. void array_insert_at_pos(Array<T, N> *arr, T value, size_t pos) {
  443. // make additional space
  444. arr->push_back(value);
  445. if ((pos >= arr->max_size()) || (pos >= arr->size())) {
  446. // we can not shift it to the given position
  447. return;
  448. }
  449. for (int i = arr->size() - 2; i >= pos; i--) {
  450. arr->at(i + 1) = arr->at(i);
  451. }
  452. arr->at(pos) = value;
  453. }