My Marlin configs for Fabrikator Mini and CTC i3 Pro B
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.

DGUSDisplay.cpp 45KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347
  1. /**
  2. * Marlin 3D Printer Firmware
  3. * Copyright (c) 2020 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
  4. *
  5. * Based on Sprinter and grbl.
  6. * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
  7. *
  8. * This program is free software: you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License as published by
  10. * the Free Software Foundation, either version 3 of the License, or
  11. * (at your option) any later version.
  12. *
  13. * This program is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU General Public License
  19. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  20. *
  21. */
  22. /* DGUS implementation written by coldtobi in 2019 for Marlin */
  23. #include "../../../../inc/MarlinConfigPre.h"
  24. #if HAS_DGUS_LCD
  25. #if HOTENDS > 2
  26. #error "More than 2 hotends not implemented on the Display UI design."
  27. #endif
  28. #include "../../ui_api.h"
  29. #include "../../../../MarlinCore.h"
  30. #include "../../../../module/temperature.h"
  31. #include "../../../../module/motion.h"
  32. #include "../../../../gcode/queue.h"
  33. #include "../../../../module/planner.h"
  34. #include "../../../../sd/cardreader.h"
  35. #include "../../../../libs/duration_t.h"
  36. #include "../../../../module/printcounter.h"
  37. #if ENABLED(POWER_LOSS_RECOVERY)
  38. #include "../../../../feature/powerloss.h"
  39. #endif
  40. #include "DGUSDisplay.h"
  41. #include "DGUSVPVariable.h"
  42. #include "DGUSDisplayDef.h"
  43. // Preamble... 2 Bytes, usually 0x5A 0xA5, but configurable
  44. constexpr uint8_t DGUS_HEADER1 = 0x5A;
  45. constexpr uint8_t DGUS_HEADER2 = 0xA5;
  46. constexpr uint8_t DGUS_CMD_WRITEVAR = 0x82;
  47. constexpr uint8_t DGUS_CMD_READVAR = 0x83;
  48. #if ENABLED(DEBUG_DGUSLCD)
  49. bool dguslcd_local_debug; // = false;
  50. #endif
  51. #if ENABLED(DGUS_FILAMENT_LOADUNLOAD)
  52. typedef struct {
  53. ExtUI::extruder_t extruder; // which extruder to operate
  54. uint8_t action; // load or unload
  55. bool heated; // heating done ?
  56. float purge_length; // the length to extrude before unload, prevent filament jam
  57. } filament_data_t;
  58. static filament_data_t filament_data;
  59. #endif
  60. uint16_t DGUSScreenVariableHandler::ConfirmVP;
  61. #if ENABLED(SDSUPPORT)
  62. int16_t DGUSScreenVariableHandler::top_file = 0;
  63. int16_t DGUSScreenVariableHandler::file_to_print = 0;
  64. static ExtUI::FileList filelist;
  65. #endif
  66. void (*DGUSScreenVariableHandler::confirm_action_cb)() = nullptr;
  67. //DGUSScreenVariableHandler ScreenHandler;
  68. DGUSLCD_Screens DGUSScreenVariableHandler::current_screen;
  69. DGUSLCD_Screens DGUSScreenVariableHandler::past_screens[NUM_PAST_SCREENS];
  70. uint8_t DGUSScreenVariableHandler::update_ptr;
  71. uint16_t DGUSScreenVariableHandler::skipVP;
  72. bool DGUSScreenVariableHandler::ScreenComplete;
  73. //DGUSDisplay dgusdisplay;
  74. rx_datagram_state_t DGUSDisplay::rx_datagram_state = DGUS_IDLE;
  75. uint8_t DGUSDisplay::rx_datagram_len = 0;
  76. bool DGUSDisplay::Initialized = false;
  77. bool DGUSDisplay::no_reentrance = false;
  78. #define dgusserial DGUS_SERIAL
  79. // endianness swap
  80. uint16_t swap16(const uint16_t value) { return (value & 0xffU) << 8U | (value >> 8U); }
  81. bool populate_VPVar(const uint16_t VP, DGUS_VP_Variable * const ramcopy) {
  82. // DEBUG_ECHOPAIR("populate_VPVar ", VP);
  83. const DGUS_VP_Variable *pvp = DGUSLCD_FindVPVar(VP);
  84. // DEBUG_ECHOLNPAIR(" pvp ", (uint16_t )pvp);
  85. if (!pvp) return false;
  86. memcpy_P(ramcopy, pvp, sizeof(DGUS_VP_Variable));
  87. return true;
  88. }
  89. void DGUSScreenVariableHandler::sendinfoscreen(const char* line1, const char* line2, const char* line3, const char* line4, bool l1inflash, bool l2inflash, bool l3inflash, bool l4inflash) {
  90. DGUS_VP_Variable ramcopy;
  91. if (populate_VPVar(VP_MSGSTR1, &ramcopy)) {
  92. ramcopy.memadr = (void*) line1;
  93. l1inflash ? DGUSScreenVariableHandler::DGUSLCD_SendStringToDisplayPGM(ramcopy) : DGUSScreenVariableHandler::DGUSLCD_SendStringToDisplay(ramcopy);
  94. }
  95. if (populate_VPVar(VP_MSGSTR2, &ramcopy)) {
  96. ramcopy.memadr = (void*) line2;
  97. l2inflash ? DGUSScreenVariableHandler::DGUSLCD_SendStringToDisplayPGM(ramcopy) : DGUSScreenVariableHandler::DGUSLCD_SendStringToDisplay(ramcopy);
  98. }
  99. if (populate_VPVar(VP_MSGSTR3, &ramcopy)) {
  100. ramcopy.memadr = (void*) line3;
  101. l3inflash ? DGUSScreenVariableHandler::DGUSLCD_SendStringToDisplayPGM(ramcopy) : DGUSScreenVariableHandler::DGUSLCD_SendStringToDisplay(ramcopy);
  102. }
  103. if (populate_VPVar(VP_MSGSTR4, &ramcopy)) {
  104. ramcopy.memadr = (void*) line4;
  105. l4inflash ? DGUSScreenVariableHandler::DGUSLCD_SendStringToDisplayPGM(ramcopy) : DGUSScreenVariableHandler::DGUSLCD_SendStringToDisplay(ramcopy);
  106. }
  107. }
  108. void DGUSScreenVariableHandler::HandleUserConfirmationPopUp(uint16_t VP, const char* line1, const char* line2, const char* line3, const char* line4, bool l1, bool l2, bool l3, bool l4) {
  109. if (current_screen == DGUSLCD_SCREEN_CONFIRM) {
  110. // Already showing a pop up, so we need to cancel that first.
  111. PopToOldScreen();
  112. }
  113. ConfirmVP = VP;
  114. sendinfoscreen(line1, line2, line3, line4, l1, l2, l3, l4);
  115. ScreenHandler.GotoScreen(DGUSLCD_SCREEN_CONFIRM);
  116. }
  117. void DGUSScreenVariableHandler::setstatusmessage(const char *msg) {
  118. DGUS_VP_Variable ramcopy;
  119. if (populate_VPVar(VP_M117, &ramcopy)) {
  120. ramcopy.memadr = (void*) msg;
  121. DGUSLCD_SendStringToDisplay(ramcopy);
  122. }
  123. }
  124. void DGUSScreenVariableHandler::setstatusmessagePGM(PGM_P const msg) {
  125. DGUS_VP_Variable ramcopy;
  126. if (populate_VPVar(VP_M117, &ramcopy)) {
  127. ramcopy.memadr = (void*) msg;
  128. DGUSLCD_SendStringToDisplayPGM(ramcopy);
  129. }
  130. }
  131. // Send an 8 bit or 16 bit value to the display.
  132. void DGUSScreenVariableHandler::DGUSLCD_SendWordValueToDisplay(DGUS_VP_Variable &var) {
  133. if (var.memadr) {
  134. //DEBUG_ECHOPAIR(" DGUS_LCD_SendWordValueToDisplay ", var.VP);
  135. //DEBUG_ECHOLNPAIR(" data ", *(uint16_t *)var.memadr);
  136. uint8_t *tmp = (uint8_t *) var.memadr;
  137. uint16_t data_to_send = (tmp[0] << 8);
  138. if (var.size >= 1) data_to_send |= tmp[1];
  139. dgusdisplay.WriteVariable(var.VP, data_to_send);
  140. }
  141. }
  142. // Send an uint8_t between 0 and 255 to the display, but scale to a percentage (0..100)
  143. void DGUSScreenVariableHandler::DGUSLCD_SendPercentageToDisplay(DGUS_VP_Variable &var) {
  144. if (var.memadr) {
  145. //DEBUG_ECHOPAIR(" DGUS_LCD_SendWordValueToDisplay ", var.VP);
  146. //DEBUG_ECHOLNPAIR(" data ", *(uint16_t *)var.memadr);
  147. uint16_t tmp = *(uint8_t *) var.memadr +1 ; // +1 -> avoid rounding issues for the display.
  148. tmp = map(tmp, 0, 255, 0, 100);
  149. uint16_t data_to_send = swap16(tmp);
  150. dgusdisplay.WriteVariable(var.VP, data_to_send);
  151. }
  152. }
  153. // Send the current print progress to the display.
  154. void DGUSScreenVariableHandler::DGUSLCD_SendPrintProgressToDisplay(DGUS_VP_Variable &var) {
  155. //DEBUG_ECHOPAIR(" DGUSLCD_SendPrintProgressToDisplay ", var.VP);
  156. uint16_t tmp = ExtUI::getProgress_percent();
  157. //DEBUG_ECHOLNPAIR(" data ", tmp);
  158. uint16_t data_to_send = swap16(tmp);
  159. dgusdisplay.WriteVariable(var.VP, data_to_send);
  160. }
  161. // Send the current print time to the display.
  162. // It is using a hex display for that: It expects BSD coded data in the format xxyyzz
  163. void DGUSScreenVariableHandler::DGUSLCD_SendPrintTimeToDisplay(DGUS_VP_Variable &var) {
  164. duration_t elapsed = print_job_timer.duration();
  165. char buf[32];
  166. elapsed.toString(buf);
  167. dgusdisplay.WriteVariable(VP_PrintTime, buf, var.size, true);
  168. }
  169. // Send an uint8_t between 0 and 100 to a variable scale to 0..255
  170. void DGUSScreenVariableHandler::DGUSLCD_PercentageToUint8(DGUS_VP_Variable &var, void *val_ptr) {
  171. if (var.memadr) {
  172. uint16_t value = swap16(*(uint16_t*)val_ptr);
  173. *(uint8_t*)var.memadr = map(constrain(value, 0, 100), 0, 100, 0, 255);
  174. }
  175. }
  176. // Sends a (RAM located) string to the DGUS Display
  177. // (Note: The DGUS Display does not clear after the \0, you have to
  178. // overwrite the remainings with spaces.// var.size has the display buffer size!
  179. void DGUSScreenVariableHandler::DGUSLCD_SendStringToDisplay(DGUS_VP_Variable &var) {
  180. char *tmp = (char*) var.memadr;
  181. dgusdisplay.WriteVariable(var.VP, tmp, var.size, true);
  182. }
  183. // Sends a (flash located) string to the DGUS Display
  184. // (Note: The DGUS Display does not clear after the \0, you have to
  185. // overwrite the remainings with spaces.// var.size has the display buffer size!
  186. void DGUSScreenVariableHandler::DGUSLCD_SendStringToDisplayPGM(DGUS_VP_Variable &var) {
  187. char *tmp = (char*) var.memadr;
  188. dgusdisplay.WriteVariablePGM(var.VP, tmp, var.size, true);
  189. }
  190. #if HAS_PID_HEATING
  191. void DGUSScreenVariableHandler::DGUSLCD_SendTemperaturePID(DGUS_VP_Variable &var) {
  192. float value = *(float *)var.memadr;
  193. float valuesend = 0;
  194. switch (var.VP) {
  195. default: return;
  196. #if HOTENDS >= 1
  197. case VP_E0_PID_P: valuesend = value; break;
  198. case VP_E0_PID_I: valuesend = unscalePID_i(value); break;
  199. case VP_E0_PID_D: valuesend = unscalePID_d(value); break;
  200. #endif
  201. #if HOTENDS >= 2
  202. case VP_E1_PID_P: valuesend = value; break;
  203. case VP_E1_PID_I: valuesend = unscalePID_i(value); break;
  204. case VP_E1_PID_D: valuesend = unscalePID_d(value); break;
  205. #endif
  206. #if HAS_HEATED_BED
  207. case VP_BED_PID_P: valuesend = value; break;
  208. case VP_BED_PID_I: valuesend = unscalePID_i(value); break;
  209. case VP_BED_PID_D: valuesend = unscalePID_d(value); break;
  210. #endif
  211. }
  212. valuesend *= cpow(10, 1);
  213. union { int16_t i; char lb[2]; } endian;
  214. char tmp[2];
  215. endian.i = valuesend;
  216. tmp[0] = endian.lb[1];
  217. tmp[1] = endian.lb[0];
  218. dgusdisplay.WriteVariable(var.VP, tmp, 2);
  219. }
  220. #endif
  221. #if ENABLED(PRINTCOUNTER)
  222. // Send the accumulate print time to the display.
  223. // It is using a hex display for that: It expects BSD coded data in the format xxyyzz
  224. void DGUSScreenVariableHandler::DGUSLCD_SendPrintAccTimeToDisplay(DGUS_VP_Variable &var) {
  225. printStatistics state = print_job_timer.getStats();
  226. char buf[21];
  227. duration_t elapsed = state.printTime;
  228. elapsed.toString(buf);
  229. dgusdisplay.WriteVariable(VP_PrintAccTime, buf, var.size, true);
  230. }
  231. void DGUSScreenVariableHandler::DGUSLCD_SendPrintsTotalToDisplay(DGUS_VP_Variable &var) {
  232. printStatistics state = print_job_timer.getStats();
  233. char buf[21];
  234. sprintf_P(buf, PSTR("%u"), state.totalPrints);
  235. dgusdisplay.WriteVariable(VP_PrintsTotal, buf, var.size, true);
  236. }
  237. #endif
  238. // Send fan status value to the display.
  239. #if HAS_FAN
  240. void DGUSScreenVariableHandler::DGUSLCD_SendFanStatusToDisplay(DGUS_VP_Variable &var) {
  241. if (var.memadr) {
  242. DEBUG_ECHOPAIR(" DGUSLCD_SendFanStatusToDisplay ", var.VP);
  243. DEBUG_ECHOLNPAIR(" data ", *(uint8_t *)var.memadr);
  244. uint16_t data_to_send = 0;
  245. if (*(uint8_t *) var.memadr) data_to_send = 1;
  246. data_to_send = swap16(data_to_send);
  247. dgusdisplay.WriteVariable(var.VP, data_to_send);
  248. }
  249. }
  250. #endif
  251. // Send heater status value to the display.
  252. void DGUSScreenVariableHandler::DGUSLCD_SendHeaterStatusToDisplay(DGUS_VP_Variable &var) {
  253. if (var.memadr) {
  254. DEBUG_ECHOPAIR(" DGUSLCD_SendHeaterStatusToDisplay ", var.VP);
  255. DEBUG_ECHOLNPAIR(" data ", *(int16_t *)var.memadr);
  256. uint16_t data_to_send = 0;
  257. if (*(int16_t *) var.memadr) data_to_send = 1;
  258. data_to_send = swap16(data_to_send);
  259. dgusdisplay.WriteVariable(var.VP, data_to_send);
  260. }
  261. }
  262. #if ENABLED(DGUS_UI_WAITING)
  263. void DGUSScreenVariableHandler::DGUSLCD_SendWaitingStatusToDisplay(DGUS_VP_Variable &var) {
  264. // In FYSETC UI design there are 10 statuses to loop
  265. static uint16_t period = 0;
  266. static uint16_t index = 0;
  267. //DEBUG_ECHOPAIR(" DGUSLCD_SendWaitingStatusToDisplay ", var.VP);
  268. //DEBUG_ECHOLNPAIR(" data ", swap16(index));
  269. if (period++ > DGUS_UI_WAITING_STATUS_PERIOD) {
  270. dgusdisplay.WriteVariable(var.VP, swap16(index));
  271. //DEBUG_ECHOLNPAIR(" data ", swap16(index));
  272. if (++index >= DGUS_UI_WAITING_STATUS) index = 0;
  273. period = 0;
  274. }
  275. }
  276. #endif
  277. #if ENABLED(SDSUPPORT)
  278. void DGUSScreenVariableHandler::ScreenChangeHookIfSD(DGUS_VP_Variable &var, void *val_ptr) {
  279. // default action executed when there is a SD card, but not printing
  280. if (ExtUI::isMediaInserted() && !ExtUI::isPrintingFromMedia()) {
  281. ScreenChangeHook(var, val_ptr);
  282. dgusdisplay.RequestScreen(current_screen);
  283. return;
  284. }
  285. // if we are printing, we jump to two screens after the requested one.
  286. // This should host e.g a print pause / print abort / print resume dialog.
  287. // This concept allows to recycle this hook for other file
  288. if (ExtUI::isPrintingFromMedia() && !card.flag.abort_sd_printing) {
  289. GotoScreen(DGUSLCD_SCREEN_SDPRINTMANIPULATION);
  290. return;
  291. }
  292. // Don't let the user in the dark why there is no reaction.
  293. if (!ExtUI::isMediaInserted()) {
  294. setstatusmessagePGM(GET_TEXT(MSG_NO_MEDIA));
  295. return;
  296. }
  297. if (card.flag.abort_sd_printing) {
  298. setstatusmessagePGM(GET_TEXT(MSG_MEDIA_ABORTING));
  299. return;
  300. }
  301. }
  302. void DGUSScreenVariableHandler::DGUSLCD_SD_ScrollFilelist(DGUS_VP_Variable& var, void *val_ptr) {
  303. auto old_top = top_file;
  304. const int16_t scroll = (int16_t)swap16(*(uint16_t*)val_ptr);
  305. if (scroll) {
  306. top_file += scroll;
  307. DEBUG_ECHOPAIR("new topfile calculated:", top_file);
  308. if (top_file < 0) {
  309. top_file = 0;
  310. DEBUG_ECHOLNPGM("Top of filelist reached");
  311. }
  312. else {
  313. int16_t max_top = filelist.count() - DGUS_SD_FILESPERSCREEN;
  314. NOLESS(max_top, 0);
  315. NOMORE(top_file, max_top);
  316. }
  317. DEBUG_ECHOPAIR("new topfile adjusted:", top_file);
  318. }
  319. else if (!filelist.isAtRootDir()) {
  320. filelist.upDir();
  321. top_file = 0;
  322. ForceCompleteUpdate();
  323. }
  324. if (old_top != top_file) ForceCompleteUpdate();
  325. }
  326. void DGUSScreenVariableHandler::DGUSLCD_SD_FileSelected(DGUS_VP_Variable &var, void *val_ptr) {
  327. uint16_t touched_nr = (int16_t)swap16(*(uint16_t*)val_ptr) + top_file;
  328. if (touched_nr > filelist.count()) return;
  329. if (!filelist.seek(touched_nr)) return;
  330. if (filelist.isDir()) {
  331. filelist.changeDir(filelist.filename());
  332. top_file = 0;
  333. ForceCompleteUpdate();
  334. return;
  335. }
  336. #if ENABLED(DGUS_PRINT_FILENAME)
  337. // Send print filename
  338. dgusdisplay.WriteVariable(VP_SD_Print_Filename, filelist.filename(), VP_SD_FileName_LEN, true);
  339. #endif
  340. // Setup Confirmation screen
  341. file_to_print = touched_nr;
  342. HandleUserConfirmationPopUp(VP_SD_FileSelectConfirm, nullptr, PSTR("Print file"), filelist.filename(), PSTR("from SD Card?"), true, true, false, true);
  343. }
  344. void DGUSScreenVariableHandler::DGUSLCD_SD_StartPrint(DGUS_VP_Variable &var, void *val_ptr) {
  345. if (!filelist.seek(file_to_print)) return;
  346. ExtUI::printFile(filelist.shortFilename());
  347. ScreenHandler.GotoScreen(
  348. #if ENABLED(DGUS_LCD_UI_ORIGIN)
  349. DGUSLCD_SCREEN_STATUS
  350. #else
  351. DGUSLCD_SCREEN_SDPRINTMANIPULATION
  352. #endif
  353. );
  354. }
  355. void DGUSScreenVariableHandler::DGUSLCD_SD_ResumePauseAbort(DGUS_VP_Variable &var, void *val_ptr) {
  356. if (!ExtUI::isPrintingFromMedia()) return; // avoid race condition when user stays in this menu and printer finishes.
  357. switch (swap16(*(uint16_t*)val_ptr)) {
  358. case 0: // Resume
  359. if (ExtUI::isPrintingFromMediaPaused()) ExtUI::resumePrint();
  360. break;
  361. case 1: // Pause
  362. if (!ExtUI::isPrintingFromMediaPaused()) ExtUI::pausePrint();
  363. break;
  364. case 2: // Abort
  365. ScreenHandler.HandleUserConfirmationPopUp(VP_SD_AbortPrintConfirmed, nullptr, PSTR("Abort printing"), filelist.filename(), PSTR("?"), true, true, false, true);
  366. break;
  367. }
  368. }
  369. void DGUSScreenVariableHandler::DGUSLCD_SD_ReallyAbort(DGUS_VP_Variable &var, void *val_ptr) {
  370. ExtUI::stopPrint();
  371. GotoScreen(DGUSLCD_SCREEN_MAIN);
  372. }
  373. void DGUSScreenVariableHandler::DGUSLCD_SD_PrintTune(DGUS_VP_Variable &var, void *val_ptr) {
  374. if (!ExtUI::isPrintingFromMedia()) return; // avoid race condition when user stays in this menu and printer finishes.
  375. GotoScreen(DGUSLCD_SCREEN_SDPRINTTUNE);
  376. }
  377. void DGUSScreenVariableHandler::DGUSLCD_SD_SendFilename(DGUS_VP_Variable& var) {
  378. uint16_t target_line = (var.VP - VP_SD_FileName0) / VP_SD_FileName_LEN;
  379. if (target_line > DGUS_SD_FILESPERSCREEN) return;
  380. char tmpfilename[VP_SD_FileName_LEN + 1] = "";
  381. var.memadr = (void*)tmpfilename;
  382. if (filelist.seek(top_file + target_line))
  383. snprintf_P(tmpfilename, VP_SD_FileName_LEN, PSTR("%s%c"), filelist.filename(), filelist.isDir() ? '/' : 0);
  384. DGUSLCD_SendStringToDisplay(var);
  385. }
  386. void DGUSScreenVariableHandler::SDCardInserted() {
  387. top_file = 0;
  388. filelist.refresh();
  389. auto cs = ScreenHandler.getCurrentScreen();
  390. if (cs == DGUSLCD_SCREEN_MAIN || cs == DGUSLCD_SCREEN_STATUS)
  391. ScreenHandler.GotoScreen(DGUSLCD_SCREEN_SDFILELIST);
  392. }
  393. void DGUSScreenVariableHandler::SDCardRemoved() {
  394. if (current_screen == DGUSLCD_SCREEN_SDFILELIST
  395. || (current_screen == DGUSLCD_SCREEN_CONFIRM && (ConfirmVP == VP_SD_AbortPrintConfirmed || ConfirmVP == VP_SD_FileSelectConfirm))
  396. || current_screen == DGUSLCD_SCREEN_SDPRINTMANIPULATION
  397. ) ScreenHandler.GotoScreen(DGUSLCD_SCREEN_MAIN);
  398. }
  399. void DGUSScreenVariableHandler::SDCardError() {
  400. DGUSScreenVariableHandler::SDCardRemoved();
  401. ScreenHandler.sendinfoscreen(PSTR("NOTICE"), nullptr, PSTR("SD card error"), nullptr, true, true, true, true);
  402. ScreenHandler.SetupConfirmAction(nullptr);
  403. ScreenHandler.GotoScreen(DGUSLCD_SCREEN_POPUP);
  404. }
  405. #endif // SDSUPPORT
  406. void DGUSScreenVariableHandler::ScreenConfirmedOK(DGUS_VP_Variable &var, void *val_ptr) {
  407. DGUS_VP_Variable ramcopy;
  408. if (!populate_VPVar(ConfirmVP, &ramcopy)) return;
  409. if (ramcopy.set_by_display_handler) ramcopy.set_by_display_handler(ramcopy, val_ptr);
  410. }
  411. const uint16_t* DGUSLCD_FindScreenVPMapList(uint8_t screen) {
  412. const uint16_t *ret;
  413. const struct VPMapping *map = VPMap;
  414. while (ret = (uint16_t*) pgm_read_ptr(&(map->VPList))) {
  415. if (pgm_read_byte(&(map->screen)) == screen) return ret;
  416. map++;
  417. }
  418. return nullptr;
  419. }
  420. const DGUS_VP_Variable* DGUSLCD_FindVPVar(const uint16_t vp) {
  421. const DGUS_VP_Variable *ret = ListOfVP;
  422. do {
  423. const uint16_t vpcheck = pgm_read_word(&(ret->VP));
  424. if (vpcheck == 0) break;
  425. if (vpcheck == vp) return ret;
  426. ++ret;
  427. } while (1);
  428. DEBUG_ECHOLNPAIR("FindVPVar NOT FOUND ", vp);
  429. return nullptr;
  430. }
  431. void DGUSScreenVariableHandler::ScreenChangeHookIfIdle(DGUS_VP_Variable &var, void *val_ptr) {
  432. if (!ExtUI::isPrinting()) {
  433. ScreenChangeHook(var, val_ptr);
  434. dgusdisplay.RequestScreen(current_screen);
  435. }
  436. }
  437. void DGUSScreenVariableHandler::ScreenChangeHook(DGUS_VP_Variable &var, void *val_ptr) {
  438. uint8_t *tmp = (uint8_t*)val_ptr;
  439. // The keycode in target is coded as <from-frame><to-frame>, so 0x0100A means
  440. // from screen 1 (main) to 10 (temperature). DGUSLCD_SCREEN_POPUP is special,
  441. // meaning "return to previous screen"
  442. DGUSLCD_Screens target = (DGUSLCD_Screens)tmp[1];
  443. if (target == DGUSLCD_SCREEN_POPUP) {
  444. // special handling for popup is to return to previous menu
  445. if (current_screen == DGUSLCD_SCREEN_POPUP && confirm_action_cb) confirm_action_cb();
  446. PopToOldScreen();
  447. return;
  448. }
  449. UpdateNewScreen(target);
  450. #ifdef DEBUG_DGUSLCD
  451. if (!DGUSLCD_FindScreenVPMapList(target)) DEBUG_ECHOLNPAIR("WARNING: No screen Mapping found for ", target);
  452. #endif
  453. }
  454. void DGUSScreenVariableHandler::HandleAllHeatersOff(DGUS_VP_Variable &var, void *val_ptr) {
  455. thermalManager.disable_all_heaters();
  456. ScreenHandler.ForceCompleteUpdate(); // hint to send all data.
  457. }
  458. void DGUSScreenVariableHandler::HandleTemperatureChanged(DGUS_VP_Variable &var, void *val_ptr) {
  459. uint16_t newvalue = swap16(*(uint16_t*)val_ptr);
  460. uint16_t acceptedvalue;
  461. switch (var.VP) {
  462. default: return;
  463. #if HOTENDS >= 1
  464. case VP_T_E0_Set:
  465. thermalManager.setTargetHotend(newvalue, 0);
  466. acceptedvalue = thermalManager.temp_hotend[0].target;
  467. break;
  468. #endif
  469. #if HOTENDS >= 2
  470. case VP_T_E1_Set:
  471. thermalManager.setTargetHotend(newvalue, 1);
  472. acceptedvalue = thermalManager.temp_hotend[1].target;
  473. break;
  474. #endif
  475. #if HAS_HEATED_BED
  476. case VP_T_Bed_Set:
  477. thermalManager.setTargetBed(newvalue);
  478. acceptedvalue = thermalManager.temp_bed.target;
  479. break;
  480. #endif
  481. }
  482. // reply to display the new value to update the view if the new value was rejected by the Thermal Manager.
  483. if (newvalue != acceptedvalue && var.send_to_display_handler) var.send_to_display_handler(var);
  484. ScreenHandler.skipVP = var.VP; // don't overwrite value the next update time as the display might autoincrement in parallel
  485. }
  486. void DGUSScreenVariableHandler::HandleFlowRateChanged(DGUS_VP_Variable &var, void *val_ptr) {
  487. #if EXTRUDERS
  488. uint16_t newvalue = swap16(*(uint16_t*)val_ptr);
  489. uint8_t target_extruder;
  490. switch (var.VP) {
  491. default: return;
  492. #if HOTENDS >= 1
  493. case VP_Flowrate_E0: target_extruder = 0; break;
  494. #endif
  495. #if HOTENDS >= 2
  496. case VP_Flowrate_E1: target_extruder = 1; break;
  497. #endif
  498. }
  499. planner.set_flow(target_extruder, newvalue);
  500. ScreenHandler.skipVP = var.VP; // don't overwrite value the next update time as the display might autoincrement in parallel
  501. #else
  502. UNUSED(var); UNUSED(val_ptr);
  503. #endif
  504. }
  505. void DGUSScreenVariableHandler::HandleManualExtrude(DGUS_VP_Variable &var, void *val_ptr) {
  506. DEBUG_ECHOLNPGM("HandleManualExtrude");
  507. int16_t movevalue = swap16(*(uint16_t*)val_ptr);
  508. float target = movevalue * 0.01f;
  509. ExtUI::extruder_t target_extruder;
  510. switch (var.VP) {
  511. #if HOTENDS >= 1
  512. case VP_MOVE_E0: target_extruder = ExtUI::extruder_t::E0; break;
  513. #endif
  514. #if HOTENDS >= 2
  515. case VP_MOVE_E1: target_extruder = ExtUI::extruder_t::E1; break
  516. #endif
  517. default: return;
  518. }
  519. target += ExtUI::getAxisPosition_mm(target_extruder);
  520. ExtUI::setAxisPosition_mm(target, target_extruder);
  521. skipVP = var.VP;
  522. }
  523. #if ENABLED(DGUS_UI_MOVE_DIS_OPTION)
  524. void DGUSScreenVariableHandler::HandleManualMoveOption(DGUS_VP_Variable &var, void *val_ptr) {
  525. DEBUG_ECHOLNPGM("HandleManualMoveOption");
  526. *(uint16_t*)var.memadr = swap16(*(uint16_t*)val_ptr);
  527. }
  528. #endif
  529. void DGUSScreenVariableHandler::HandleManualMove(DGUS_VP_Variable &var, void *val_ptr) {
  530. DEBUG_ECHOLNPGM("HandleManualMove");
  531. int16_t movevalue = swap16(*(uint16_t*)val_ptr);
  532. #if ENABLED(DGUS_UI_MOVE_DIS_OPTION)
  533. if (movevalue) {
  534. const uint16_t choice = *(uint16_t*)var.memadr;
  535. movevalue = movevalue < 0 ? -choice : choice;
  536. }
  537. #endif
  538. char axiscode;
  539. unsigned int speed = 1500; //FIXME: get default feedrate for manual moves, dont hardcode.
  540. switch (var.VP) {
  541. default: return;
  542. case VP_MOVE_X:
  543. axiscode = 'X';
  544. if (!ExtUI::canMove(ExtUI::axis_t::X)) goto cannotmove;
  545. break;
  546. case VP_MOVE_Y:
  547. axiscode = 'Y';
  548. if (!ExtUI::canMove(ExtUI::axis_t::Y)) goto cannotmove;
  549. break;
  550. case VP_MOVE_Z:
  551. axiscode = 'Z';
  552. speed = 300; // default to 5mm/s
  553. if (!ExtUI::canMove(ExtUI::axis_t::Z)) goto cannotmove;
  554. break;
  555. case VP_HOME_ALL: // only used for homing
  556. axiscode = '\0';
  557. movevalue = 0; // ignore value sent from display, this VP is _ONLY_ for homing.
  558. break;
  559. }
  560. if (!movevalue) {
  561. // homing
  562. DEBUG_ECHOPAIR(" homing ", axiscode);
  563. char buf[6] = "G28 X";
  564. buf[4] = axiscode;
  565. //DEBUG_ECHOPAIR(" ", buf);
  566. queue.enqueue_one_now(buf);
  567. //DEBUG_ECHOLNPGM(" ✓");
  568. ScreenHandler.ForceCompleteUpdate();
  569. return;
  570. }
  571. else {
  572. //movement
  573. DEBUG_ECHOPAIR(" move ", axiscode);
  574. bool old_relative_mode = relative_mode;
  575. if (!relative_mode) {
  576. //DEBUG_ECHOPGM(" G91");
  577. queue.enqueue_now_P(PSTR("G91"));
  578. //DEBUG_ECHOPGM(" ✓ ");
  579. }
  580. char buf[32]; // G1 X9999.99 F12345
  581. unsigned int backup_speed = MMS_TO_MMM(feedrate_mm_s);
  582. char sign[]="\0";
  583. int16_t value = movevalue / 100;
  584. if (movevalue < 0) { value = -value; sign[0] = '-'; }
  585. int16_t fraction = ABS(movevalue) % 100;
  586. snprintf_P(buf, 32, PSTR("G0 %c%s%d.%02d F%d"), axiscode, sign, value, fraction, speed);
  587. //DEBUG_ECHOPAIR(" ", buf);
  588. queue.enqueue_one_now(buf);
  589. //DEBUG_ECHOLNPGM(" ✓ ");
  590. if (backup_speed != speed) {
  591. snprintf_P(buf, 32, PSTR("G0 F%d"), backup_speed);
  592. queue.enqueue_one_now(buf);
  593. //DEBUG_ECHOPAIR(" ", buf);
  594. }
  595. //while (!enqueue_and_echo_command(buf)) idle();
  596. //DEBUG_ECHOLNPGM(" ✓ ");
  597. if (!old_relative_mode) {
  598. //DEBUG_ECHOPGM("G90");
  599. queue.enqueue_now_P(PSTR("G90"));
  600. //DEBUG_ECHOPGM(" ✓ ");
  601. }
  602. }
  603. ScreenHandler.ForceCompleteUpdate();
  604. DEBUG_ECHOLNPGM("manmv done.");
  605. return;
  606. cannotmove:
  607. DEBUG_ECHOLNPAIR(" cannot move ", axiscode);
  608. return;
  609. }
  610. void DGUSScreenVariableHandler::HandleMotorLockUnlock(DGUS_VP_Variable &var, void *val_ptr) {
  611. DEBUG_ECHOLNPGM("HandleMotorLockUnlock");
  612. char buf[4];
  613. const int16_t lock = swap16(*(uint16_t*)val_ptr);
  614. strcpy_P(buf, lock ? PSTR("M18") : PSTR("M17"));
  615. //DEBUG_ECHOPAIR(" ", buf);
  616. queue.enqueue_one_now(buf);
  617. }
  618. #if ENABLED(POWER_LOSS_RECOVERY)
  619. void DGUSScreenVariableHandler::HandlePowerLossRecovery(DGUS_VP_Variable &var, void *val_ptr) {
  620. uint16_t value = swap16(*(uint16_t*)val_ptr);
  621. if (value) {
  622. queue.inject_P(PSTR("M1000"));
  623. ScreenHandler.GotoScreen(DGUSLCD_SCREEN_SDPRINTMANIPULATION);
  624. }
  625. else {
  626. recovery.cancel();
  627. ScreenHandler.GotoScreen(DGUSLCD_SCREEN_STATUS);
  628. }
  629. }
  630. #endif
  631. void DGUSScreenVariableHandler::HandleSettings(DGUS_VP_Variable &var, void *val_ptr) {
  632. DEBUG_ECHOLNPGM("HandleSettings");
  633. uint16_t value = swap16(*(uint16_t*)val_ptr);
  634. switch (value) {
  635. default: break;
  636. case 1:
  637. TERN_(PRINTCOUNTER, print_job_timer.initStats());
  638. queue.enqueue_now_P(PSTR("M502\nM500"));
  639. break;
  640. case 2: queue.enqueue_now_P(PSTR("M501")); break;
  641. case 3: queue.enqueue_now_P(PSTR("M500")); break;
  642. }
  643. }
  644. void DGUSScreenVariableHandler::HandleStepPerMMChanged(DGUS_VP_Variable &var, void *val_ptr) {
  645. DEBUG_ECHOLNPGM("HandleStepPerMMChanged");
  646. uint16_t value_raw = swap16(*(uint16_t*)val_ptr);
  647. DEBUG_ECHOLNPAIR("value_raw:", value_raw);
  648. float value = (float)value_raw/10;
  649. ExtUI::axis_t axis;
  650. switch (var.VP) {
  651. case VP_X_STEP_PER_MM: axis = ExtUI::axis_t::X; break;
  652. case VP_Y_STEP_PER_MM: axis = ExtUI::axis_t::Y; break;
  653. case VP_Z_STEP_PER_MM: axis = ExtUI::axis_t::Z; break;
  654. default: return;
  655. }
  656. DEBUG_ECHOLNPAIR_F("value:", value);
  657. ExtUI::setAxisSteps_per_mm(value, axis);
  658. DEBUG_ECHOLNPAIR_F("value_set:", ExtUI::getAxisSteps_per_mm(axis));
  659. ScreenHandler.skipVP = var.VP; // don't overwrite value the next update time as the display might autoincrement in parallel
  660. return;
  661. }
  662. void DGUSScreenVariableHandler::HandleStepPerMMExtruderChanged(DGUS_VP_Variable &var, void *val_ptr) {
  663. DEBUG_ECHOLNPGM("HandleStepPerMMExtruderChanged");
  664. uint16_t value_raw = swap16(*(uint16_t*)val_ptr);
  665. DEBUG_ECHOLNPAIR("value_raw:", value_raw);
  666. float value = (float)value_raw/10;
  667. ExtUI::extruder_t extruder;
  668. switch (var.VP) {
  669. default: return;
  670. #if HOTENDS >= 1
  671. case VP_E0_STEP_PER_MM: extruder = ExtUI::extruder_t::E0; break;
  672. #endif
  673. #if HOTENDS >= 2
  674. case VP_E1_STEP_PER_MM: extruder = ExtUI::extruder_t::E1; break;
  675. #endif
  676. }
  677. DEBUG_ECHOLNPAIR_F("value:", value);
  678. ExtUI::setAxisSteps_per_mm(value,extruder);
  679. DEBUG_ECHOLNPAIR_F("value_set:", ExtUI::getAxisSteps_per_mm(extruder));
  680. ScreenHandler.skipVP = var.VP; // don't overwrite value the next update time as the display might autoincrement in parallel
  681. return;
  682. }
  683. #if HAS_PID_HEATING
  684. void DGUSScreenVariableHandler::HandleTemperaturePIDChanged(DGUS_VP_Variable &var, void *val_ptr) {
  685. uint16_t rawvalue = swap16(*(uint16_t*)val_ptr);
  686. DEBUG_ECHOLNPAIR("V1:", rawvalue);
  687. float value = (float)rawvalue / 10;
  688. DEBUG_ECHOLNPAIR("V2:", value);
  689. float newvalue = 0;
  690. switch (var.VP) {
  691. default: return;
  692. #if HOTENDS >= 1
  693. case VP_E0_PID_P: newvalue = value; break;
  694. case VP_E0_PID_I: newvalue = scalePID_i(value); break;
  695. case VP_E0_PID_D: newvalue = scalePID_d(value); break;
  696. #endif
  697. #if HOTENDS >= 2
  698. case VP_E1_PID_P: newvalue = value; break;
  699. case VP_E1_PID_I: newvalue = scalePID_i(value); break;
  700. case VP_E1_PID_D: newvalue = scalePID_d(value); break;
  701. #endif
  702. #if HAS_HEATED_BED
  703. case VP_BED_PID_P: newvalue = value; break;
  704. case VP_BED_PID_I: newvalue = scalePID_i(value); break;
  705. case VP_BED_PID_D: newvalue = scalePID_d(value); break;
  706. #endif
  707. }
  708. DEBUG_ECHOLNPAIR_F("V3:", newvalue);
  709. *(float *)var.memadr = newvalue;
  710. ScreenHandler.skipVP = var.VP; // don't overwrite value the next update time as the display might autoincrement in parallel
  711. }
  712. void DGUSScreenVariableHandler::HandlePIDAutotune(DGUS_VP_Variable &var, void *val_ptr) {
  713. DEBUG_ECHOLNPGM("HandlePIDAutotune");
  714. char buf[32] = {0};
  715. switch (var.VP) {
  716. default: break;
  717. #if ENABLED(PIDTEMP)
  718. #if HOTENDS >= 1
  719. case VP_PID_AUTOTUNE_E0: // Autotune Extruder 0
  720. sprintf(buf, "M303 E%d C5 S210 U1", ExtUI::extruder_t::E0);
  721. break;
  722. #endif
  723. #if HOTENDS >= 2
  724. case VP_PID_AUTOTUNE_E1:
  725. sprintf(buf, "M303 E%d C5 S210 U1", ExtUI::extruder_t::E1);
  726. break;
  727. #endif
  728. #endif
  729. #if ENABLED(PIDTEMPBED)
  730. case VP_PID_AUTOTUNE_BED:
  731. sprintf(buf, "M303 E-1 C5 S70 U1");
  732. break;
  733. #endif
  734. }
  735. if (buf[0]) queue.enqueue_one_now(buf);
  736. #if ENABLED(DGUS_UI_WAITING)
  737. sendinfoscreen(PSTR("PID is autotuning"), PSTR("please wait"), NUL_STR, NUL_STR, true, true, true, true);
  738. GotoScreen(DGUSLCD_SCREEN_WAITING);
  739. #endif
  740. }
  741. #endif
  742. void DGUSScreenVariableHandler::HandleProbeOffsetZChanged(DGUS_VP_Variable &var, void *val_ptr) {
  743. DEBUG_ECHOLNPGM("HandleProbeOffsetZChanged");
  744. const float offset = float(int16_t(swap16(*(uint16_t*)val_ptr))) / 100.0f;
  745. ExtUI::setZOffset_mm(offset);
  746. ScreenHandler.skipVP = var.VP; // don't overwrite value the next update time as the display might autoincrement in parallel
  747. return;
  748. }
  749. #if ENABLED(BABYSTEPPING)
  750. void DGUSScreenVariableHandler::HandleLiveAdjustZ(DGUS_VP_Variable &var, void *val_ptr) {
  751. DEBUG_ECHOLNPGM("HandleLiveAdjustZ");
  752. int16_t flag = swap16(*(uint16_t*)val_ptr);
  753. int16_t steps = flag ? -20 : 20;
  754. ExtUI::smartAdjustAxis_steps(steps, ExtUI::axis_t::Z, true);
  755. ScreenHandler.ForceCompleteUpdate();
  756. return;
  757. }
  758. #endif
  759. #if HAS_FAN
  760. void DGUSScreenVariableHandler::HandleFanControl(DGUS_VP_Variable &var, void *val_ptr) {
  761. DEBUG_ECHOLNPGM("HandleFanControl");
  762. *(uint8_t*)var.memadr = *(uint8_t*)var.memadr > 0 ? 0 : 255;
  763. }
  764. #endif
  765. void DGUSScreenVariableHandler::HandleHeaterControl(DGUS_VP_Variable &var, void *val_ptr) {
  766. DEBUG_ECHOLNPGM("HandleHeaterControl");
  767. uint8_t preheat_temp = 0;
  768. switch (var.VP) {
  769. #if HOTENDS >= 1
  770. case VP_E0_CONTROL:
  771. #endif
  772. #if HOTENDS >= 2
  773. case VP_E1_CONTROL:
  774. #endif
  775. #if HOTENDS >= 3
  776. case VP_E2_CONTROL:
  777. #endif
  778. preheat_temp = PREHEAT_1_TEMP_HOTEND;
  779. break;
  780. case VP_BED_CONTROL:
  781. preheat_temp = PREHEAT_1_TEMP_BED;
  782. break;
  783. }
  784. *(int16_t*)var.memadr = *(int16_t*)var.memadr > 0 ? 0 : preheat_temp;
  785. }
  786. #if ENABLED(DGUS_PREHEAT_UI)
  787. void DGUSScreenVariableHandler::HandlePreheat(DGUS_VP_Variable &var, void *val_ptr) {
  788. DEBUG_ECHOLNPGM("HandlePreheat");
  789. uint8_t e_temp = 0;
  790. uint8_t bed_temp = 0;
  791. const uint16_t preheat_option = swap16(*(uint16_t*)val_ptr);
  792. switch (preheat_option) {
  793. case 0: // Preheat PLA
  794. #if defined(PREHEAT_1_TEMP_HOTEND) && defined(PREHEAT_1_TEMP_BED)
  795. e_temp = PREHEAT_1_TEMP_HOTEND;
  796. bed_temp = PREHEAT_1_TEMP_BED;
  797. #endif
  798. break;
  799. case 1: // Preheat ABS
  800. #if defined(PREHEAT_2_TEMP_HOTEND) && defined(PREHEAT_2_TEMP_BED)
  801. e_temp = PREHEAT_2_TEMP_HOTEND;
  802. bed_temp = PREHEAT_2_TEMP_BED;
  803. #endif
  804. break;
  805. case 2: // Preheat PET
  806. #if defined(PREHEAT_3_TEMP_HOTEND) && defined(PREHEAT_3_TEMP_BED)
  807. e_temp = PREHEAT_3_TEMP_HOTEND;
  808. bed_temp = PREHEAT_3_TEMP_BED;
  809. #endif
  810. break;
  811. case 3: // Preheat FLEX
  812. #if defined(PREHEAT_4_TEMP_HOTEND) && defined(PREHEAT_4_TEMP_BED)
  813. e_temp = PREHEAT_4_TEMP_HOTEND;
  814. bed_temp = PREHEAT_4_TEMP_BED;
  815. #endif
  816. break;
  817. case 7: // Custom preheat
  818. break;
  819. case 9: // Cool down
  820. e_temp = 0;
  821. bed_temp = 0;
  822. break;
  823. default:
  824. #if defined(PREHEAT_1_TEMP_HOTEND) && defined(PREHEAT_1_TEMP_BED)
  825. e_temp = PREHEAT_1_TEMP_HOTEND;
  826. bed_temp = PREHEAT_1_TEMP_BED;
  827. #endif
  828. break;
  829. }
  830. switch (var.VP) {
  831. default: return;
  832. #if HOTENDS >= 1
  833. case VP_E0_BED_PREHEAT:
  834. thermalManager.setTargetHotend(e_temp, 0);
  835. TERN_(HAS_HEATED_BED, thermalManager.setTargetBed(bed_temp));
  836. break;
  837. #endif
  838. #if HOTENDS >= 2
  839. case VP_E1_BED_PREHEAT:
  840. thermalManager.setTargetHotend(e_temp, 1);
  841. TERN_(HAS_HEATED_BED, thermalManager.setTargetBed(bed_temp));
  842. break;
  843. #endif
  844. }
  845. // Go to the preheat screen to show the heating progress
  846. GotoScreen(DGUSLCD_SCREEN_PREHEAT);
  847. }
  848. #endif
  849. #if ENABLED(DGUS_FILAMENT_LOADUNLOAD)
  850. void DGUSScreenVariableHandler::HandleFilamentOption(DGUS_VP_Variable &var, void *val_ptr) {
  851. DEBUG_ECHOLNPGM("HandleFilamentOption");
  852. uint8_t e_temp = 0;
  853. filament_data.heated = false;
  854. uint16_t preheat_option = swap16(*(uint16_t*)val_ptr);
  855. if (preheat_option <= 8) // Load filament type
  856. filament_data.action = 1;
  857. else if (preheat_option >= 10) { // Unload filament type
  858. preheat_option -= 10;
  859. filament_data.action = 2;
  860. filament_data.purge_length = DGUS_FILAMENT_PURGE_LENGTH;
  861. }
  862. else // Cancel filament operation
  863. filament_data.action = 0;
  864. switch (preheat_option) {
  865. case 0: // Load PLA
  866. #ifdef PREHEAT_1_TEMP_HOTEND
  867. e_temp = PREHEAT_1_TEMP_HOTEND;
  868. #endif
  869. break;
  870. case 1: // Load ABS
  871. TERN_(PREHEAT_2_TEMP_HOTEND, e_temp = PREHEAT_2_TEMP_HOTEND);
  872. break;
  873. case 2: // Load PET
  874. #ifdef PREHEAT_3_TEMP_HOTEND
  875. e_temp = PREHEAT_3_TEMP_HOTEND;
  876. #endif
  877. break;
  878. case 3: // Load FLEX
  879. #ifdef PREHEAT_4_TEMP_HOTEND
  880. e_temp = PREHEAT_4_TEMP_HOTEND;
  881. #endif
  882. break;
  883. case 9: // Cool down
  884. default:
  885. e_temp = 0;
  886. break;
  887. }
  888. if (filament_data.action == 0) { // Go back to utility screen
  889. #if HOTENDS >= 1
  890. thermalManager.setTargetHotend(e_temp, ExtUI::extruder_t::E0);
  891. #endif
  892. #if HOTENDS >= 2
  893. thermalManager.setTargetHotend(e_temp, ExtUI::extruder_t::E1);
  894. #endif
  895. GotoScreen(DGUSLCD_SCREEN_UTILITY);
  896. }
  897. else { // Go to the preheat screen to show the heating progress
  898. switch (var.VP) {
  899. default: return;
  900. #if HOTENDS >= 1
  901. case VP_E0_FILAMENT_LOAD_UNLOAD:
  902. filament_data.extruder = ExtUI::extruder_t::E0;
  903. thermalManager.setTargetHotend(e_temp, filament_data.extruder);
  904. break;
  905. #endif
  906. #if HOTENDS >= 2
  907. case VP_E1_FILAMENT_LOAD_UNLOAD:
  908. filament_data.extruder = ExtUI::extruder_t::E1;
  909. thermalManager.setTargetHotend(e_temp, filament_data.extruder);
  910. break;
  911. #endif
  912. }
  913. GotoScreen(DGUSLCD_SCREEN_FILAMENT_HEATING);
  914. }
  915. }
  916. void DGUSScreenVariableHandler::HandleFilamentLoadUnload(DGUS_VP_Variable &var) {
  917. DEBUG_ECHOLNPGM("HandleFilamentLoadUnload");
  918. if (filament_data.action <= 0) return;
  919. // If we close to the target temperature, we can start load or unload the filament
  920. if (thermalManager.hotEnoughToExtrude(filament_data.extruder) && \
  921. thermalManager.targetHotEnoughToExtrude(filament_data.extruder)) {
  922. float movevalue = DGUS_FILAMENT_LOAD_LENGTH_PER_TIME;
  923. if (filament_data.action == 1) { // load filament
  924. if (!filament_data.heated) {
  925. GotoScreen(DGUSLCD_SCREEN_FILAMENT_LOADING);
  926. filament_data.heated = true;
  927. }
  928. movevalue = ExtUI::getAxisPosition_mm(filament_data.extruder)+movevalue;
  929. }
  930. else { // unload filament
  931. if (!filament_data.heated) {
  932. GotoScreen(DGUSLCD_SCREEN_FILAMENT_UNLOADING);
  933. filament_data.heated = true;
  934. }
  935. // Before unloading extrude to prevent jamming
  936. if (filament_data.purge_length >= 0) {
  937. movevalue = ExtUI::getAxisPosition_mm(filament_data.extruder) + movevalue;
  938. filament_data.purge_length -= movevalue;
  939. }
  940. else
  941. movevalue = ExtUI::getAxisPosition_mm(filament_data.extruder) - movevalue;
  942. }
  943. ExtUI::setAxisPosition_mm(movevalue, filament_data.extruder);
  944. }
  945. }
  946. #endif
  947. void DGUSScreenVariableHandler::UpdateNewScreen(DGUSLCD_Screens newscreen, bool popup) {
  948. DEBUG_ECHOLNPAIR("SetNewScreen: ", newscreen);
  949. if (!popup) {
  950. memmove(&past_screens[1], &past_screens[0], sizeof(past_screens) - 1);
  951. past_screens[0] = current_screen;
  952. }
  953. current_screen = newscreen;
  954. skipVP = 0;
  955. ForceCompleteUpdate();
  956. }
  957. void DGUSScreenVariableHandler::PopToOldScreen() {
  958. DEBUG_ECHOLNPAIR("PopToOldScreen s=", past_screens[0]);
  959. GotoScreen(past_screens[0], true);
  960. memmove(&past_screens[0], &past_screens[1], sizeof(past_screens) - 1);
  961. past_screens[sizeof(past_screens) - 1] = DGUSLCD_SCREEN_MAIN;
  962. }
  963. void DGUSScreenVariableHandler::UpdateScreenVPData() {
  964. DEBUG_ECHOPAIR(" UpdateScreenVPData Screen: ", current_screen);
  965. const uint16_t *VPList = DGUSLCD_FindScreenVPMapList(current_screen);
  966. if (!VPList) {
  967. DEBUG_ECHOLNPAIR(" NO SCREEN FOR: ", current_screen);
  968. ScreenComplete = true;
  969. return; // nothing to do, likely a bug or boring screen.
  970. }
  971. // Round-robin updating of all VPs.
  972. VPList += update_ptr;
  973. bool sent_one = false;
  974. do {
  975. uint16_t VP = pgm_read_word(VPList);
  976. DEBUG_ECHOPAIR(" VP: ", VP);
  977. if (!VP) {
  978. update_ptr = 0;
  979. DEBUG_ECHOLNPGM(" UpdateScreenVPData done");
  980. ScreenComplete = true;
  981. return; // Screen completed.
  982. }
  983. if (VP == skipVP) { skipVP = 0; continue; }
  984. DGUS_VP_Variable rcpy;
  985. if (populate_VPVar(VP, &rcpy)) {
  986. uint8_t expected_tx = 6 + rcpy.size; // expected overhead is 6 bytes + payload.
  987. // Send the VP to the display, but try to avoid overrunning the Tx Buffer.
  988. // But send at least one VP, to avoid getting stalled.
  989. if (rcpy.send_to_display_handler && (!sent_one || expected_tx <= dgusdisplay.GetFreeTxBuffer())) {
  990. //DEBUG_ECHOPAIR(" calling handler for ", rcpy.VP);
  991. sent_one = true;
  992. rcpy.send_to_display_handler(rcpy);
  993. }
  994. else {
  995. //auto x=dgusdisplay.GetFreeTxBuffer();
  996. //DEBUG_ECHOLNPAIR(" tx almost full: ", x);
  997. //DEBUG_ECHOPAIR(" update_ptr ", update_ptr);
  998. ScreenComplete = false;
  999. return; // please call again!
  1000. }
  1001. }
  1002. } while (++update_ptr, ++VPList, true);
  1003. }
  1004. void DGUSDisplay::loop() {
  1005. // protect against recursion… ProcessRx() may indirectly call idle() when injecting gcode commands.
  1006. if (!no_reentrance) {
  1007. no_reentrance = true;
  1008. ProcessRx();
  1009. no_reentrance = false;
  1010. }
  1011. }
  1012. void DGUSDisplay::InitDisplay() {
  1013. dgusserial.begin(DGUS_BAUDRATE);
  1014. if (true
  1015. #if ENABLED(POWER_LOSS_RECOVERY)
  1016. && !recovery.valid()
  1017. #endif
  1018. )
  1019. RequestScreen(
  1020. #if ENABLED(SHOW_BOOTSCREEN)
  1021. DGUSLCD_SCREEN_BOOT
  1022. #else
  1023. DGUSLCD_SCREEN_MAIN
  1024. #endif
  1025. );
  1026. }
  1027. void DGUSDisplay::WriteVariable(uint16_t adr, const void* values, uint8_t valueslen, bool isstr) {
  1028. const char* myvalues = static_cast<const char*>(values);
  1029. bool strend = !myvalues;
  1030. WriteHeader(adr, DGUS_CMD_WRITEVAR, valueslen);
  1031. while (valueslen--) {
  1032. char x;
  1033. if (!strend) x = *myvalues++;
  1034. if ((isstr && !x) || strend) {
  1035. strend = true;
  1036. x = ' ';
  1037. }
  1038. dgusserial.write(x);
  1039. }
  1040. }
  1041. void DGUSDisplay::WriteVariablePGM(uint16_t adr, const void* values, uint8_t valueslen, bool isstr) {
  1042. const char* myvalues = static_cast<const char*>(values);
  1043. bool strend = !myvalues;
  1044. WriteHeader(adr, DGUS_CMD_WRITEVAR, valueslen);
  1045. while (valueslen--) {
  1046. char x;
  1047. if (!strend) x = pgm_read_byte(myvalues++);
  1048. if ((isstr && !x) || strend) {
  1049. strend = true;
  1050. x = ' ';
  1051. }
  1052. dgusserial.write(x);
  1053. }
  1054. }
  1055. void DGUSScreenVariableHandler::GotoScreen(DGUSLCD_Screens screen, bool ispopup) {
  1056. dgusdisplay.RequestScreen(screen);
  1057. UpdateNewScreen(screen, ispopup);
  1058. }
  1059. bool DGUSScreenVariableHandler::loop() {
  1060. dgusdisplay.loop();
  1061. const millis_t ms = millis();
  1062. static millis_t next_event_ms = 0;
  1063. if (!IsScreenComplete() || ELAPSED(ms, next_event_ms)) {
  1064. next_event_ms = ms + DGUS_UPDATE_INTERVAL_MS;
  1065. UpdateScreenVPData();
  1066. }
  1067. #if ENABLED(SHOW_BOOTSCREEN)
  1068. static bool booted = false;
  1069. if (!booted && TERN0(POWER_LOSS_RECOVERY, recovery.valid()))
  1070. booted = true;
  1071. if (!booted && ELAPSED(ms, BOOTSCREEN_TIMEOUT)) {
  1072. booted = true;
  1073. GotoScreen(DGUSLCD_SCREEN_MAIN);
  1074. }
  1075. #endif
  1076. return IsScreenComplete();
  1077. }
  1078. void DGUSDisplay::RequestScreen(DGUSLCD_Screens screen) {
  1079. DEBUG_ECHOLNPAIR("GotoScreen ", screen);
  1080. const unsigned char gotoscreen[] = { 0x5A, 0x01, (unsigned char) (screen >> 8U), (unsigned char) (screen & 0xFFU) };
  1081. WriteVariable(0x84, gotoscreen, sizeof(gotoscreen));
  1082. }
  1083. void DGUSDisplay::ProcessRx() {
  1084. #if ENABLED(DGUS_SERIAL_STATS_RX_BUFFER_OVERRUNS)
  1085. if (!dgusserial.available() && dgusserial.buffer_overruns()) {
  1086. // Overrun, but reset the flag only when the buffer is empty
  1087. // We want to extract as many as valid datagrams possible...
  1088. DEBUG_ECHOPGM("OVFL");
  1089. rx_datagram_state = DGUS_IDLE;
  1090. //dgusserial.reset_rx_overun();
  1091. dgusserial.flush();
  1092. }
  1093. #endif
  1094. uint8_t receivedbyte;
  1095. while (dgusserial.available()) {
  1096. switch (rx_datagram_state) {
  1097. case DGUS_IDLE: // Waiting for the first header byte
  1098. receivedbyte = dgusserial.read();
  1099. //DEBUG_ECHOPAIR("< ",x);
  1100. if (DGUS_HEADER1 == receivedbyte) rx_datagram_state = DGUS_HEADER1_SEEN;
  1101. break;
  1102. case DGUS_HEADER1_SEEN: // Waiting for the second header byte
  1103. receivedbyte = dgusserial.read();
  1104. //DEBUG_ECHOPAIR(" ",x);
  1105. rx_datagram_state = (DGUS_HEADER2 == receivedbyte) ? DGUS_HEADER2_SEEN : DGUS_IDLE;
  1106. break;
  1107. case DGUS_HEADER2_SEEN: // Waiting for the length byte
  1108. rx_datagram_len = dgusserial.read();
  1109. DEBUG_ECHOPAIR(" (", rx_datagram_len, ") ");
  1110. // Telegram min len is 3 (command and one word of payload)
  1111. rx_datagram_state = WITHIN(rx_datagram_len, 3, DGUS_RX_BUFFER_SIZE) ? DGUS_WAIT_TELEGRAM : DGUS_IDLE;
  1112. break;
  1113. case DGUS_WAIT_TELEGRAM: // wait for complete datagram to arrive.
  1114. if (dgusserial.available() < rx_datagram_len) return;
  1115. Initialized = true; // We've talked to it, so we defined it as initialized.
  1116. uint8_t command = dgusserial.read();
  1117. DEBUG_ECHOPAIR("# ", command);
  1118. uint8_t readlen = rx_datagram_len - 1; // command is part of len.
  1119. unsigned char tmp[rx_datagram_len - 1];
  1120. unsigned char *ptmp = tmp;
  1121. while (readlen--) {
  1122. receivedbyte = dgusserial.read();
  1123. DEBUG_ECHOPAIR(" ", receivedbyte);
  1124. *ptmp++ = receivedbyte;
  1125. }
  1126. DEBUG_ECHOPGM(" # ");
  1127. // mostly we'll get this: 5A A5 03 82 4F 4B -- ACK on 0x82, so discard it.
  1128. if (command == DGUS_CMD_WRITEVAR && 'O' == tmp[0] && 'K' == tmp[1]) {
  1129. DEBUG_ECHOLNPGM(">");
  1130. rx_datagram_state = DGUS_IDLE;
  1131. break;
  1132. }
  1133. /* AutoUpload, (and answer to) Command 0x83 :
  1134. | tmp[0 1 2 3 4 ... ]
  1135. | Example 5A A5 06 83 20 01 01 78 01 ……
  1136. | / / | | \ / | \ \
  1137. | Header | | | | \_____\_ DATA (Words!)
  1138. | DatagramLen / VPAdr |
  1139. | Command DataLen (in Words) */
  1140. if (command == DGUS_CMD_READVAR) {
  1141. const uint16_t vp = tmp[0] << 8 | tmp[1];
  1142. const uint8_t dlen = tmp[2] << 1; // Convert to Bytes. (Display works with words)
  1143. //DEBUG_ECHOPAIR(" vp=", vp, " dlen=", dlen);
  1144. DGUS_VP_Variable ramcopy;
  1145. if (populate_VPVar(vp, &ramcopy)) {
  1146. if (ramcopy.set_by_display_handler)
  1147. ramcopy.set_by_display_handler(ramcopy, &tmp[3]);
  1148. else
  1149. DEBUG_ECHOLNPGM(" VPVar found, no handler.");
  1150. }
  1151. else
  1152. DEBUG_ECHOLNPAIR(" VPVar not found:", vp);
  1153. rx_datagram_state = DGUS_IDLE;
  1154. break;
  1155. }
  1156. // discard anything else
  1157. rx_datagram_state = DGUS_IDLE;
  1158. }
  1159. }
  1160. }
  1161. size_t DGUSDisplay::GetFreeTxBuffer() { return DGUS_SERIAL_GET_TX_BUFFER_FREE(); }
  1162. void DGUSDisplay::WriteHeader(uint16_t adr, uint8_t cmd, uint8_t payloadlen) {
  1163. dgusserial.write(DGUS_HEADER1);
  1164. dgusserial.write(DGUS_HEADER2);
  1165. dgusserial.write(payloadlen + 3);
  1166. dgusserial.write(cmd);
  1167. dgusserial.write(adr >> 8);
  1168. dgusserial.write(adr & 0xFF);
  1169. }
  1170. void DGUSDisplay::WritePGM(const char str[], uint8_t len) {
  1171. while (len--) dgusserial.write(pgm_read_byte(str++));
  1172. }
  1173. // A SW memory barrier, to ensure GCC does not overoptimize loops
  1174. #define sw_barrier() asm volatile("": : :"memory");
  1175. #endif // HAS_DGUS_LCD