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.

dwin.cpp 128KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529
  1. /**
  2. * DWIN UI Enhanced implementation
  3. * Author: Miguel A. Risco-Castillo
  4. * Version: 3.6.1
  5. * Date: 2021/08/29
  6. *
  7. * This program is free software: you can redistribute it and/or modify
  8. * it under the terms of the GNU Lesser General Public License as
  9. * published by the Free Software Foundation, either version 3 of the License, or
  10. * (at your option) any later version.
  11. *
  12. * This program is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU Lesser General Public License
  18. * along with this program. If not, see <https://www.gnu.org/licenses/>.
  19. *
  20. */
  21. #include "../../../inc/MarlinConfigPre.h"
  22. #if ENABLED(DWIN_CREALITY_LCD_ENHANCED)
  23. #include "dwin.h"
  24. #include "../../fontutils.h"
  25. #include "../../marlinui.h"
  26. #include "../../../sd/cardreader.h"
  27. #include "../../../MarlinCore.h"
  28. #include "../../../core/serial.h"
  29. #include "../../../core/macros.h"
  30. #include "../../../gcode/queue.h"
  31. #include "../../../module/temperature.h"
  32. #include "../../../module/printcounter.h"
  33. #include "../../../module/motion.h"
  34. #include "../../../module/planner.h"
  35. #if HAS_FILAMENT_SENSOR
  36. #include "../../../feature/runout.h"
  37. #endif
  38. #if ENABLED(EEPROM_SETTINGS)
  39. #include "../../../module/settings.h"
  40. #endif
  41. #if ENABLED(HOST_ACTION_COMMANDS)
  42. #include "../../../feature/host_actions.h"
  43. #endif
  44. #if HAS_ONESTEP_LEVELING
  45. #include "../../../feature/bedlevel/bedlevel.h"
  46. #endif
  47. #if HAS_BED_PROBE
  48. #include "../../../module/probe.h"
  49. #endif
  50. #if EITHER(BABYSTEP_ZPROBE_OFFSET, JUST_BABYSTEP)
  51. #include "../../../feature/babystep.h"
  52. #endif
  53. #if ENABLED(POWER_LOSS_RECOVERY)
  54. #include "../../../feature/powerloss.h"
  55. #endif
  56. #include <WString.h>
  57. #include <stdio.h>
  58. #include <string.h>
  59. #ifndef MACHINE_SIZE
  60. #define MACHINE_SIZE STRINGIFY(X_BED_SIZE) "x" STRINGIFY(Y_BED_SIZE) "x" STRINGIFY(Z_MAX_POS)
  61. #endif
  62. #include "lockscreen.h"
  63. #ifndef CORP_WEBSITE
  64. #define CORP_WEBSITE WEBSITE_URL
  65. #endif
  66. #define PAUSE_HEAT
  67. #define USE_STRING_HEADINGS
  68. #define USE_STRING_TITLES
  69. #define MENU_CHAR_LIMIT 24
  70. // Print speed limit
  71. #define MIN_PRINT_SPEED 10
  72. #define MAX_PRINT_SPEED 999
  73. // Print flow limit
  74. #define MIN_PRINT_FLOW 10
  75. #define MAX_PRINT_FLOW 299
  76. // Load and Unload limits
  77. #define MAX_LOAD_UNLOAD 500
  78. // Feedspeed limit (max feedspeed = DEFAULT_MAX_FEEDRATE * 2)
  79. #define MIN_MAXFEEDSPEED 1
  80. #define MIN_MAXACCELERATION 1
  81. #define MIN_MAXJERK 0.1
  82. #define MIN_STEP 1
  83. #define MAX_STEP 999.9
  84. // Extruder's temperature limits
  85. #define MIN_ETEMP HEATER_0_MINTEMP
  86. #define MAX_ETEMP (HEATER_0_MAXTEMP - HOTEND_OVERSHOOT)
  87. #define FEEDRATE_E (60)
  88. // Minimum unit (0.1) : multiple (10)
  89. #define UNITFDIGITS 1
  90. #define MINUNITMULT POW(10, UNITFDIGITS)
  91. #define ENCODER_WAIT_MS 20
  92. #define DWIN_VAR_UPDATE_INTERVAL 1024
  93. #define DWIN_SCROLL_UPDATE_INTERVAL SEC_TO_MS(2)
  94. #define DWIN_REMAIN_TIME_UPDATE_INTERVAL SEC_TO_MS(20)
  95. #define BABY_Z_VAR TERN(HAS_BED_PROBE, probe.offset.z, dwin_zoffset)
  96. // Structs
  97. HMI_value_t HMI_value;
  98. HMI_flag_t HMI_flag{0};
  99. HMI_data_t HMI_data;
  100. millis_t dwin_heat_time = 0;
  101. uint8_t checkkey = MainMenu;
  102. uint8_t last_checkkey = MainMenu;
  103. typedef struct {
  104. uint8_t now, last;
  105. void set(uint8_t v) { now = last = v; }
  106. void reset() { set(0); }
  107. bool changed() { bool c = (now != last); if (c) last = now; return c; }
  108. bool dec() { if (now) now--; return changed(); }
  109. bool inc(uint8_t v) { if (now < (v - 1)) now++; else now = (v - 1); return changed(); }
  110. } select_t;
  111. select_t select_page{0}, select_file{0}, select_print{0};
  112. uint8_t index_file = MROWS;
  113. bool dwin_abort_flag = false; // Flag to reset feedrate, return to Home
  114. constexpr float default_max_feedrate[] = DEFAULT_MAX_FEEDRATE;
  115. constexpr float default_max_acceleration[] = DEFAULT_MAX_ACCELERATION;
  116. #if HAS_CLASSIC_JERK
  117. constexpr float default_max_jerk[] = { DEFAULT_XJERK, DEFAULT_YJERK, DEFAULT_ZJERK, DEFAULT_EJERK };
  118. #endif
  119. static uint8_t _percent_done = 0;
  120. static uint32_t _remain_time = 0;
  121. // Additional Aux Host Support
  122. static bool sdprint = false;
  123. #if ENABLED(PAUSE_HEAT)
  124. #if HAS_HOTEND
  125. uint16_t resume_hotend_temp = 0;
  126. #endif
  127. #if HAS_HEATED_BED
  128. uint16_t resume_bed_temp = 0;
  129. #endif
  130. #if HAS_FAN
  131. uint16_t resume_fan = 0;
  132. #endif
  133. #endif
  134. #if HAS_ZOFFSET_ITEM
  135. float dwin_zoffset = 0, last_zoffset = 0;
  136. #endif
  137. #if HAS_HOTEND
  138. float last_E = 0;
  139. #endif
  140. // New menu system pointers
  141. MenuClass *PrepareMenu = nullptr;
  142. MenuClass *LevBedMenu = nullptr;
  143. MenuClass *MoveMenu = nullptr;
  144. MenuClass *ControlMenu = nullptr;
  145. MenuClass *AdvancedSettings = nullptr;
  146. TERN_(HAS_HOME_OFFSET, MenuClass *HomeOffMenu = nullptr);
  147. TERN_(HAS_BED_PROBE, MenuClass *ProbeSetMenu = nullptr);
  148. MenuClass *FilSetMenu = nullptr;
  149. MenuClass *SelectColorMenu = nullptr;
  150. MenuClass *GetColorMenu = nullptr;
  151. MenuClass *TuneMenu = nullptr;
  152. MenuClass *MotionMenu = nullptr;
  153. MenuClass *FilamentMenu = nullptr;
  154. TERN_(MESH_BED_LEVELING, MenuClass *ManualMesh = nullptr);
  155. TERN_(HAS_HOTEND, MenuClass *PreheatMenu = nullptr);
  156. MenuClass *TemperatureMenu = nullptr;
  157. MenuClass *MaxSpeedMenu = nullptr;
  158. MenuClass *MaxAccelMenu = nullptr;
  159. MenuClass *MaxJerkMenu = nullptr;
  160. MenuClass *StepsMenu = nullptr;
  161. MenuClass *HotendPIDMenu = nullptr;
  162. MenuClass *BedPIDMenu = nullptr;
  163. #if EITHER(HAS_BED_PROBE, BABYSTEPPING)
  164. MenuClass *ZOffsetWizMenu = nullptr;
  165. #endif
  166. // Updatable menuitems pointers
  167. MenuItemClass *HotendTargetItem = nullptr;
  168. MenuItemClass *BedTargetItem = nullptr;
  169. MenuItemClass *FanSpeedItem = nullptr;
  170. MenuItemClass *MMeshMoveZItem = nullptr;
  171. #define DWIN_LANGUAGE_EEPROM_ADDRESS 0x01 // Between 0x01 and 0x63 (EEPROM_OFFSET-1)
  172. // BL24CXX::check() uses 0x00
  173. inline bool HMI_IsChinese() { return HMI_flag.language == DWIN_CHINESE; }
  174. void HMI_SetLanguageCache() {
  175. DWIN_JPG_CacheTo1(HMI_IsChinese() ? Language_Chinese : Language_English);
  176. }
  177. void HMI_SetLanguage() {
  178. #if BOTH(EEPROM_SETTINGS, IIC_BL24CXX_EEPROM)
  179. BL24CXX::read(DWIN_LANGUAGE_EEPROM_ADDRESS, (uint8_t*)&HMI_flag.language, sizeof(HMI_flag.language));
  180. #endif
  181. HMI_SetLanguageCache();
  182. }
  183. void HMI_ToggleLanguage() {
  184. HMI_flag.language = HMI_IsChinese() ? DWIN_ENGLISH : DWIN_CHINESE;
  185. HMI_SetLanguageCache();
  186. #if BOTH(EEPROM_SETTINGS, IIC_BL24CXX_EEPROM)
  187. BL24CXX::write(DWIN_LANGUAGE_EEPROM_ADDRESS, (uint8_t*)&HMI_flag.language, sizeof(HMI_flag.language));
  188. #endif
  189. }
  190. typedef struct { uint16_t x, y[2], w, h; } text_info_t;
  191. void ICON_Button(const bool here, const int iconid, const frame_rect_t &ico, const text_info_t (&txt)[2]) {
  192. const bool cn = HMI_IsChinese();
  193. DWIN_ICON_Show(1, 0, 0, ICON, iconid + here, ico.x, ico.y);
  194. if (here) DWIN_Draw_Rectangle(0, HMI_data.Highlight_Color, ico.x, ico.y, ico.x + ico.w - 1, ico.y + ico.h - 1);
  195. DWIN_Frame_AreaCopy(1, txt[cn].x, txt[cn].y[here], txt[cn].x + txt[cn].w - 1, txt[cn].y[here] + txt[cn].h - 1, ico.x + (ico.w - txt[cn].w) / 2, (ico.y + ico.h - 28) - txt[cn].h/2);
  196. }
  197. //
  198. // Main Menu: "Print"
  199. //
  200. void ICON_Print() {
  201. constexpr frame_rect_t ico = { 17, 110, 110, 100 };
  202. constexpr text_info_t txt[2] = {
  203. { 1, { 417, 449 }, 30, 14 },
  204. { 1, { 405, 447 }, 27, 15 }
  205. };
  206. ICON_Button(select_page.now == 0, ICON_Print_0, ico, txt);
  207. }
  208. //
  209. // Main Menu: "Prepare"
  210. //
  211. void ICON_Prepare() {
  212. constexpr frame_rect_t ico = { 145, 110, 110, 100 };
  213. constexpr text_info_t txt[2] = {
  214. { 33, { 417, 449 }, 51, 14 },
  215. { 31, { 405, 447 }, 27, 15 }
  216. };
  217. ICON_Button(select_page.now == 1, ICON_Prepare_0, ico, txt);
  218. }
  219. //
  220. // Main Menu: "Control"
  221. //
  222. void ICON_Control() {
  223. constexpr frame_rect_t ico = { 17, 226, 110, 100 };
  224. constexpr text_info_t txt[2] = {
  225. { 85, { 417, 449 }, 46, 14 },
  226. { 61, { 405, 447 }, 27, 15 }
  227. };
  228. ICON_Button(select_page.now == 2, ICON_Control_0, ico, txt);
  229. }
  230. //
  231. // Main Menu: "Info"
  232. //
  233. void ICON_StartInfo() {
  234. constexpr frame_rect_t ico = { 145, 226, 110, 100 };
  235. constexpr text_info_t txt[2] = {
  236. { 133, { 417, 449 }, 23, 14 },
  237. { 91, { 405, 447 }, 27, 15 }
  238. };
  239. ICON_Button(select_page.now == 3, ICON_Info_0, ico, txt);
  240. }
  241. //
  242. // Main Menu: "Level"
  243. //
  244. void ICON_Leveling() {
  245. constexpr frame_rect_t ico = { 145, 226, 110, 100 };
  246. constexpr text_info_t txt[2] = {
  247. { 88, { 433, 464 }, 36, 14 },
  248. { 211, { 405, 447 }, 27, 15 }
  249. };
  250. ICON_Button(select_page.now == 3, ICON_Leveling_0, ico, txt);
  251. }
  252. //
  253. // Printing: "Tune"
  254. //
  255. void ICON_Tune() {
  256. constexpr frame_rect_t ico = { 8, 232, 80, 100 };
  257. constexpr text_info_t txt[2] = {
  258. { 0, { 433, 464 }, 32, 14 },
  259. { 121, { 405, 447 }, 27, 15 }
  260. };
  261. ICON_Button(select_print.now == 0, ICON_Setup_0, ico, txt);
  262. }
  263. //
  264. // Printing: "Pause"
  265. //
  266. void ICON_Pause() {
  267. constexpr frame_rect_t ico = { 96, 232, 80, 100 };
  268. constexpr text_info_t txt[2] = {
  269. { 157, { 417, 449 }, 39, 14 },
  270. { 181, { 405, 447 }, 27, 15 }
  271. };
  272. ICON_Button(select_print.now == 1, ICON_Pause_0, ico, txt);
  273. }
  274. //
  275. // Printing: "Resume"
  276. //
  277. void ICON_Resume() {
  278. constexpr frame_rect_t ico = { 96, 232, 80, 100 };
  279. constexpr text_info_t txt[2] = {
  280. { 33, { 433, 464 }, 53, 14 },
  281. { 1, { 405, 447 }, 27, 15 }
  282. };
  283. ICON_Button(select_print.now == 1, ICON_Continue_0, ico, txt);
  284. }
  285. //
  286. // Printing: "Stop"
  287. //
  288. void ICON_Stop() {
  289. constexpr frame_rect_t ico = { 184, 232, 80, 100 };
  290. constexpr text_info_t txt[2] = {
  291. { 196, { 417, 449 }, 29, 14 },
  292. { 151, { 405, 447 }, 27, 12 }
  293. };
  294. ICON_Button(select_print.now == 2, ICON_Stop_0, ico, txt);
  295. }
  296. void Draw_Menu_Cursor(const uint8_t line) {
  297. DWIN_Draw_Rectangle(1, HMI_data.Cursor_color, 0, MBASE(line) - 18, 14, MBASE(line + 1) - 20);
  298. }
  299. void Erase_Menu_Cursor(const uint8_t line) {
  300. DWIN_Draw_Rectangle(1, HMI_data.Background_Color, 0, MBASE(line) - 18, 14, MBASE(line + 1) - 20);
  301. }
  302. void Move_Highlight(const int16_t from, const uint16_t newline) {
  303. Erase_Menu_Cursor(newline - from);
  304. Draw_Menu_Cursor(newline);
  305. }
  306. void Add_Menu_Line() {
  307. Move_Highlight(1, MROWS);
  308. DWIN_Draw_Line(HMI_data.SplitLine_Color, 16, MBASE(MROWS + 1) - 20, 256, MBASE(MROWS + 1) - 19);
  309. }
  310. void Scroll_Menu(const uint8_t dir) {
  311. DWIN_Frame_AreaMove(1, dir, MLINE, HMI_data.Background_Color, 0, 31, DWIN_WIDTH, 349);
  312. switch (dir) {
  313. case DWIN_SCROLL_DOWN: Move_Highlight(-1, 0); break;
  314. case DWIN_SCROLL_UP: Add_Menu_Line(); break;
  315. }
  316. }
  317. inline uint16_t nr_sd_menu_items() {
  318. return card.get_num_Files() + !card.flag.workDirIsRoot;
  319. }
  320. void Erase_Menu_Text(const uint8_t line) {
  321. DWIN_Draw_Rectangle(1, HMI_data.Background_Color, LBLX, MBASE(line) - 14, 271, MBASE(line) + 28);
  322. }
  323. void Draw_Menu_Line(const uint8_t line, const uint8_t icon=0, const char * const label=nullptr, bool more=false) {
  324. if (label) DWINUI::Draw_String(LBLX, MBASE(line) - 1, (char*)label);
  325. if (icon) DWINUI::Draw_Icon(icon, 26, MBASE(line) - 3);
  326. if (more) DWINUI::Draw_Icon(ICON_More, 226, MBASE(line) - 3);
  327. DWIN_Draw_Line(HMI_data.SplitLine_Color, 16, MBASE(line) + 33, 256, MBASE(line) + 33);
  328. }
  329. void Draw_Chkb_Line(const uint8_t line, const bool checked) {
  330. DWINUI::Draw_Checkbox(HMI_data.Text_Color, HMI_data.Background_Color, VALX + 16, MBASE(line) - 1, checked);
  331. }
  332. void Draw_Menu_IntValue(uint16_t bcolor, const uint8_t line, uint8_t iNum, const uint16_t value=0) {
  333. DWINUI::Draw_Int(HMI_data.Text_Color, bcolor, iNum , VALX, MBASE(line) - 1, value);
  334. }
  335. // The "Back" label is always on the first line
  336. void Draw_Back_Label() {
  337. if (HMI_IsChinese())
  338. DWIN_Frame_AreaCopy(1, 129, 72, 156, 84, LBLX, MBASE(0));
  339. else
  340. DWIN_Frame_AreaCopy(1, 223, 179, 254, 189, LBLX, MBASE(0));
  341. }
  342. // Draw "Back" line at the top
  343. void Draw_Back_First(const bool is_sel=true) {
  344. Draw_Menu_Line(0, ICON_Back);
  345. Draw_Back_Label();
  346. if (is_sel) Draw_Menu_Cursor(0);
  347. }
  348. inline ENCODER_DiffState get_encoder_state() {
  349. static millis_t Encoder_ms = 0;
  350. const millis_t ms = millis();
  351. if (PENDING(ms, Encoder_ms)) return ENCODER_DIFF_NO;
  352. const ENCODER_DiffState state = Encoder_ReceiveAnalyze();
  353. if (state != ENCODER_DIFF_NO) Encoder_ms = ms + ENCODER_WAIT_MS;
  354. return state;
  355. }
  356. template<typename T>
  357. inline bool Apply_Encoder(const ENCODER_DiffState &encoder_diffState, T &valref) {
  358. if (encoder_diffState == ENCODER_DIFF_CW)
  359. valref += EncoderRate.encoderMoveValue;
  360. else if (encoder_diffState == ENCODER_DIFF_CCW)
  361. valref -= EncoderRate.encoderMoveValue;
  362. return encoder_diffState == ENCODER_DIFF_ENTER;
  363. }
  364. //
  365. // Draw Popup Windows
  366. //
  367. inline void Draw_Popup_Bkgd_60() {
  368. DWIN_Draw_Rectangle(1, HMI_data.PopupBg_color, 14, 60, 258, 330);
  369. DWIN_Draw_Rectangle(0, HMI_data.Highlight_Color, 14, 60, 258, 330);
  370. }
  371. inline void Draw_Popup_Bkgd_105() {
  372. DWIN_Draw_Rectangle(1, HMI_data.PopupBg_color, 14, 105, 258, 374);
  373. DWIN_Draw_Rectangle(0, HMI_data.Highlight_Color, 14, 105, 258, 374);
  374. }
  375. void Clear_Popup_Area() {
  376. DWIN_Draw_Rectangle(1, HMI_data.Background_Color, 0, 31, DWIN_WIDTH, DWIN_HEIGHT);
  377. }
  378. void DWIN_Draw_Popup(uint8_t icon=0, const char * const msg1=nullptr, const char * const msg2=nullptr, uint8_t button=0) {
  379. DWINUI::ClearMenuArea();
  380. Draw_Popup_Bkgd_60();
  381. if (icon) DWINUI::Draw_Icon(icon, 101, 105);
  382. if (msg1) DWINUI::Draw_CenteredString(HMI_data.PopupTxt_Color, 210, msg1);
  383. if (msg2) DWINUI::Draw_CenteredString(HMI_data.PopupTxt_Color, 240, msg2);
  384. if (button) DWINUI::Draw_Icon(button, 86, 280);
  385. }
  386. void DWIN_Popup_Confirm(uint8_t icon, const char * const msg1, const char * const msg2) {
  387. HMI_SaveProcessID(WaitResponse);
  388. DWIN_Draw_Popup(icon, msg1, msg2, ICON_Confirm_E); // Button Confirm
  389. DWIN_UpdateLCD();
  390. }
  391. void DWIN_Popup_Continue(uint8_t icon, const char * const msg1, const char * const msg2) {
  392. HMI_SaveProcessID(WaitResponse);
  393. DWIN_Draw_Popup(icon, msg1, msg2, ICON_Continue_E); // Button Continue
  394. DWIN_UpdateLCD();
  395. }
  396. #if HAS_HOTEND
  397. void Popup_Window_ETempTooLow() {
  398. if (HMI_IsChinese()) {
  399. HMI_SaveProcessID(WaitResponse);
  400. DWINUI::ClearMenuArea();
  401. Draw_Popup_Bkgd_60();
  402. DWINUI::Draw_Icon(ICON_TempTooLow, 102, 105);
  403. DWIN_Frame_AreaCopy(1, 103, 371, 136, 386, 69, 240);
  404. DWIN_Frame_AreaCopy(1, 170, 371, 270, 386, 102, 240);
  405. DWINUI::Draw_Icon(ICON_Confirm_C, 86, 280);
  406. DWIN_UpdateLCD();
  407. }
  408. else
  409. DWIN_Popup_Confirm(ICON_TempTooLow, "Nozzle is too cold", "Preheat the hotend");
  410. }
  411. #endif
  412. void Popup_Window_Resume() {
  413. Clear_Popup_Area();
  414. Draw_Popup_Bkgd_105();
  415. if (HMI_IsChinese()) {
  416. DWIN_Frame_AreaCopy(1, 160, 338, 235, 354, 98, 135);
  417. DWIN_Frame_AreaCopy(1, 103, 321, 271, 335, 52, 192);
  418. DWINUI::Draw_Icon(ICON_Cancel_C, 26, 307);
  419. DWINUI::Draw_Icon(ICON_Continue_C, 146, 307);
  420. }
  421. else {
  422. DWINUI::Draw_CenteredString(HMI_data.PopupTxt_Color, 115, F("Continue Print"));
  423. DWINUI::Draw_CenteredString(HMI_data.PopupTxt_Color, 192, F("It looks like the last"));
  424. DWINUI::Draw_CenteredString(HMI_data.PopupTxt_Color, 212, F("file was interrupted."));
  425. DWINUI::Draw_Icon(ICON_Cancel_E, 26, 307);
  426. DWINUI::Draw_Icon(ICON_Continue_E, 146, 307);
  427. }
  428. }
  429. void Draw_Select_Highlight(const bool sel) {
  430. HMI_flag.select_flag = sel;
  431. const uint16_t c1 = sel ? HMI_data.Highlight_Color : HMI_data.PopupBg_color,
  432. c2 = sel ? HMI_data.PopupBg_color : HMI_data.Highlight_Color;
  433. DWIN_Draw_Rectangle(0, c1, 25, 279, 126, 318);
  434. DWIN_Draw_Rectangle(0, c1, 24, 278, 127, 319);
  435. DWIN_Draw_Rectangle(0, c2, 145, 279, 246, 318);
  436. DWIN_Draw_Rectangle(0, c2, 144, 278, 247, 319);
  437. }
  438. void Popup_window_PauseOrStop() {
  439. if (HMI_IsChinese()) {
  440. DWINUI::ClearMenuArea();
  441. Draw_Popup_Bkgd_60();
  442. if (select_print.now == 1) DWIN_Frame_AreaCopy(1, 237, 338, 269, 356, 98, 150);
  443. else if (select_print.now == 2) DWIN_Frame_AreaCopy(1, 221, 320, 253, 336, 98, 150);
  444. DWIN_Frame_AreaCopy(1, 220, 304, 264, 319, 130, 150);
  445. DWINUI::Draw_Icon(ICON_Confirm_C, 26, 280);
  446. DWINUI::Draw_Icon(ICON_Cancel_C, 146, 280);
  447. }
  448. else {
  449. DWIN_Draw_Popup(ICON_BLTouch, "Please confirm",(select_print.now == 1) ? GET_TEXT(MSG_PAUSE_PRINT) : GET_TEXT(MSG_STOP_PRINT));
  450. DWINUI::Draw_Icon(ICON_Confirm_E, 26, 280);
  451. DWINUI::Draw_Icon(ICON_Cancel_E, 146, 280);
  452. }
  453. Draw_Select_Highlight(true);
  454. }
  455. #if HAS_HOTEND || HAS_HEATED_BED
  456. void DWIN_Popup_Temperature(const bool toohigh) {
  457. Clear_Popup_Area();
  458. Draw_Popup_Bkgd_105();
  459. if (toohigh) {
  460. DWINUI::Draw_Icon(ICON_TempTooHigh, 102, 165);
  461. if (HMI_IsChinese()) {
  462. DWIN_Frame_AreaCopy(1, 103, 371, 237, 386, 52, 285);
  463. DWIN_Frame_AreaCopy(1, 151, 389, 185, 402, 187, 285);
  464. DWIN_Frame_AreaCopy(1, 189, 389, 271, 402, 95, 310);
  465. }
  466. else {
  467. DWINUI::Draw_String(HMI_data.PopupTxt_Color, 36, 300, F("Nozzle or Bed temperature"));
  468. DWINUI::Draw_String(HMI_data.PopupTxt_Color, 92, 300, F("is too high"));
  469. }
  470. }
  471. else {
  472. DWINUI::Draw_Icon(ICON_TempTooLow, 102, 165);
  473. if (HMI_IsChinese()) {
  474. DWIN_Frame_AreaCopy(1, 103, 371, 270, 386, 52, 285);
  475. DWIN_Frame_AreaCopy(1, 189, 389, 271, 402, 95, 310);
  476. }
  477. else {
  478. DWINUI::Draw_String(HMI_data.PopupTxt_Color, 36, 300, F("Nozzle or Bed temperature"));
  479. DWINUI::Draw_String(HMI_data.PopupTxt_Color, 92, 300, F("is too low"));
  480. }
  481. }
  482. }
  483. #endif
  484. void Draw_Print_Labels() {
  485. if (HMI_IsChinese()) {
  486. Title.FrameCopy(30, 1, 42, 14); // "Printing"
  487. DWIN_Frame_AreaCopy(1, 0, 72, 63, 86, 41, 173); // Printing Time
  488. DWIN_Frame_AreaCopy(1, 65, 72, 128, 86, 176, 173); // Remain
  489. }
  490. else {
  491. #ifdef USE_STRING_TITLES
  492. Title.ShowCaption(GET_TEXT(MSG_PRINTING));
  493. DWINUI::Draw_String( 46, 173, F("Print Time"));
  494. DWINUI::Draw_String(181, 173, F("Remain"));
  495. #else
  496. const uint16_t y = 168;
  497. Title.FrameCopy(42, 0, 47, 14); // "Printing"
  498. DWIN_Frame_AreaCopy(1, 0, 44, 96, 58, 41, y); // Printing Time
  499. DWIN_Frame_AreaCopy(1, 98, 44, 152, 58, 176, y); // Remain
  500. #endif
  501. }
  502. }
  503. void Draw_Print_ProgressBar() {
  504. DWINUI::Draw_Icon(ICON_Bar, 15, 93);
  505. DWIN_Draw_Rectangle(1, HMI_data.Barfill_Color, 16 + _percent_done * 240 / 100, 93, 256, 113);
  506. DWINUI::Draw_Int(HMI_data.PercentTxt_Color, HMI_data.Background_Color, 3, 117, 133, _percent_done);
  507. DWINUI::Draw_String(HMI_data.PercentTxt_Color, 142, 133, F("%"));
  508. }
  509. void Draw_Print_ProgressElapsed() {
  510. char buf[10];
  511. duration_t elapsed = print_job_timer.duration(); // print timer
  512. sprintf_P(buf, PSTR("%02i:%02i"), (uint16_t)(elapsed.value / 3600), ((uint16_t)elapsed.value % 3600) / 60);
  513. DWINUI::Draw_String(HMI_data.Text_Color, HMI_data.Background_Color, 47, 192, buf);
  514. }
  515. void Draw_Print_ProgressRemain() {
  516. char buf[10];
  517. sprintf_P(buf, PSTR("%02i:%02i"), (uint16_t)(_remain_time / 3600), ((uint16_t)_remain_time % 3600) / 60);
  518. DWINUI::Draw_String(HMI_data.Text_Color, HMI_data.Background_Color, 181, 192, buf);
  519. }
  520. void ICON_ResumeOrPause() {
  521. if (printingIsPaused() || HMI_flag.pause_flag || HMI_flag.pause_action)
  522. ICON_Resume();
  523. else
  524. ICON_Pause();
  525. }
  526. void Draw_PrintProcess() {
  527. DWINUI::ClearMenuArea();
  528. Draw_Print_Labels();
  529. ICON_Tune();
  530. ICON_ResumeOrPause();
  531. ICON_Stop();
  532. DWIN_Print_Header(sdprint ? card.longest_filename() : nullptr);
  533. DWINUI::Draw_Icon(ICON_PrintTime, 15, 173);
  534. DWINUI::Draw_Icon(ICON_RemainTime, 150, 171);
  535. Draw_Print_ProgressBar();
  536. Draw_Print_ProgressElapsed();
  537. Draw_Print_ProgressRemain();
  538. DWIN_UpdateLCD();
  539. }
  540. void Goto_PrintProcess() {
  541. checkkey = PrintProcess;
  542. Draw_PrintProcess();
  543. }
  544. void Draw_PrintDone() {
  545. // show percent bar and value
  546. _percent_done = 100;
  547. _remain_time = 0;
  548. DWINUI::ClearMenuArea();
  549. DWIN_Print_Header(nullptr);
  550. Draw_Print_Labels();
  551. DWINUI::Draw_Icon(ICON_PrintTime, 15, 173);
  552. DWINUI::Draw_Icon(ICON_RemainTime, 150, 171);
  553. Draw_Print_ProgressBar();
  554. Draw_Print_ProgressElapsed();
  555. Draw_Print_ProgressRemain();
  556. // show print done confirm
  557. DWIN_Draw_Rectangle(1, HMI_data.Background_Color, 0, 240, DWIN_WIDTH - 1, STATUS_Y - 1);
  558. DWINUI::Draw_Icon(HMI_IsChinese() ? ICON_Confirm_C : ICON_Confirm_E, 86, 283);
  559. }
  560. void Draw_Main_Menu() {
  561. DWINUI::ClearMenuArea();
  562. if (HMI_IsChinese())
  563. Title.FrameCopy(2, 2, 26, 13); // "Home" etc
  564. else {
  565. #ifdef USE_STRING_HEADINGS
  566. Title.ShowCaption(MACHINE_NAME);
  567. #else
  568. Title.FrameCopy(0, 2, 40, 11); // "Home"
  569. #endif
  570. }
  571. DWINUI::Draw_Icon(ICON_LOGO, 71, 52); // CREALITY logo
  572. ICON_Print();
  573. ICON_Prepare();
  574. ICON_Control();
  575. TERN(HAS_ONESTEP_LEVELING, ICON_Leveling, ICON_StartInfo)();
  576. DWIN_UpdateLCD();
  577. }
  578. void Goto_Main_Menu() {
  579. checkkey = MainMenu;
  580. DWIN_StatusChanged(nullptr);
  581. Draw_Main_Menu();
  582. }
  583. // Draw X, Y, Z and blink if in an un-homed or un-trusted state
  584. void _update_axis_value(const AxisEnum axis, const uint16_t x, const uint16_t y, const bool blink, const bool force) {
  585. const bool draw_qmark = axis_should_home(axis),
  586. draw_empty = NONE(HOME_AFTER_DEACTIVATE, DISABLE_REDUCED_ACCURACY_WARNING) && !draw_qmark && !axis_is_trusted(axis);
  587. // Check for a position change
  588. static xyz_pos_t oldpos = { -1, -1, -1 };
  589. const float p = current_position[axis];
  590. const bool changed = oldpos[axis] != p;
  591. if (changed) oldpos[axis] = p;
  592. if (force || changed || draw_qmark || draw_empty) {
  593. if (blink && draw_qmark)
  594. DWINUI::Draw_String(HMI_data.Coordinate_Color, HMI_data.Background_Color, x, y, F("--?--"));
  595. else if (blink && draw_empty)
  596. DWINUI::Draw_String(HMI_data.Coordinate_Color, HMI_data.Background_Color, x, y, F(" "));
  597. else
  598. DWINUI::Draw_Signed_Float(HMI_data.Coordinate_Color, HMI_data.Background_Color, 3, 1, x, y, p * 10);
  599. }
  600. }
  601. void _draw_xyz_position(const bool force) {
  602. //SERIAL_ECHOPGM("Draw XYZ:");
  603. static bool _blink = false;
  604. const bool blink = !!(millis() & 0x400UL);
  605. if (force || blink != _blink) {
  606. _blink = blink;
  607. //SERIAL_ECHOPGM(" (blink)");
  608. _update_axis_value(X_AXIS, 35, 459, blink, true);
  609. _update_axis_value(Y_AXIS, 120, 459, blink, true);
  610. _update_axis_value(Z_AXIS, 205, 459, blink, true);
  611. }
  612. //SERIAL_EOL();
  613. }
  614. void update_variable() {
  615. #if HAS_HOTEND
  616. static celsius_t _hotendtemp = 0, _hotendtarget = 0;
  617. const celsius_t hc = thermalManager.wholeDegHotend(0),
  618. ht = thermalManager.degTargetHotend(0);
  619. const bool _new_hotend_temp = _hotendtemp != hc,
  620. _new_hotend_target = _hotendtarget != ht;
  621. if (_new_hotend_temp) _hotendtemp = hc;
  622. if (_new_hotend_target) _hotendtarget = ht;
  623. #endif
  624. #if HAS_HEATED_BED
  625. static celsius_t _bedtemp = 0, _bedtarget = 0;
  626. const celsius_t bc = thermalManager.wholeDegBed(),
  627. bt = thermalManager.degTargetBed();
  628. const bool _new_bed_temp = _bedtemp != bc,
  629. _new_bed_target = _bedtarget != bt;
  630. if (_new_bed_temp) _bedtemp = bc;
  631. if (_new_bed_target) _bedtarget = bt;
  632. #endif
  633. #if HAS_FAN
  634. static uint8_t _fanspeed = 0;
  635. const bool _new_fanspeed = _fanspeed != thermalManager.fan_speed[0];
  636. if (_new_fanspeed) _fanspeed = thermalManager.fan_speed[0];
  637. #endif
  638. if (checkkey == Menu && (CurrentMenu == TuneMenu || CurrentMenu == TemperatureMenu)) {
  639. // Tune page temperature update
  640. #if HAS_HOTEND
  641. if (_new_hotend_target)
  642. HotendTargetItem->Draw(CurrentMenu->line(HotendTargetItem->pos));
  643. #endif
  644. #if HAS_HEATED_BED
  645. if (_new_bed_target)
  646. BedTargetItem->Draw(CurrentMenu->line(BedTargetItem->pos));
  647. #endif
  648. #if HAS_FAN
  649. if (_new_fanspeed)
  650. FanSpeedItem->Draw(CurrentMenu->line(FanSpeedItem->pos));
  651. #endif
  652. }
  653. // Bottom temperature update
  654. #if HAS_HOTEND
  655. if (_new_hotend_temp)
  656. DWINUI::Draw_Int(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 3, 28, 384, _hotendtemp);
  657. if (_new_hotend_target)
  658. DWINUI::Draw_Int(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 3, 25 + 4 * STAT_CHR_W + 6, 384, _hotendtarget);
  659. static int16_t _flow = planner.flow_percentage[0];
  660. if (_flow != planner.flow_percentage[0]) {
  661. _flow = planner.flow_percentage[0];
  662. DWINUI::Draw_Int(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 3, 116 + 2 * STAT_CHR_W, 417, _flow);
  663. }
  664. #endif
  665. #if HAS_HEATED_BED
  666. if (_new_bed_temp)
  667. DWINUI::Draw_Int(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 3, 28, 417, _bedtemp);
  668. if (_new_bed_target)
  669. DWINUI::Draw_Int(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 3, 25 + 4 * STAT_CHR_W + 6, 417, _bedtarget);
  670. #endif
  671. static int16_t _feedrate = 100;
  672. if (_feedrate != feedrate_percentage) {
  673. _feedrate = feedrate_percentage;
  674. DWINUI::Draw_Int(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 3, 116 + 2 * STAT_CHR_W, 384, _feedrate);
  675. }
  676. #if HAS_FAN
  677. if (_new_fanspeed) {
  678. _fanspeed = thermalManager.fan_speed[0];
  679. DWINUI::Draw_Int(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 3, 195 + 2 * STAT_CHR_W, 384, _fanspeed);
  680. }
  681. #endif
  682. static float _offset = 0;
  683. if (BABY_Z_VAR != _offset) {
  684. _offset = BABY_Z_VAR;
  685. DWINUI::Draw_Signed_Float(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 2, 2, 210, 417, _offset * 100);
  686. }
  687. _draw_xyz_position(false);
  688. }
  689. /**
  690. * Read and cache the working directory.
  691. *
  692. * TODO: New code can follow the pattern of menu_media.cpp
  693. * and rely on Marlin caching for performance. No need to
  694. * cache files here.
  695. */
  696. #ifndef strcasecmp_P
  697. #define strcasecmp_P(a, b) strcasecmp((a), (b))
  698. #endif
  699. void make_name_without_ext(char *dst, char *src, size_t maxlen=MENU_CHAR_LIMIT) {
  700. char * const name = card.longest_filename();
  701. size_t pos = strlen(name); // index of ending nul
  702. // For files, remove the extension
  703. // which may be .gcode, .gco, or .g
  704. if (!card.flag.filenameIsDir)
  705. while (pos && src[pos] != '.') pos--; // find last '.' (stop at 0)
  706. size_t len = pos; // nul or '.'
  707. if (len > maxlen) { // Keep the name short
  708. pos = len = maxlen; // move nul down
  709. dst[--pos] = '.'; // insert dots
  710. dst[--pos] = '.';
  711. dst[--pos] = '.';
  712. }
  713. dst[len] = '\0'; // end it
  714. // Copy down to 0
  715. while (pos--) dst[pos] = src[pos];
  716. }
  717. void HMI_SDCardInit() { card.cdroot(); }
  718. void MarlinUI::refresh() { /* Nothing to see here */ }
  719. #define ICON_Folder ICON_More
  720. #if ENABLED(SCROLL_LONG_FILENAMES)
  721. char shift_name[LONG_FILENAME_LENGTH + 1];
  722. int8_t shift_amt; // = 0
  723. millis_t shift_ms; // = 0
  724. // Init the shift name based on the highlighted item
  725. void Init_Shift_Name() {
  726. const bool is_subdir = !card.flag.workDirIsRoot;
  727. const int8_t filenum = select_file.now - 1 - is_subdir; // Skip "Back" and ".."
  728. const uint16_t fileCnt = card.get_num_Files();
  729. if (WITHIN(filenum, 0, fileCnt - 1)) {
  730. card.getfilename_sorted(SD_ORDER(filenum, fileCnt));
  731. char * const name = card.longest_filename();
  732. make_name_without_ext(shift_name, name, 100);
  733. }
  734. }
  735. void Init_SDItem_Shift() {
  736. shift_amt = 0;
  737. shift_ms = select_file.now > 0 && strlen(shift_name) > MENU_CHAR_LIMIT ? millis() + 750UL : 0;
  738. }
  739. #endif
  740. /**
  741. * Display an SD item, adding a CDUP for subfolders.
  742. */
  743. void Draw_SDItem(const uint16_t item, int16_t row=-1) {
  744. if (row < 0) row = item + 1 + MROWS - index_file;
  745. const bool is_subdir = !card.flag.workDirIsRoot;
  746. if (is_subdir && item == 0)
  747. return Draw_Menu_Line(row, ICON_Folder, "..");
  748. card.getfilename_sorted(SD_ORDER(item - is_subdir, card.get_num_Files()));
  749. char * const name = card.longest_filename();
  750. #if ENABLED(SCROLL_LONG_FILENAMES)
  751. // Init the current selected name
  752. // This is used during scroll drawing
  753. if (item == select_file.now - 1) {
  754. make_name_without_ext(shift_name, name, 100);
  755. Init_SDItem_Shift();
  756. }
  757. #endif
  758. // Draw the file/folder with name aligned left
  759. char str[strlen(name) + 1];
  760. make_name_without_ext(str, name);
  761. Draw_Menu_Line(row, card.flag.filenameIsDir ? ICON_Folder : ICON_File, str);
  762. }
  763. #if ENABLED(SCROLL_LONG_FILENAMES)
  764. void Draw_SDItem_Shifted(uint8_t &shift) {
  765. // Limit to the number of chars past the cutoff
  766. const size_t len = strlen(shift_name);
  767. NOMORE(shift, _MAX(len - MENU_CHAR_LIMIT, 0U));
  768. // Shorten to the available space
  769. const size_t lastchar = _MIN((signed)len, shift + MENU_CHAR_LIMIT);
  770. const char c = shift_name[lastchar];
  771. shift_name[lastchar] = '\0';
  772. const uint8_t row = select_file.now + MROWS - index_file; // skip "Back" and scroll
  773. Erase_Menu_Text(row);
  774. Draw_Menu_Line(row, 0, &shift_name[shift]);
  775. shift_name[lastchar] = c;
  776. }
  777. #endif
  778. // Redraw the first set of SD Files
  779. void Redraw_SD_List() {
  780. select_file.reset();
  781. index_file = MROWS;
  782. DWINUI::ClearMenuArea(); // Leave title bar unchanged
  783. Draw_Back_First();
  784. if (card.isMounted()) {
  785. // As many files as will fit
  786. LOOP_L_N(i, _MIN(nr_sd_menu_items(), MROWS))
  787. Draw_SDItem(i, i + 1);
  788. TERN_(SCROLL_LONG_FILENAMES, Init_SDItem_Shift());
  789. }
  790. else {
  791. DWIN_Draw_Rectangle(1, HMI_data.AlertBg_Color, 10, MBASE(3) - 10, DWIN_WIDTH - 10, MBASE(4));
  792. DWINUI::Draw_CenteredString(font16x32, HMI_data.AlertTxt_Color, MBASE(3), F("No Media"));
  793. }
  794. }
  795. bool DWIN_lcd_sd_status = false;
  796. void SDCard_Up() {
  797. card.cdup();
  798. Redraw_SD_List();
  799. DWIN_lcd_sd_status = false; // On next DWIN_Update
  800. }
  801. void SDCard_Folder(char * const dirname) {
  802. card.cd(dirname);
  803. Redraw_SD_List();
  804. DWIN_lcd_sd_status = false; // On next DWIN_Update
  805. }
  806. //
  807. // Watch for media mount / unmount
  808. //
  809. void HMI_SDCardUpdate() {
  810. if (HMI_flag.home_flag) return;
  811. if (DWIN_lcd_sd_status != card.isMounted()) {
  812. DWIN_lcd_sd_status = card.isMounted();
  813. //SERIAL_ECHOLNPAIR("HMI_SDCardUpdate: ", DWIN_lcd_sd_status);
  814. if (DWIN_lcd_sd_status) {
  815. if (checkkey == SelectFile)
  816. Redraw_SD_List();
  817. }
  818. else {
  819. // clean file icon
  820. if (checkkey == SelectFile) {
  821. Redraw_SD_List();
  822. }
  823. else if (sdprint && card.isPrinting() && printingIsActive()) {
  824. // TODO: Move card removed abort handling
  825. // to CardReader::manage_media.
  826. card.abortFilePrintSoon();
  827. wait_for_heatup = wait_for_user = false;
  828. dwin_abort_flag = true; // Reset feedrate, return to Home
  829. }
  830. }
  831. DWIN_UpdateLCD();
  832. }
  833. }
  834. //
  835. // The status area is always on-screen, except during
  836. // full-screen modal dialogs. (TODO: Keep alive during dialogs)
  837. //
  838. void Draw_Status_Area(const bool with_update) {
  839. DWIN_Draw_Rectangle(1, HMI_data.Background_Color, 0, STATUS_Y + 21, DWIN_WIDTH, DWIN_HEIGHT - 1);
  840. #if HAS_HOTEND
  841. DWINUI::Draw_Icon(ICON_HotendTemp, 10, 383);
  842. DWINUI::Draw_Int(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 3, 28, 384, thermalManager.wholeDegHotend(0));
  843. DWINUI::Draw_String(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 25 + 3 * STAT_CHR_W + 5, 384, F("/"));
  844. DWINUI::Draw_Int(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 3, 25 + 4 * STAT_CHR_W + 6, 384, thermalManager.degTargetHotend(0));
  845. DWINUI::Draw_Icon(ICON_StepE, 112, 417);
  846. DWINUI::Draw_Int(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 3, 116 + 2 * STAT_CHR_W, 417, planner.flow_percentage[0]);
  847. DWINUI::Draw_String(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 116 + 5 * STAT_CHR_W + 2, 417, F("%"));
  848. #endif
  849. #if HAS_HEATED_BED
  850. DWINUI::Draw_Icon(ICON_BedTemp, 10, 416);
  851. DWINUI::Draw_Int(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 3, 28, 417, thermalManager.wholeDegBed());
  852. DWINUI::Draw_String(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 25 + 3 * STAT_CHR_W + 5, 417, F("/"));
  853. DWINUI::Draw_Int(true, true, 0, DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 3, 25 + 4 * STAT_CHR_W + 6, 417, thermalManager.degTargetBed());
  854. #endif
  855. DWINUI::Draw_Icon(ICON_Speed, 113, 383);
  856. DWINUI::Draw_Int(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 3, 116 + 2 * STAT_CHR_W, 384, feedrate_percentage);
  857. DWINUI::Draw_String(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 116 + 5 * STAT_CHR_W + 2, 384, F("%"));
  858. #if HAS_FAN
  859. DWINUI::Draw_Icon(ICON_FanSpeed, 187, 383);
  860. DWINUI::Draw_Int(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 3, 195 + 2 * STAT_CHR_W, 384, thermalManager.fan_speed[0]);
  861. #endif
  862. #if HAS_ZOFFSET_ITEM
  863. DWINUI::Draw_Icon(ICON_Zoffset, 187, 416);
  864. #endif
  865. if (BABY_Z_VAR < 0) {
  866. DWINUI::Draw_Float(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 2, 2, 207, 417, -BABY_Z_VAR * 100);
  867. DWINUI::Draw_String(HMI_data.Indicator_Color, 205, 419, F("-"));
  868. }
  869. else {
  870. DWINUI::Draw_Float(DWIN_FONT_STAT, HMI_data.Indicator_Color, HMI_data.Background_Color, 2, 2, 207, 417, BABY_Z_VAR * 100);
  871. DWINUI::Draw_String(HMI_data.Indicator_Color, 205, 419, F(" "));
  872. }
  873. DWIN_Draw_Rectangle(1, HMI_data.SplitLine_Color, 0, 449, DWIN_WIDTH, 451);
  874. DWINUI::Draw_Icon(ICON_MaxSpeedX, 10, 456);
  875. DWINUI::Draw_Icon(ICON_MaxSpeedY, 95, 456);
  876. DWINUI::Draw_Icon(ICON_MaxSpeedZ, 180, 456);
  877. _draw_xyz_position(true);
  878. if (with_update) {
  879. DWIN_UpdateLCD();
  880. delay(5);
  881. }
  882. }
  883. void HMI_StartFrame(const bool with_update) {
  884. Goto_Main_Menu();
  885. Draw_Status_Area(with_update);
  886. }
  887. void Draw_Info_Menu() {
  888. DWINUI::ClearMenuArea();
  889. Draw_Back_First();
  890. DWINUI::Draw_CenteredString(122, F(MACHINE_SIZE));
  891. DWINUI::Draw_CenteredString(195, F(SHORT_BUILD_VERSION));
  892. if (HMI_IsChinese()) {
  893. Title.FrameCopy(30, 17, 28, 13); // "Info"
  894. DWIN_Frame_AreaCopy(1, 197, 149, 252, 161, 108, 102); // "Size"
  895. DWIN_Frame_AreaCopy(1, 1, 164, 56, 176, 108, 175); // "Firmware Version"
  896. DWIN_Frame_AreaCopy(1, 58, 164, 113, 176, 105, 248); // "Contact Details"
  897. }
  898. else {
  899. #ifdef USE_STRING_HEADINGS
  900. Title.ShowCaption(GET_TEXT_F(MSG_INFO_SCREEN));
  901. #else
  902. Title.FrameCopy(192, 15, 23, 12); // "Info"
  903. #endif
  904. DWIN_Frame_AreaCopy(1, 120, 150, 146, 161, 124, 102); // "Size"
  905. DWIN_Frame_AreaCopy(1, 146, 151, 254, 161, 82, 175); // "Firmware Version"
  906. DWIN_Frame_AreaCopy(1, 1, 164, 96, 175, 89, 248); // "Contact details"
  907. }
  908. DWINUI::Draw_CenteredString(268, F(CORP_WEBSITE));
  909. LOOP_L_N(i, 3) {
  910. DWINUI::Draw_Icon(ICON_PrintSize + i, 26, 99 + i * 73);
  911. DWIN_Draw_Line(HMI_data.SplitLine_Color, 16, MBASE(2) + i * 73, 256, 156 + i * 73);
  912. }
  913. DWIN_UpdateLCD();
  914. }
  915. void Draw_Print_File_Menu() {
  916. if (HMI_IsChinese())
  917. Title.FrameCopy(0, 31, 56, 14); // "Print file"
  918. else {
  919. #ifdef USE_STRING_HEADINGS
  920. Title.ShowCaption(GET_TEXT_F(MSG_MEDIA_MENU));
  921. #else
  922. Title.FrameCopy(52, 31, 86, 11); // "Print file"
  923. #endif
  924. }
  925. Redraw_SD_List();
  926. }
  927. // Main Process
  928. void HMI_MainMenu() {
  929. ENCODER_DiffState encoder_diffState = get_encoder_state();
  930. if (encoder_diffState == ENCODER_DIFF_NO) return;
  931. if (encoder_diffState == ENCODER_DIFF_CW) {
  932. if (select_page.inc(4)) {
  933. switch (select_page.now) {
  934. case 0: ICON_Print(); break;
  935. case 1: ICON_Print(); ICON_Prepare(); break;
  936. case 2: ICON_Prepare(); ICON_Control(); break;
  937. case 3: ICON_Control(); TERN(HAS_ONESTEP_LEVELING, ICON_Leveling, ICON_StartInfo)(); break;
  938. }
  939. }
  940. }
  941. else if (encoder_diffState == ENCODER_DIFF_CCW) {
  942. if (select_page.dec()) {
  943. switch (select_page.now) {
  944. case 0: ICON_Print(); ICON_Prepare(); break;
  945. case 1: ICON_Prepare(); ICON_Control(); break;
  946. case 2: ICON_Control(); TERN(HAS_ONESTEP_LEVELING, ICON_Leveling, ICON_StartInfo)(); break;
  947. case 3: TERN(HAS_ONESTEP_LEVELING, ICON_Leveling, ICON_StartInfo)(); break;
  948. }
  949. }
  950. }
  951. else if (encoder_diffState == ENCODER_DIFF_ENTER) {
  952. switch (select_page.now) {
  953. case 0: // Print File
  954. checkkey = SelectFile;
  955. Draw_Print_File_Menu();
  956. break;
  957. case 1: // Prepare
  958. Draw_Prepare_Menu();
  959. break;
  960. case 2: // Control
  961. Draw_Control_Menu();
  962. break;
  963. case 3: // Leveling or Info
  964. #if HAS_ONESTEP_LEVELING
  965. queue.inject_P(PSTR("G28XYO\nG28Z\nG29"));
  966. #else
  967. checkkey = Info;
  968. Draw_Info_Menu();
  969. #endif
  970. break;
  971. }
  972. }
  973. DWIN_UpdateLCD();
  974. }
  975. // Select (and Print) File
  976. void HMI_SelectFile() {
  977. ENCODER_DiffState encoder_diffState = get_encoder_state();
  978. const uint16_t hasUpDir = !card.flag.workDirIsRoot;
  979. if (encoder_diffState == ENCODER_DIFF_NO) {
  980. #if ENABLED(SCROLL_LONG_FILENAMES)
  981. if (shift_ms && select_file.now >= 1 + hasUpDir) {
  982. // Scroll selected filename every second
  983. const millis_t ms = millis();
  984. if (ELAPSED(ms, shift_ms)) {
  985. const bool was_reset = shift_amt < 0;
  986. shift_ms = ms + 375UL + was_reset * 250UL; // ms per character
  987. uint8_t shift_new = shift_amt + 1; // Try to shift by...
  988. Draw_SDItem_Shifted(shift_new); // Draw the item
  989. if (!was_reset && shift_new == 0) // Was it limited to 0?
  990. shift_ms = 0; // No scrolling needed
  991. else if (shift_new == shift_amt) // Scroll reached the end
  992. shift_new = -1; // Reset
  993. shift_amt = shift_new; // Set new scroll
  994. }
  995. }
  996. #endif
  997. return;
  998. }
  999. // First pause is long. Easy.
  1000. // On reset, long pause must be after 0.
  1001. const uint16_t fullCnt = nr_sd_menu_items();
  1002. if (encoder_diffState == ENCODER_DIFF_CW && fullCnt) {
  1003. if (select_file.inc(1 + fullCnt)) {
  1004. const uint8_t itemnum = select_file.now - 1; // -1 for "Back"
  1005. if (TERN0(SCROLL_LONG_FILENAMES, shift_ms)) { // If line was shifted
  1006. Erase_Menu_Text(itemnum + MROWS - index_file); // Erase and
  1007. Draw_SDItem(itemnum - 1); // redraw
  1008. }
  1009. if (select_file.now > MROWS && select_file.now > index_file) { // Cursor past the bottom
  1010. index_file = select_file.now; // New bottom line
  1011. Scroll_Menu(DWIN_SCROLL_UP);
  1012. Draw_SDItem(itemnum, MROWS); // Draw and init the shift name
  1013. }
  1014. else {
  1015. Move_Highlight(1, select_file.now + MROWS - index_file); // Just move highlight
  1016. TERN_(SCROLL_LONG_FILENAMES, Init_Shift_Name()); // ...and init the shift name
  1017. }
  1018. TERN_(SCROLL_LONG_FILENAMES, Init_SDItem_Shift());
  1019. }
  1020. }
  1021. else if (encoder_diffState == ENCODER_DIFF_CCW && fullCnt) {
  1022. if (select_file.dec()) {
  1023. const uint8_t itemnum = select_file.now - 1; // -1 for "Back"
  1024. if (TERN0(SCROLL_LONG_FILENAMES, shift_ms)) { // If line was shifted
  1025. Erase_Menu_Text(select_file.now + 1 + MROWS - index_file); // Erase and
  1026. Draw_SDItem(itemnum + 1); // redraw
  1027. }
  1028. if (select_file.now < index_file - MROWS) { // Cursor past the top
  1029. index_file--; // New bottom line
  1030. Scroll_Menu(DWIN_SCROLL_DOWN);
  1031. if (index_file == MROWS) {
  1032. Draw_Back_First();
  1033. TERN_(SCROLL_LONG_FILENAMES, shift_ms = 0);
  1034. }
  1035. else {
  1036. Draw_SDItem(itemnum, 0); // Draw the item (and init shift name)
  1037. }
  1038. }
  1039. else {
  1040. Move_Highlight(-1, select_file.now + MROWS - index_file); // Just move highlight
  1041. TERN_(SCROLL_LONG_FILENAMES, Init_Shift_Name()); // ...and init the shift name
  1042. }
  1043. TERN_(SCROLL_LONG_FILENAMES, Init_SDItem_Shift()); // Reset left. Init timer.
  1044. }
  1045. }
  1046. else if (encoder_diffState == ENCODER_DIFF_ENTER) {
  1047. if (select_file.now == 0) { // Back
  1048. select_page.set(0);
  1049. Goto_Main_Menu();
  1050. }
  1051. else if (hasUpDir && select_file.now == 1) { // CD-Up
  1052. SDCard_Up();
  1053. goto HMI_SelectFileExit;
  1054. }
  1055. else {
  1056. const uint16_t filenum = select_file.now - 1 - hasUpDir;
  1057. card.getfilename_sorted(SD_ORDER(filenum, card.get_num_Files()));
  1058. // Enter that folder!
  1059. if (card.flag.filenameIsDir) {
  1060. SDCard_Folder(card.filename);
  1061. goto HMI_SelectFileExit;
  1062. }
  1063. // Reset highlight for next entry
  1064. select_print.reset();
  1065. select_file.reset();
  1066. // Start choice and print SD file
  1067. HMI_flag.heat_flag = true;
  1068. HMI_flag.print_finish = false;
  1069. card.openAndPrintFile(card.filename);
  1070. #if HAS_FAN
  1071. // All fans on for Ender 3 v2 ?
  1072. // The slicer should manage this for us.
  1073. //for (uint8_t i = 0; i < FAN_COUNT; i++)
  1074. // thermalManager.fan_speed[i] = 255;
  1075. #endif
  1076. DWIN_Print_Started(true);
  1077. }
  1078. }
  1079. HMI_SelectFileExit:
  1080. DWIN_UpdateLCD();
  1081. }
  1082. // Printing
  1083. void HMI_Printing() {
  1084. ENCODER_DiffState encoder_diffState = get_encoder_state();
  1085. if (encoder_diffState == ENCODER_DIFF_NO) return;
  1086. // Avoid flicker by updating only the previous menu
  1087. if (encoder_diffState == ENCODER_DIFF_CW) {
  1088. if (select_print.inc(3)) {
  1089. switch (select_print.now) {
  1090. case 0: ICON_Tune(); break;
  1091. case 1:
  1092. ICON_Tune();
  1093. ICON_ResumeOrPause();
  1094. break;
  1095. case 2:
  1096. ICON_ResumeOrPause();
  1097. ICON_Stop();
  1098. break;
  1099. }
  1100. }
  1101. }
  1102. else if (encoder_diffState == ENCODER_DIFF_CCW) {
  1103. if (select_print.dec()) {
  1104. switch (select_print.now) {
  1105. case 0:
  1106. ICON_Tune();
  1107. ICON_ResumeOrPause();
  1108. break;
  1109. case 1:
  1110. ICON_ResumeOrPause();
  1111. ICON_Stop();
  1112. break;
  1113. case 2: ICON_Stop(); break;
  1114. }
  1115. }
  1116. }
  1117. else if (encoder_diffState == ENCODER_DIFF_ENTER) {
  1118. switch (select_print.now) {
  1119. case 0: // Tune
  1120. Draw_Tune_Menu();
  1121. break;
  1122. case 1: // Pause
  1123. if (HMI_flag.pause_flag) {
  1124. ICON_Pause();
  1125. #ifndef ADVANCED_PAUSE_FEATURE
  1126. char cmd[40];
  1127. cmd[0] = '\0';
  1128. #if BOTH(HAS_HEATED_BED, PAUSE_HEAT)
  1129. if (resume_bed_temp) sprintf_P(cmd, PSTR("M190 S%i\n"), resume_bed_temp);
  1130. #endif
  1131. #if BOTH(HAS_HOTEND, PAUSE_HEAT)
  1132. if (resume_hotend_temp) sprintf_P(&cmd[strlen(cmd)], PSTR("M109 S%i\n"), resume_hotend_temp);
  1133. #endif
  1134. #if HAS_FAN
  1135. if (resume_fan) thermalManager.fan_speed[0] = resume_fan;
  1136. #endif
  1137. strcat_P(cmd, M24_STR);
  1138. queue.inject(cmd);
  1139. #endif
  1140. }
  1141. else {
  1142. HMI_flag.select_flag = true;
  1143. checkkey = PauseOrStop;
  1144. Popup_window_PauseOrStop();
  1145. }
  1146. break;
  1147. case 2: // Stop
  1148. HMI_flag.select_flag = true;
  1149. checkkey = PauseOrStop;
  1150. Popup_window_PauseOrStop();
  1151. break;
  1152. default: break;
  1153. }
  1154. }
  1155. DWIN_UpdateLCD();
  1156. }
  1157. // Print done
  1158. void HMI_PrintDone() {
  1159. ENCODER_DiffState encoder_diffState = get_encoder_state();
  1160. if (encoder_diffState == ENCODER_DIFF_NO) return;
  1161. if (encoder_diffState == ENCODER_DIFF_ENTER) {
  1162. dwin_abort_flag = true; // Reset feedrate, return to Home
  1163. Goto_Main_Menu(); // Return to Main menu after print done
  1164. }
  1165. }
  1166. // Pause or Stop popup
  1167. void HMI_PauseOrStop() {
  1168. ENCODER_DiffState encoder_diffState = get_encoder_state();
  1169. if (encoder_diffState == ENCODER_DIFF_NO) return;
  1170. if (encoder_diffState == ENCODER_DIFF_CW)
  1171. Draw_Select_Highlight(false);
  1172. else if (encoder_diffState == ENCODER_DIFF_CCW)
  1173. Draw_Select_Highlight(true);
  1174. else if (encoder_diffState == ENCODER_DIFF_ENTER) {
  1175. if (select_print.now == 1) { // pause window
  1176. if (HMI_flag.select_flag) {
  1177. HMI_flag.pause_action = true;
  1178. ICON_Resume();
  1179. queue.inject_P(PSTR("M25"));
  1180. }
  1181. else {
  1182. // cancel pause
  1183. }
  1184. Goto_PrintProcess();
  1185. }
  1186. else if (select_print.now == 2) { // stop window
  1187. if (HMI_flag.select_flag) {
  1188. checkkey = MainMenu;
  1189. if (HMI_flag.home_flag) planner.synchronize(); // Wait for planner moves to finish!
  1190. wait_for_heatup = wait_for_user = false; // Stop waiting for heating/user
  1191. card.abortFilePrintSoon(); // Let the main loop handle SD abort
  1192. dwin_abort_flag = true; // Reset feedrate, return to Home
  1193. #ifdef ACTION_ON_CANCEL
  1194. host_action_cancel();
  1195. #endif
  1196. DWIN_Draw_Popup(ICON_BLTouch, "Stopping..." , "Please wait until done.");
  1197. }
  1198. else
  1199. Goto_PrintProcess(); // cancel stop
  1200. }
  1201. }
  1202. DWIN_UpdateLCD();
  1203. }
  1204. #include "../../../libs/buzzer.h"
  1205. void HMI_AudioFeedback(const bool success/*=true*/) {
  1206. #if HAS_BUZZER
  1207. if (success) {
  1208. BUZZ(100, 659);
  1209. BUZZ(10, 0);
  1210. BUZZ(100, 698);
  1211. }
  1212. else
  1213. BUZZ(40, 440);
  1214. #endif
  1215. }
  1216. void Draw_Main_Area() {
  1217. switch (checkkey) {
  1218. case MainMenu: Draw_Main_Menu(); break;
  1219. case SelectFile: Draw_Print_File_Menu(); break;
  1220. case PrintProcess: Draw_PrintProcess(); break;
  1221. case PrintDone: Draw_PrintDone(); break;
  1222. case Info: Draw_Info_Menu(); break;
  1223. case PauseOrStop: Popup_window_PauseOrStop(); break;
  1224. #if ENABLED(ADVANCED_PAUSE_FEATURE)
  1225. case FilamentPurge: Draw_Popup_FilamentPurge(); break;
  1226. #endif
  1227. case Locked: LockScreen.Draw(); break;
  1228. case Menu:
  1229. case SetInt:
  1230. case SetPInt:
  1231. case SetIntNoDraw:
  1232. case SetFloat:
  1233. case SetPFloat: CurrentMenu->Draw(); break;
  1234. default: break;
  1235. }
  1236. }
  1237. void HMI_ReturnScreen() {
  1238. checkkey = last_checkkey;
  1239. Draw_Main_Area();
  1240. DWIN_UpdateLCD();
  1241. return;
  1242. }
  1243. void HMI_Popup() {
  1244. ENCODER_DiffState encoder_diffState = get_encoder_state();
  1245. if (encoder_diffState == ENCODER_DIFF_NO) return;
  1246. if (encoder_diffState == ENCODER_DIFF_ENTER) {
  1247. wait_for_user = false;
  1248. HMI_ReturnScreen();
  1249. }
  1250. }
  1251. void HMI_Init() {
  1252. HMI_SDCardInit();
  1253. for (uint16_t t = 0; t <= 100; t += 2) {
  1254. DWINUI::Draw_Icon(ICON_Bar, 15, 260);
  1255. DWIN_Draw_Rectangle(1, HMI_data.Background_Color, 15 + t * 242 / 100, 260, 257, 280);
  1256. DWIN_UpdateLCD();
  1257. delay(20);
  1258. }
  1259. HMI_SetLanguage();
  1260. }
  1261. void DWIN_Update() {
  1262. EachMomentUpdate(); // Status update
  1263. HMI_SDCardUpdate(); // SD card update
  1264. DWIN_HandleScreen(); // Rotary encoder update
  1265. }
  1266. void EachMomentUpdate() {
  1267. static millis_t next_var_update_ms = 0, next_rts_update_ms = 0;
  1268. const millis_t ms = millis();
  1269. if (ELAPSED(ms, next_var_update_ms)) {
  1270. next_var_update_ms = ms + DWIN_VAR_UPDATE_INTERVAL;
  1271. update_variable();
  1272. }
  1273. if (PENDING(ms, next_rts_update_ms)) return;
  1274. next_rts_update_ms = ms + DWIN_SCROLL_UPDATE_INTERVAL;
  1275. if (checkkey == PrintProcess) {
  1276. // if print done
  1277. if (HMI_flag.print_finish) {
  1278. HMI_flag.print_finish = false;
  1279. TERN_(POWER_LOSS_RECOVERY, recovery.cancel());
  1280. planner.finish_and_disable();
  1281. checkkey = PrintDone;
  1282. Draw_PrintDone();
  1283. }
  1284. else if (HMI_flag.pause_flag != printingIsPaused()) {
  1285. // print status update
  1286. HMI_flag.pause_flag = printingIsPaused();
  1287. ICON_ResumeOrPause();
  1288. }
  1289. }
  1290. // pause after homing
  1291. if (HMI_flag.pause_action && printingIsPaused() && !planner.has_blocks_queued()) {
  1292. HMI_flag.pause_action = false;
  1293. #if ENABLED(PAUSE_HEAT)
  1294. if (sdprint) {
  1295. TERN_(HAS_HOTEND, resume_hotend_temp = thermalManager.degTargetHotend(0));
  1296. TERN_(HAS_HEATED_BED, resume_bed_temp = thermalManager.degTargetBed());
  1297. }
  1298. else {
  1299. TERN_(HAS_HOTEND, resume_hotend_temp = thermalManager.wholeDegHotend(0));
  1300. TERN_(HAS_HEATED_BED, resume_bed_temp = thermalManager.wholeDegBed());
  1301. }
  1302. TERN_(HAS_FAN, resume_fan = thermalManager.fan_speed[0]);
  1303. #endif
  1304. #if DISABLED(ADVANCED_PAUSE_FEATURE)
  1305. thermalManager.disable_all_heaters();
  1306. #endif
  1307. #if DISABLED(PARK_HEAD_ON_PAUSE)
  1308. queue.inject_P(PSTR("G1 F1200 X0 Y0"));
  1309. #endif
  1310. }
  1311. if (checkkey == PrintProcess) { // print process
  1312. duration_t elapsed = print_job_timer.duration(); // print timer
  1313. if (sdprint && card.isPrinting()) {
  1314. uint8_t percentDone = card.percentDone();
  1315. static uint8_t last_percentValue = 101;
  1316. if (last_percentValue != percentDone) { // print percent
  1317. last_percentValue = percentDone;
  1318. if (percentDone) {
  1319. _percent_done = percentDone;
  1320. Draw_Print_ProgressBar();
  1321. }
  1322. }
  1323. // Estimate remaining time every 20 seconds
  1324. static millis_t next_remain_time_update = 0;
  1325. if (_percent_done > 1 && ELAPSED(ms, next_remain_time_update) && !HMI_flag.heat_flag) {
  1326. _remain_time = (elapsed.value - dwin_heat_time) / (_percent_done * 0.01f) - (elapsed.value - dwin_heat_time);
  1327. next_remain_time_update += DWIN_REMAIN_TIME_UPDATE_INTERVAL;
  1328. Draw_Print_ProgressRemain();
  1329. }
  1330. }
  1331. // Print time so far
  1332. static uint16_t last_Printtime = 0;
  1333. const uint16_t min = (elapsed.value % 3600) / 60;
  1334. if (last_Printtime != min) { // 1 minute update
  1335. last_Printtime = min;
  1336. Draw_Print_ProgressElapsed();
  1337. }
  1338. }
  1339. else if (dwin_abort_flag && !HMI_flag.home_flag) { // Print Stop
  1340. dwin_abort_flag = false;
  1341. dwin_zoffset = BABY_Z_VAR;
  1342. select_page.set(0);
  1343. Goto_Main_Menu();
  1344. }
  1345. #if ENABLED(POWER_LOSS_RECOVERY)
  1346. else if (DWIN_lcd_sd_status && recovery.dwin_flag) { // resume print before power off
  1347. static bool recovery_flag = false;
  1348. recovery.dwin_flag = false;
  1349. recovery_flag = true;
  1350. auto update_selection = [&](const bool sel) {
  1351. HMI_flag.select_flag = sel;
  1352. const uint16_t c1 = sel ? HMI_data.PopupBg_color : HMI_data.Highlight_Color;
  1353. DWIN_Draw_Rectangle(0, c1, 25, 306, 126, 345);
  1354. DWIN_Draw_Rectangle(0, c1, 24, 305, 127, 346);
  1355. const uint16_t c2 = sel ? HMI_data.Highlight_Color : HMI_data.PopupBg_color;
  1356. DWIN_Draw_Rectangle(0, c2, 145, 306, 246, 345);
  1357. DWIN_Draw_Rectangle(0, c2, 144, 305, 247, 346);
  1358. };
  1359. Popup_Window_Resume();
  1360. update_selection(true);
  1361. // TODO: Get the name of the current file from someplace
  1362. //
  1363. //(void)recovery.interrupted_file_exists();
  1364. SdFile *dir = nullptr;
  1365. const char * const filename = card.diveToFile(true, dir, recovery.info.sd_filename);
  1366. card.selectFileByName(filename);
  1367. DWINUI::Draw_CenteredString(HMI_data.PopupTxt_Color, 252, card.longest_filename());
  1368. DWIN_UpdateLCD();
  1369. while (recovery_flag) {
  1370. ENCODER_DiffState encoder_diffState = Encoder_ReceiveAnalyze();
  1371. if (encoder_diffState != ENCODER_DIFF_NO) {
  1372. if (encoder_diffState == ENCODER_DIFF_ENTER) {
  1373. recovery_flag = false;
  1374. if (HMI_flag.select_flag) break;
  1375. TERN_(POWER_LOSS_RECOVERY, queue.inject_P(PSTR("M1000C")));
  1376. return HMI_StartFrame(true);
  1377. }
  1378. else
  1379. update_selection(encoder_diffState == ENCODER_DIFF_CW);
  1380. DWIN_UpdateLCD();
  1381. }
  1382. watchdog_refresh();
  1383. }
  1384. select_print.set(0);
  1385. queue.inject_P(PSTR("M1000"));
  1386. sdprint = true;
  1387. Goto_PrintProcess();
  1388. Draw_Status_Area(true);
  1389. }
  1390. #endif // POWER_LOSS_RECOVERY
  1391. DWIN_UpdateLCD();
  1392. }
  1393. void DWIN_HandleScreen() {
  1394. switch (checkkey) {
  1395. case MainMenu: HMI_MainMenu(); break;
  1396. case Menu: HMI_Menu(); break;
  1397. case SetInt: HMI_SetInt(); break;
  1398. case SetPInt: HMI_SetPInt(); break;
  1399. case SetIntNoDraw: HMI_SetIntNoDraw(); break;
  1400. case SetFloat: HMI_SetFloat(); break;
  1401. case SetPFloat: HMI_SetPFloat(); break;
  1402. case SelectFile: HMI_SelectFile(); break;
  1403. case Homing: break;
  1404. case Leveling: break;
  1405. case PrintProcess: HMI_Printing(); break;
  1406. case PrintDone: HMI_PrintDone(); break;
  1407. case PauseOrStop: HMI_PauseOrStop(); break;
  1408. case Info: HMI_Popup(); break;
  1409. case WaitResponse: HMI_Popup(); break;
  1410. #if ENABLED(ADVANCED_PAUSE_FEATURE)
  1411. case FilamentPurge: HMI_FilamentPurge(); break;
  1412. #endif
  1413. case NothingToDo: break;
  1414. case Locked: HMI_LockScreen(); break;
  1415. default: break;
  1416. }
  1417. }
  1418. void HMI_SaveProcessID(const uint8_t id) {
  1419. if (checkkey != id) {
  1420. if ((checkkey != NothingToDo) &&
  1421. (checkkey != WaitResponse) &&
  1422. (checkkey != Homing) &&
  1423. (checkkey != Leveling) &&
  1424. (checkkey != PauseOrStop) &&
  1425. (checkkey != FilamentPurge)) last_checkkey = checkkey; // if not a popup
  1426. checkkey = id;
  1427. }
  1428. }
  1429. void DWIN_StartHoming() {
  1430. HMI_flag.home_flag = true;
  1431. HMI_SaveProcessID(Homing);
  1432. DWIN_Draw_Popup(ICON_BLTouch, "Axis Homing", "Please wait until done.");
  1433. }
  1434. void DWIN_CompletedHoming() {
  1435. HMI_flag.home_flag = false;
  1436. dwin_zoffset = TERN0(HAS_BED_PROBE, probe.offset.z);
  1437. if (dwin_abort_flag) {
  1438. planner.finish_and_disable();
  1439. }
  1440. HMI_ReturnScreen();
  1441. }
  1442. void DWIN_MeshLevelingStart() {
  1443. #if HAS_ONESTEP_LEVELING
  1444. HMI_SaveProcessID(Leveling);
  1445. DWIN_Draw_Popup(ICON_AutoLeveling, GET_TEXT(MSG_BED_LEVELING), "Please wait until done.");
  1446. #elif ENABLED(MESH_BED_LEVELING)
  1447. Draw_ManualMesh_Menu();
  1448. #endif
  1449. }
  1450. void DWIN_CompletedLeveling() { HMI_ReturnScreen(); }
  1451. #if ENABLED(MESH_BED_LEVELING)
  1452. void DWIN_MeshUpdate(const int8_t xpos, const int8_t ypos, const float zval) {
  1453. char msg[33] = "";
  1454. char str_1[6] = "";
  1455. sprintf_P(msg, PSTR(S_FMT " %i/%i Z=%s"), GET_TEXT(MSG_PROBING_MESH), xpos, ypos,
  1456. dtostrf(zval, 1, 2, str_1));
  1457. ui.set_status(msg);
  1458. }
  1459. #endif
  1460. // PID process
  1461. void DWIN_PidTuning(pidresult_t result) {
  1462. switch (result) {
  1463. case PID_BED_START:
  1464. HMI_SaveProcessID(NothingToDo);
  1465. DWIN_Draw_Popup(ICON_TempTooHigh, GET_TEXT(MSG_PID_AUTOTUNE), "for BED is running.");
  1466. break;
  1467. case PID_EXTR_START:
  1468. HMI_SaveProcessID(NothingToDo);
  1469. DWIN_Draw_Popup(ICON_TempTooHigh, GET_TEXT(MSG_PID_AUTOTUNE), "for Nozzle is running.");
  1470. break;
  1471. case PID_BAD_EXTRUDER_NUM:
  1472. checkkey = last_checkkey;
  1473. DWIN_Popup_Confirm(ICON_TempTooLow, "PID Autotune failed!", "Bad extruder");
  1474. break;
  1475. case PID_TUNING_TIMEOUT:
  1476. checkkey = last_checkkey;
  1477. DWIN_Popup_Confirm(ICON_TempTooHigh, "Error", GET_TEXT(MSG_PID_TIMEOUT));
  1478. break;
  1479. case PID_TEMP_TOO_HIGH:
  1480. checkkey = last_checkkey;
  1481. DWIN_Popup_Confirm(ICON_TempTooHigh, "PID Autotune failed!", "Temperature too high");
  1482. break;
  1483. case PID_DONE:
  1484. checkkey = last_checkkey;
  1485. DWIN_Popup_Confirm(ICON_TempTooLow, GET_TEXT(MSG_PID_AUTOTUNE), GET_TEXT(MSG_BUTTON_DONE));
  1486. break;
  1487. default:
  1488. checkkey = last_checkkey;
  1489. break;
  1490. }
  1491. }
  1492. // Update filename on print
  1493. void DWIN_Print_Header(const char *text = nullptr) {
  1494. static char headertxt[31] = ""; // Print header text
  1495. if (text != nullptr) {
  1496. const int8_t size = _MIN((unsigned) 30, strlen_P(text));
  1497. LOOP_L_N(i, size) headertxt[i] = text[i];
  1498. headertxt[size] = '\0';
  1499. }
  1500. if (checkkey == PrintProcess || checkkey == PrintDone) {
  1501. DWIN_Draw_Rectangle(1, HMI_data.Background_Color, 0, 60, DWIN_WIDTH, 60+16);
  1502. DWINUI::Draw_CenteredString(60, headertxt);
  1503. }
  1504. }
  1505. void Draw_Title(TitleClass* title) {
  1506. DWIN_Draw_Rectangle(1, HMI_data.TitleBg_color, 0, 0, DWIN_WIDTH - 1, TITLE_HEIGHT - 1);
  1507. if (title->frameid)
  1508. DWIN_Frame_AreaCopy(title->frameid, title->frame.left, title->frame.top, title->frame.right, title->frame.bottom, 14, (TITLE_HEIGHT - (title->frame.bottom - title->frame.top)) / 2 - 1);
  1509. else
  1510. DWIN_Draw_String(false, false, DWIN_FONT_HEAD, HMI_data.TitleTxt_color, HMI_data.TitleBg_color, 14, (TITLE_HEIGHT - DWINUI::Get_font_height(DWIN_FONT_HEAD)) / 2 - 1, title->caption);
  1511. }
  1512. void Draw_Menu(MenuClass* menu) {
  1513. DWINUI::SetColors(HMI_data.Text_Color, HMI_data.Background_Color);
  1514. DWIN_Draw_Rectangle(1, DWINUI::backcolor, 0, TITLE_HEIGHT, DWIN_WIDTH - 1, STATUS_Y - 1);
  1515. ui.set_status("");
  1516. }
  1517. // Startup routines
  1518. void DWIN_Startup() {
  1519. DWINUI::Init();
  1520. DWINUI::onCursorDraw = Draw_Menu_Cursor;
  1521. DWINUI::onCursorErase = Erase_Menu_Cursor;
  1522. DWINUI::onTitleDraw = Draw_Title;
  1523. DWINUI::onMenuDraw = Draw_Menu;
  1524. HMI_SetLanguage();
  1525. }
  1526. void DWIN_DrawStatusLine(const uint16_t color, const uint16_t bgcolor, const char *text) {
  1527. DWIN_Draw_Rectangle(1, bgcolor, 0, STATUS_Y, DWIN_WIDTH, STATUS_Y + 20);
  1528. if (text) DWINUI::Draw_CenteredString(color, STATUS_Y + 2, text);
  1529. DWIN_UpdateLCD();
  1530. }
  1531. // Update Status line
  1532. void DWIN_StatusChanged(const char *text) {
  1533. DWIN_DrawStatusLine(HMI_data.StatusTxt_Color, HMI_data.StatusBg_Color, text);
  1534. }
  1535. void DWIN_StatusChanged_P(PGM_P const pstr) {
  1536. char str[strlen_P((const char*)pstr) + 1];
  1537. strcpy_P(str, (const char*)pstr);
  1538. DWIN_StatusChanged(str);
  1539. }
  1540. // Started a Print Job
  1541. void DWIN_Print_Started(const bool sd) {
  1542. sdprint = card.isPrinting() || sd;
  1543. _percent_done = 0;
  1544. _remain_time = 0;
  1545. Goto_PrintProcess();
  1546. }
  1547. // Ended print job
  1548. void DWIN_Print_Finished() {
  1549. if (checkkey == PrintProcess || printingIsActive()) {
  1550. thermalManager.disable_all_heaters();
  1551. thermalManager.zero_fan_speeds();
  1552. HMI_flag.print_finish = true;
  1553. }
  1554. }
  1555. // Progress Bar update
  1556. void DWIN_Progress_Update() {
  1557. if (parser.seenval('P')) _percent_done = parser.byteval('P');
  1558. if (parser.seenval('R')) _remain_time = parser.ulongval('R') * 60;
  1559. if (checkkey == PrintProcess) {
  1560. Draw_Print_ProgressBar();
  1561. Draw_Print_ProgressRemain();
  1562. Draw_Print_ProgressElapsed();
  1563. }
  1564. }
  1565. #if HAS_FILAMENT_SENSOR
  1566. // Filament Runout process
  1567. void DWIN_FilamentRunout(const uint8_t extruder) { ui.set_status_P(GET_TEXT(MSG_RUNOUT_SENSOR)); }
  1568. #endif
  1569. void DWIN_SetColorDefaults() {
  1570. HMI_data.Background_Color = Def_Background_Color;
  1571. HMI_data.Cursor_color = Def_Cursor_color;
  1572. HMI_data.TitleBg_color = Def_TitleBg_color;
  1573. HMI_data.TitleTxt_color = Def_TitleTxt_color;
  1574. HMI_data.Text_Color = Def_Text_Color;
  1575. HMI_data.Selected_Color = Def_Selected_Color;
  1576. HMI_data.SplitLine_Color = Def_SplitLine_Color;
  1577. HMI_data.Highlight_Color = Def_Highlight_Color;
  1578. HMI_data.StatusBg_Color = Def_StatusBg_Color;
  1579. HMI_data.StatusTxt_Color = Def_StatusTxt_Color;
  1580. HMI_data.PopupBg_color = Def_PopupBg_color;
  1581. HMI_data.PopupTxt_Color = Def_PopupTxt_Color;
  1582. HMI_data.AlertBg_Color = Def_AlertBg_Color;
  1583. HMI_data.AlertTxt_Color = Def_AlertTxt_Color;
  1584. HMI_data.PercentTxt_Color = Def_PercentTxt_Color;
  1585. HMI_data.Barfill_Color = Def_Barfill_Color;
  1586. HMI_data.Indicator_Color = Def_Indicator_Color;
  1587. HMI_data.Coordinate_Color = Def_Coordinate_Color;
  1588. }
  1589. void DWIN_SetDataDefaults() {
  1590. DWIN_SetColorDefaults();
  1591. DWINUI::SetColors(HMI_data.Text_Color, HMI_data.Background_Color);
  1592. TERN_(HAS_HOTEND, HMI_data.HotendPidT = PREHEAT_1_TEMP_HOTEND);
  1593. TERN_(HAS_HEATED_BED, HMI_data.BedPidT = PREHEAT_1_TEMP_BED);
  1594. TERN_(HAS_HOTEND, HMI_data.PidCycles = 5);
  1595. }
  1596. void DWIN_StoreSettings(char *buff) {
  1597. memcpy(buff, &HMI_data, min(sizeof(HMI_data), eeprom_data_size));
  1598. }
  1599. void DWIN_LoadSettings(const char *buff) {
  1600. memcpy(&HMI_data, buff, min(sizeof(HMI_data), eeprom_data_size));
  1601. dwin_zoffset = TERN0(HAS_BED_PROBE, probe.offset.z);
  1602. if (HMI_data.Text_Color == HMI_data.Background_Color) DWIN_SetColorDefaults();
  1603. DWINUI::SetColors(HMI_data.Text_Color, HMI_data.Background_Color);
  1604. TERN_(PREVENT_COLD_EXTRUSION, ApplyExtMinT());
  1605. }
  1606. void MarlinUI::kill_screen(PGM_P lcd_error, PGM_P lcd_component) {
  1607. DWIN_Draw_Popup(ICON_BLTouch, lcd_error, lcd_component);
  1608. DWIN_UpdateLCD();
  1609. }
  1610. void DWIN_RebootScreen() {
  1611. DWIN_Frame_Clear(Color_Bg_Black);
  1612. DWINUI::Draw_Icon(ICON_LOGO, 71, 150); // CREALITY logo
  1613. DWINUI::Draw_CenteredString(Color_White, 200, F("Please wait until reboot."));
  1614. DWIN_UpdateLCD();
  1615. delay(500);
  1616. }
  1617. void DWIN_Redraw_screen() {
  1618. Draw_Main_Area();
  1619. DWIN_StatusChanged(ui.status_message);
  1620. Draw_Status_Area(false);
  1621. }
  1622. #if ENABLED(ADVANCED_PAUSE_FEATURE)
  1623. void DWIN_Popup_Pause(const char *msg, uint8_t button = 0) {
  1624. HMI_SaveProcessID(button ? WaitResponse : NothingToDo);
  1625. DWIN_Draw_Popup(ICON_BLTouch, "Advanced Pause", msg, button);
  1626. ui.reset_status(true);
  1627. }
  1628. void MarlinUI::pause_show_message(const PauseMessage message, const PauseMode mode/*=PAUSE_MODE_SAME*/, const uint8_t extruder/*=active_extruder*/) {
  1629. switch (message) {
  1630. case PAUSE_MESSAGE_PARKING: DWIN_Popup_Pause(GET_TEXT(MSG_PAUSE_PRINT_PARKING)); break;
  1631. case PAUSE_MESSAGE_CHANGING: DWIN_Popup_Pause(GET_TEXT(MSG_FILAMENT_CHANGE_INIT)); break;
  1632. case PAUSE_MESSAGE_UNLOAD: DWIN_Popup_Pause(GET_TEXT(MSG_FILAMENT_CHANGE_UNLOAD)); break;
  1633. case PAUSE_MESSAGE_WAITING: DWIN_Popup_Pause(GET_TEXT(MSG_ADVANCED_PAUSE_WAITING), ICON_Continue_E); break;
  1634. case PAUSE_MESSAGE_INSERT: DWIN_Popup_Continue(ICON_BLTouch, "Advanced Pause", GET_TEXT(MSG_FILAMENT_CHANGE_INSERT)); break;
  1635. case PAUSE_MESSAGE_LOAD: DWIN_Popup_Pause(GET_TEXT(MSG_FILAMENT_CHANGE_LOAD)); break;
  1636. case PAUSE_MESSAGE_PURGE: DWIN_Popup_Pause(GET_TEXT(MSG_FILAMENT_CHANGE_PURGE)); break;
  1637. case PAUSE_MESSAGE_OPTION: DWIN_Popup_FilamentPurge(); break;
  1638. case PAUSE_MESSAGE_RESUME: DWIN_Popup_Pause(GET_TEXT(MSG_FILAMENT_CHANGE_RESUME)); break;
  1639. case PAUSE_MESSAGE_HEAT: DWIN_Popup_Pause(GET_TEXT(MSG_FILAMENT_CHANGE_HEAT), ICON_Continue_E); break;
  1640. case PAUSE_MESSAGE_HEATING: ui.set_status_P(GET_TEXT(MSG_FILAMENT_CHANGE_HEATING)); break;
  1641. case PAUSE_MESSAGE_STATUS: HMI_ReturnScreen(); break;
  1642. default: break;
  1643. }
  1644. }
  1645. void Draw_Popup_FilamentPurge() {
  1646. DWIN_Draw_Popup(ICON_BLTouch, "Advanced Pause", "Purge or Continue?");
  1647. DWINUI::Draw_Icon(ICON_Confirm_E, 26, 280);
  1648. DWINUI::Draw_Icon(ICON_Continue_E, 146, 280);
  1649. Draw_Select_Highlight(true);
  1650. }
  1651. // Handle responses such as:
  1652. // - Purge More, Continue
  1653. // - General "Continue" response
  1654. void DWIN_Popup_FilamentPurge() {
  1655. HMI_SaveProcessID(FilamentPurge);
  1656. pause_menu_response = PAUSE_RESPONSE_WAIT_FOR;
  1657. Draw_Popup_FilamentPurge();
  1658. }
  1659. void HMI_FilamentPurge() {
  1660. ENCODER_DiffState encoder_diffState = get_encoder_state();
  1661. if (encoder_diffState == ENCODER_DIFF_NO) return;
  1662. if (encoder_diffState == ENCODER_DIFF_CW)
  1663. Draw_Select_Highlight(false);
  1664. else if (encoder_diffState == ENCODER_DIFF_CCW)
  1665. Draw_Select_Highlight(true);
  1666. else if (encoder_diffState == ENCODER_DIFF_ENTER) {
  1667. if (HMI_flag.select_flag)
  1668. pause_menu_response = PAUSE_RESPONSE_EXTRUDE_MORE; // "Purge More" button
  1669. else {
  1670. HMI_SaveProcessID(NothingToDo);
  1671. pause_menu_response = PAUSE_RESPONSE_RESUME_PRINT; // "Continue" button
  1672. }
  1673. }
  1674. DWIN_UpdateLCD();
  1675. }
  1676. #endif // ADVANCED_PAUSE_FEATURE
  1677. void HMI_LockScreen() {
  1678. ENCODER_DiffState encoder_diffState = get_encoder_state();
  1679. if (encoder_diffState == ENCODER_DIFF_NO) return;
  1680. LockScreen.onEncoderState(encoder_diffState);
  1681. if (LockScreen.isUnlocked()) {
  1682. if (CurrentMenu == AdvancedSettings)
  1683. Draw_AdvancedSettings_Menu();
  1684. else
  1685. Draw_Tune_Menu();
  1686. }
  1687. }
  1688. void DWIN_LockScreen(const bool flag) {
  1689. HMI_flag.lock_flag = flag;
  1690. checkkey = Locked;
  1691. LockScreen.Init();
  1692. }
  1693. //
  1694. // NEW MENU SUBSYSTEM =========================================================
  1695. //
  1696. // On click functions
  1697. // Generic onclick event without draw anything
  1698. // process: process id HMI destiny
  1699. // lo: low limit
  1700. // hi: high limit
  1701. // dp: decimal places, 0 for integers
  1702. // val: value
  1703. // LiveUpdate: live update function when the encoder changes
  1704. // Apply: update function when the encoder is pressed
  1705. void SetOnClick(uint8_t process, const int32_t lo, const int32_t hi, uint8_t dp, const int32_t val, void (*Apply)() = nullptr, void (*LiveUpdate)() = nullptr) {
  1706. last_checkkey = checkkey;
  1707. checkkey = process;
  1708. HMI_value.MinValue = lo;
  1709. HMI_value.MaxValue = hi;
  1710. HMI_value.dp = dp;
  1711. HMI_value.Apply = Apply;
  1712. HMI_value.LiveUpdate = LiveUpdate;
  1713. HMI_value.Value = val;
  1714. EncoderRate.enabled = true;
  1715. }
  1716. // Generic onclick event for set values (dp = 0: integer, dp > 0: float)
  1717. // process: process id HMI destiny
  1718. // lo: scaled low limit
  1719. // hi: scaled high limit
  1720. // dp: decimal places, 0 for integers
  1721. // val: scaled value
  1722. // LiveUpdate: live update function when the encoder changes
  1723. // Apply: update function when the encoder is pressed
  1724. void SetValueOnClick(uint8_t process, const int32_t lo, const int32_t hi, uint8_t dp, const int32_t val, void (*Apply)() = nullptr, void (*LiveUpdate)() = nullptr) {
  1725. SetOnClick(process, lo, hi, dp, val, Apply, LiveUpdate);
  1726. dp ? DWINUI::Draw_Signed_Float(HMI_data.Text_Color, HMI_data.Selected_Color, 3, dp, VALX - dp * DWINUI::Get_font_width(DWIN_FONT_MENU), MBASE(CurrentMenu->line()), HMI_value.Value)
  1727. : Draw_Menu_IntValue(HMI_data.Selected_Color, CurrentMenu->line(), 4, HMI_value.Value);
  1728. }
  1729. // Generic onclick event for integer values
  1730. // lo: scaled low limit
  1731. // hi: scaled high limit
  1732. // val: value
  1733. // LiveUpdate: live update function when the encoder changes
  1734. // Apply: update function when the encoder is pressed
  1735. void SetIntOnClick(const int32_t lo, const int32_t hi, const int32_t val, void (*Apply)() = nullptr, void (*LiveUpdate)() = nullptr) {
  1736. SetValueOnClick(SetInt, lo, hi, 0, val, Apply, LiveUpdate);
  1737. }
  1738. // Generic onclick event for set pointer to 16 bit uinteger values
  1739. // lo: low limit
  1740. // hi: high limit
  1741. // LiveUpdate: live update function when the encoder changes
  1742. // Apply: update function when the encoder is pressed
  1743. void SetPIntOnClick(const int32_t lo, const int32_t hi, void (*Apply)() = nullptr, void (*LiveUpdate)() = nullptr) {
  1744. HMI_value.P_Int = (int16_t*)static_cast<MenuItemPtrClass*>(CurrentMenu->SelectedItem())->value;
  1745. const int32_t value = *HMI_value.P_Int;
  1746. SetValueOnClick(SetPInt, lo, hi, 0, value, Apply, LiveUpdate);
  1747. }
  1748. // Generic onclick event for float values
  1749. // process: process id HMI destiny
  1750. // lo: low limit
  1751. // hi: high limit
  1752. // dp: decimal places
  1753. // val: value
  1754. void SetFloatOnClick(const float lo, const float hi, uint8_t dp, const float val, void (*Apply)() = nullptr, void (*LiveUpdate)() = nullptr) {
  1755. SetValueOnClick(SetFloat, lo * POW(10, dp), hi * POW(10, dp), dp, val * POW(10, dp), Apply, LiveUpdate);
  1756. }
  1757. // Generic onclick event for set pointer to float values
  1758. // lo: low limit
  1759. // hi: high limit
  1760. // LiveUpdate: live update function when the encoder changes
  1761. // Apply: update function when the encoder is pressed
  1762. void SetPFloatOnClick(const float lo, const float hi, uint8_t dp, void (*Apply)() = nullptr, void (*LiveUpdate)() = nullptr) {
  1763. HMI_value.P_Float = (float*)static_cast<MenuItemPtrClass*>(CurrentMenu->SelectedItem())->value;
  1764. const int32_t value = *HMI_value.P_Float * POW(10, dp);
  1765. SetValueOnClick(SetPFloat, lo * POW(10, dp), hi * POW(10, dp), dp, value, Apply, LiveUpdate);
  1766. }
  1767. #if ENABLED(EEPROM_SETTINGS)
  1768. void WriteEeprom() {
  1769. const bool success = settings.save();
  1770. HMI_AudioFeedback(success);
  1771. }
  1772. void ReadEeprom() {
  1773. const bool success = settings.load();
  1774. DWIN_Redraw_screen();
  1775. HMI_AudioFeedback(success);
  1776. }
  1777. void ResetEeprom() {
  1778. settings.reset();
  1779. DWIN_Redraw_screen();
  1780. HMI_AudioFeedback();
  1781. }
  1782. #endif
  1783. // Reset Printer
  1784. void RebootPrinter() {
  1785. dwin_abort_flag = true;
  1786. wait_for_heatup = wait_for_user = false; // Stop waiting for heating/user
  1787. thermalManager.disable_all_heaters();
  1788. planner.finish_and_disable();
  1789. DWIN_RebootScreen();
  1790. HAL_reboot();
  1791. }
  1792. void Goto_InfoMenu(){
  1793. checkkey = Info;
  1794. Draw_Info_Menu();
  1795. }
  1796. void DisableMotors() {
  1797. queue.inject_P(PSTR("M84"));
  1798. }
  1799. void AutoHome() {
  1800. queue.inject_P(G28_STR);
  1801. }
  1802. void SetHome() {
  1803. // Apply workspace offset, making the current position 0,0,0
  1804. queue.inject_P(PSTR("G92 X0 Y0 Z0"));
  1805. HMI_AudioFeedback();
  1806. }
  1807. #if HAS_ZOFFSET_ITEM
  1808. bool printer_busy() { return planner.movesplanned() || printingIsActive(); }
  1809. void ApplyZOffset() { TERN_(EEPROM_SETTINGS, settings.save()); }
  1810. void LiveZOffset() {
  1811. last_zoffset = dwin_zoffset;
  1812. dwin_zoffset = HMI_value.Value / 100.0f;
  1813. #if EITHER(BABYSTEP_ZPROBE_OFFSET, JUST_BABYSTEP)
  1814. if (BABYSTEP_ALLOWED()) babystep.add_mm(Z_AXIS, dwin_zoffset - last_zoffset);
  1815. #endif
  1816. }
  1817. #if EITHER(HAS_BED_PROBE, BABYSTEPPING)
  1818. void SetZOffset() { SetPFloatOnClick(Z_PROBE_OFFSET_RANGE_MIN, Z_PROBE_OFFSET_RANGE_MAX, 2, ApplyZOffset, LiveZOffset); }
  1819. #endif
  1820. #endif
  1821. #if HAS_PREHEAT
  1822. void SetPreheat(const uint8_t i) {
  1823. TERN_(HAS_HOTEND, thermalManager.setTargetHotend(ui.material_preset[i].hotend_temp, 0));
  1824. TERN_(HAS_HEATED_BED, thermalManager.setTargetBed(ui.material_preset[i].bed_temp));
  1825. TERN_(HAS_FAN, thermalManager.set_fan_speed(0, ui.material_preset[i].fan_speed));
  1826. }
  1827. void SetPreheat0() { SetPreheat(0); }
  1828. void SetPreheat1() { SetPreheat(1); }
  1829. void SetPreheat2() { SetPreheat(2); }
  1830. void SetCoolDown() {
  1831. TERN_(HAS_FAN, thermalManager.zero_fan_speeds());
  1832. #if HAS_HOTEND || HAS_HEATED_BED
  1833. thermalManager.disable_all_heaters();
  1834. #endif
  1835. }
  1836. #endif
  1837. void SetLanguage() {
  1838. HMI_ToggleLanguage();
  1839. CurrentMenu = nullptr; // Invalidate menu to full redraw
  1840. Draw_Prepare_Menu();
  1841. }
  1842. void LiveMove() {
  1843. *HMI_value.P_Float = HMI_value.Value / MINUNITMULT;
  1844. if (!planner.is_full()) {
  1845. planner.synchronize();
  1846. planner.buffer_line(current_position, homing_feedrate(HMI_value.axis));
  1847. }
  1848. }
  1849. void ApplyMoveE() {
  1850. last_E = HMI_value.Value / MINUNITMULT;
  1851. if (!planner.is_full()) {
  1852. planner.synchronize();
  1853. planner.buffer_line(current_position, MMM_TO_MMS(FEEDRATE_E));
  1854. }
  1855. }
  1856. void SetMoveX() { HMI_value.axis = X_AXIS; SetPFloatOnClick(X_MIN_POS, X_MAX_POS, UNITFDIGITS, planner.synchronize, LiveMove);}
  1857. void SetMoveY() { HMI_value.axis = Y_AXIS; SetPFloatOnClick(Y_MIN_POS, Y_MAX_POS, UNITFDIGITS, planner.synchronize, LiveMove);}
  1858. void SetMoveZ() { HMI_value.axis = Z_AXIS; SetPFloatOnClick(Z_MIN_POS, Z_MAX_POS, UNITFDIGITS, planner.synchronize, LiveMove);}
  1859. #if HAS_HOTEND
  1860. void SetMoveE() {
  1861. #ifdef PREVENT_COLD_EXTRUSION
  1862. if (thermalManager.tooColdToExtrude(0)) {
  1863. Popup_Window_ETempTooLow();
  1864. return;
  1865. }
  1866. #endif
  1867. SetPFloatOnClick(last_E - (EXTRUDE_MAXLENGTH), last_E + (EXTRUDE_MAXLENGTH), UNITFDIGITS, ApplyMoveE);
  1868. }
  1869. #endif
  1870. void SetMoveZto0() {
  1871. char cmd[48] = "";
  1872. char str_1[5] = "", str_2[5] = "";
  1873. sprintf_P(cmd, PSTR("G28OXY\nG28Z\nG0X%sY%sF5000\nG0Z0F300"),
  1874. dtostrf(X_CENTER, 1, 1, str_1),
  1875. dtostrf(Y_CENTER, 1, 1, str_2));
  1876. gcode.process_subcommands_now_P(cmd);
  1877. planner.synchronize();
  1878. ui.set_status_P(PSTR("Now adjust Z Offset"));
  1879. HMI_AudioFeedback(true);
  1880. }
  1881. void SetPID(celsius_t t, heater_id_t h) {
  1882. char cmd[48] = "";
  1883. char str_1[5] = "", str_2[5] = "";
  1884. sprintf_P(cmd, PSTR("G28OXY\nG0Z5F300\nG0X%sY%sF5000\nM84"),
  1885. dtostrf(X_CENTER, 1, 1, str_1),
  1886. dtostrf(Y_CENTER, 1, 1, str_2));
  1887. gcode.process_subcommands_now_P(cmd);
  1888. planner.synchronize();
  1889. thermalManager.PID_autotune(t, h, HMI_data.PidCycles, true);
  1890. }
  1891. TERN_(HAS_HOTEND, void HotendPID() { SetPID(HMI_data.HotendPidT, H_E0); })
  1892. TERN_(HAS_HEATED_BED, void BedPID() { SetPID(HMI_data.BedPidT, H_BED); })
  1893. void SetPwrLossr() {
  1894. recovery.enable(!recovery.enabled);
  1895. Draw_Chkb_Line(CurrentMenu->line(), recovery.enabled);
  1896. DWIN_UpdateLCD();
  1897. }
  1898. #if HAS_LCD_BRIGHTNESS
  1899. void ApplyBrightness() { ui.set_brightness(HMI_value.Value); }
  1900. void LiveBrightness() { DWIN_LCD_Brightness(HMI_value.Value); }
  1901. void SetBrightness() { SetIntOnClick(MIN_LCD_BRIGHTNESS, MAX_LCD_BRIGHTNESS, ui.brightness, ApplyBrightness, LiveBrightness); }
  1902. #endif
  1903. #if ENABLED(SOUND_MENU_ITEM)
  1904. void SetEnableSound() {
  1905. ui.buzzer_enabled = !ui.buzzer_enabled;
  1906. Draw_Chkb_Line(CurrentMenu->line(), ui.buzzer_enabled);
  1907. DWIN_UpdateLCD();
  1908. }
  1909. #endif
  1910. void Goto_LockScreen() {
  1911. DWIN_LockScreen(true);
  1912. }
  1913. #if HAS_HOME_OFFSET
  1914. void ApplyHomeOffset() { set_home_offset(HMI_value.axis, HMI_value.Value / MINUNITMULT); }
  1915. void SetHomeOffsetX() { HMI_value.axis = X_AXIS; SetPFloatOnClick(-50, 50, UNITFDIGITS, ApplyHomeOffset); }
  1916. void SetHomeOffsetY() { HMI_value.axis = Y_AXIS; SetPFloatOnClick(-50, 50, UNITFDIGITS, ApplyHomeOffset); }
  1917. void SetHomeOffsetZ() { HMI_value.axis = Z_AXIS; SetPFloatOnClick( -2, 2, UNITFDIGITS, ApplyHomeOffset); }
  1918. #endif
  1919. #if HAS_BED_PROBE
  1920. void SetProbeOffsetX() { SetPFloatOnClick(-50, 50, UNITFDIGITS); }
  1921. void SetProbeOffsetY() { SetPFloatOnClick(-50, 50, UNITFDIGITS); }
  1922. void ProbeTest() {
  1923. ui.set_status_P(GET_TEXT(MSG_M48_TEST));
  1924. queue.inject_P(PSTR("G28O\nM48 P10"));
  1925. }
  1926. #endif
  1927. #if HAS_FILAMENT_SENSOR
  1928. void SetRunoutEnable() {
  1929. runout.reset();
  1930. runout.enabled = !runout.enabled;
  1931. Draw_Chkb_Line(CurrentMenu->line(), runout.enabled);
  1932. DWIN_UpdateLCD();
  1933. }
  1934. #if HAS_FILAMENT_RUNOUT_DISTANCE
  1935. void ApplyRunoutDistance() { runout.set_runout_distance(HMI_value.Value / MINUNITMULT); }
  1936. void SetRunoutDistance() { SetFloatOnClick(0, 999, UNITFDIGITS, runout.runout_distance(), ApplyRunoutDistance); }
  1937. #endif
  1938. #endif
  1939. TERN_(ADVANCED_PAUSE_FEATURE, void SetFilLoad() { SetPFloatOnClick(0, MAX_LOAD_UNLOAD, UNITFDIGITS); });
  1940. TERN_(ADVANCED_PAUSE_FEATURE, void SetFilUnload() { SetPFloatOnClick(0, MAX_LOAD_UNLOAD, UNITFDIGITS); });
  1941. TERN_(PREVENT_COLD_EXTRUSION, void ApplyExtMinT() { thermalManager.extrude_min_temp = HMI_data.ExtMinT; thermalManager.allow_cold_extrude = (HMI_data.ExtMinT == 0); });
  1942. TERN_(PREVENT_COLD_EXTRUSION, void SetExtMinT() { SetPIntOnClick(MIN_ETEMP, MAX_ETEMP, ApplyExtMinT); });
  1943. void RestoreDefaultsColors() {
  1944. DWIN_SetColorDefaults();
  1945. DWINUI::SetColors(HMI_data.Text_Color, HMI_data.Background_Color);
  1946. DWIN_Redraw_screen();
  1947. }
  1948. void SelColor() {
  1949. HMI_value.P_Int = (int16_t*)static_cast<MenuItemPtrClass*>(CurrentMenu->SelectedItem())->value;
  1950. HMI_value.Color[2] = GetRColor(*HMI_value.P_Int); // Red
  1951. HMI_value.Color[1] = GetGColor(*HMI_value.P_Int); // Green
  1952. HMI_value.Color[0] = GetBColor(*HMI_value.P_Int); // Blue
  1953. Draw_GetColor_Menu();
  1954. }
  1955. void LiveRGBColor() {
  1956. HMI_value.Color[CurrentMenu->line() - 2] = HMI_value.Value;
  1957. uint16_t color = RGB(HMI_value.Color[2], HMI_value.Color[1], HMI_value.Color[0]);
  1958. DWIN_Draw_Rectangle(1, color, 20, 315, DWIN_WIDTH - 20, 335);
  1959. }
  1960. void SetRGBColor() {
  1961. const uint8_t line = CurrentMenu->line() - 2;
  1962. SetIntOnClick(0, (line == 1) ? 63 : 31, HMI_value.Color[CurrentMenu->SelectedItem()->icon], nullptr, LiveRGBColor);
  1963. }
  1964. void DWIN_ApplyColor() {
  1965. *HMI_value.P_Int = RGB(HMI_value.Color[2], HMI_value.Color[1], HMI_value.Color[0]);
  1966. DWINUI::SetColors(HMI_data.Text_Color, HMI_data.Background_Color);
  1967. Draw_Status_Area(false);
  1968. Draw_SelectColors_Menu();
  1969. ui.set_status_P(PSTR("Colors applied"));
  1970. }
  1971. void SetSpeed() { SetPIntOnClick(MIN_PRINT_SPEED, MAX_PRINT_SPEED); }
  1972. #if HAS_HOTEND
  1973. void ApplyHotendTemp() { thermalManager.setTargetHotend(HMI_value.Value, 0); }
  1974. void SetHotendTemp() { SetIntOnClick(HEATER_0_MINTEMP, HEATER_0_MAXTEMP, thermalManager.degTargetHotend(0), ApplyHotendTemp); }
  1975. #endif
  1976. #if HAS_HEATED_BED
  1977. void ApplyBedTemp() { thermalManager.setTargetBed(HMI_value.Value); }
  1978. void SetBedTemp() { SetIntOnClick(BED_MINTEMP, BED_MAX_TARGET, thermalManager.degTargetBed(), ApplyBedTemp); }
  1979. #endif
  1980. #if HAS_FAN
  1981. void ApplyFanSpeed() { thermalManager.set_fan_speed(0, HMI_value.Value); }
  1982. void SetFanSpeed() { SetIntOnClick(0, 255, thermalManager.fan_speed[0], ApplyFanSpeed); }
  1983. #endif
  1984. #if ENABLED(ADVANCED_PAUSE_FEATURE)
  1985. void ChangeFilament() {
  1986. HMI_SaveProcessID(NothingToDo);
  1987. queue.inject_P(PSTR("M600 B2"));
  1988. }
  1989. void ParkHead(){
  1990. ui.set_status_P(GET_TEXT(MSG_FILAMENT_PARK_ENABLED));
  1991. queue.inject_P(PSTR("G28O\nG27"));
  1992. }
  1993. #if ENABLED(FILAMENT_LOAD_UNLOAD_GCODES)
  1994. void UnloadFilament(){
  1995. ui.set_status_P(GET_TEXT(MSG_FILAMENTUNLOAD));
  1996. queue.inject_P(PSTR("M702 Z20"));
  1997. }
  1998. void LoadFilament(){
  1999. ui.set_status_P(GET_TEXT(MSG_FILAMENTLOAD));
  2000. queue.inject_P(PSTR("M701 Z20"));
  2001. }
  2002. #endif
  2003. #endif
  2004. void SetFlow() { SetPIntOnClick(MIN_PRINT_FLOW, MAX_PRINT_FLOW); }
  2005. // Leveling Bed Corners
  2006. void LevBed(uint8_t point) {
  2007. char cmd[100] = "";
  2008. #if HAS_ONESTEP_LEVELING
  2009. char str_1[6] = "", str_2[6] = "", str_3[6] = "";
  2010. #define fmt "X:%s, Y:%s, Z:%s"
  2011. float xpos = 0, ypos = 0, zval = 0;
  2012. float margin = PROBING_MARGIN;
  2013. #else
  2014. #define fmt "M420 S0\nG28O\nG90\nG0 Z5 F300\nG0 X%i Y%i F5000\nG0 Z0 F300"
  2015. int16_t xpos = 0, ypos = 0;
  2016. int16_t margin = 30;
  2017. #endif
  2018. switch (point) {
  2019. case 0:
  2020. ui.set_status_P(GET_TEXT(MSG_LEVBED_FL));
  2021. xpos = ypos = margin;
  2022. break;
  2023. case 1:
  2024. ui.set_status_P(GET_TEXT(MSG_LEVBED_FR));
  2025. xpos = X_BED_SIZE - margin; ypos = margin;
  2026. break;
  2027. case 2:
  2028. ui.set_status_P(GET_TEXT(MSG_LEVBED_BR));
  2029. xpos = X_BED_SIZE - margin; ypos = Y_BED_SIZE - margin;
  2030. break;
  2031. case 3:
  2032. ui.set_status_P(GET_TEXT(MSG_LEVBED_BL));
  2033. xpos = margin; ypos = Y_BED_SIZE - margin;
  2034. break;
  2035. case 4:
  2036. ui.set_status_P(GET_TEXT(MSG_LEVBED_C));
  2037. xpos = X_BED_SIZE / 2; ypos = Y_BED_SIZE / 2;
  2038. break;
  2039. }
  2040. #if HAS_ONESTEP_LEVELING
  2041. planner.synchronize();
  2042. gcode.process_subcommands_now_P(PSTR("M420S0\nG28O"));
  2043. planner.synchronize();
  2044. zval = probe.probe_at_point(xpos, ypos, PROBE_PT_STOW);
  2045. sprintf_P(cmd, PSTR(fmt),
  2046. dtostrf(xpos, 1, 1, str_1),
  2047. dtostrf(ypos, 1, 1, str_2),
  2048. dtostrf(zval, 1, 2, str_3)
  2049. );
  2050. ui.set_status_P(cmd);
  2051. #else
  2052. planner.synchronize();
  2053. sprintf_P(cmd, PSTR(fmt), xpos, ypos);
  2054. queue.inject(cmd);
  2055. #endif
  2056. }
  2057. void LevBedFL() { LevBed(0); }
  2058. void LevBedFR() { LevBed(1); }
  2059. void LevBedBR() { LevBed(2); }
  2060. void LevBedBL() { LevBed(3); }
  2061. void LevBedC () { LevBed(4); }
  2062. #if ENABLED(MESH_BED_LEVELING)
  2063. void ManualMeshStart(){
  2064. ui.set_status_P(GET_TEXT(MSG_UBL_BUILD_MESH_MENU));
  2065. gcode.process_subcommands_now_P(PSTR("G28 XYO\nG28 Z\nM211 S0\nG29S1"));
  2066. planner.synchronize();
  2067. #ifdef MANUAL_PROBE_START_Z
  2068. MMeshMoveZItem->Draw(CurrentMenu->line(MMeshMoveZItem->pos));
  2069. #endif
  2070. }
  2071. void SetMMeshMoveZ() { HMI_value.axis = Z_AXIS; SetPFloatOnClick(-1, 1, 2, planner.synchronize, LiveMove);}
  2072. void ManualMeshContinue(){
  2073. gcode.process_subcommands_now_P(PSTR("G29S2"));
  2074. planner.synchronize();
  2075. MMeshMoveZItem->Draw(CurrentMenu->line(MMeshMoveZItem->pos));
  2076. }
  2077. void ManualMeshSave(){
  2078. ui.set_status_P(GET_TEXT(MSG_UBL_STORAGE_MESH_MENU));
  2079. queue.inject_P(PSTR("M211 S1\nM500"));
  2080. }
  2081. #endif
  2082. #if HAS_PREHEAT
  2083. TERN_(HAS_HOTEND, void SetPreheatEndTemp() { SetPIntOnClick(MIN_ETEMP, MAX_ETEMP); });
  2084. TERN_(HAS_HEATED_BED, void SetPreheatBedTemp() { SetPIntOnClick(BED_MINTEMP, BED_MAX_TARGET); });
  2085. TERN_(HAS_FAN, void SetPreheatFanSpeed() { SetPIntOnClick(0, 255); });
  2086. #endif
  2087. void ApplyMaxSpeed() { planner.set_max_feedrate(HMI_value.axis, HMI_value.Value / MINUNITMULT); }
  2088. void SetMaxSpeedX() { HMI_value.axis = X_AXIS, SetFloatOnClick(MIN_MAXFEEDSPEED, default_max_feedrate[X_AXIS] * 2, UNITFDIGITS, planner.settings.max_feedrate_mm_s[X_AXIS], ApplyMaxSpeed); }
  2089. void SetMaxSpeedY() { HMI_value.axis = Y_AXIS, SetFloatOnClick(MIN_MAXFEEDSPEED, default_max_feedrate[Y_AXIS] * 2, UNITFDIGITS, planner.settings.max_feedrate_mm_s[Y_AXIS], ApplyMaxSpeed); }
  2090. void SetMaxSpeedZ() { HMI_value.axis = Z_AXIS, SetFloatOnClick(MIN_MAXFEEDSPEED, default_max_feedrate[Z_AXIS] * 2, UNITFDIGITS, planner.settings.max_feedrate_mm_s[Z_AXIS], ApplyMaxSpeed); }
  2091. TERN_(HAS_HOTEND, void SetMaxSpeedE() { HMI_value.axis = E_AXIS; SetFloatOnClick(MIN_MAXFEEDSPEED, default_max_feedrate[E_AXIS] * 2, UNITFDIGITS, planner.settings.max_feedrate_mm_s[E_AXIS], ApplyMaxSpeed); });
  2092. void ApplyMaxAccel() { planner.set_max_acceleration(HMI_value.axis, HMI_value.Value); }
  2093. void SetMaxAccelX() { HMI_value.axis = X_AXIS, SetIntOnClick(MIN_MAXACCELERATION, default_max_acceleration[X_AXIS] * 2, planner.settings.max_acceleration_mm_per_s2[X_AXIS], ApplyMaxAccel); }
  2094. void SetMaxAccelY() { HMI_value.axis = Y_AXIS, SetIntOnClick(MIN_MAXACCELERATION, default_max_acceleration[Y_AXIS] * 2, planner.settings.max_acceleration_mm_per_s2[Y_AXIS], ApplyMaxAccel); }
  2095. void SetMaxAccelZ() { HMI_value.axis = Z_AXIS, SetIntOnClick(MIN_MAXACCELERATION, default_max_acceleration[Z_AXIS] * 2, planner.settings.max_acceleration_mm_per_s2[Z_AXIS], ApplyMaxAccel); }
  2096. TERN_(HAS_HOTEND, void SetMaxAccelE() { HMI_value.axis = E_AXIS; SetIntOnClick(MIN_MAXACCELERATION, default_max_acceleration[E_AXIS] * 2, planner.settings.max_acceleration_mm_per_s2[E_AXIS], ApplyMaxAccel); });
  2097. #if HAS_CLASSIC_JERK
  2098. void ApplyMaxJerk() { planner.set_max_jerk(HMI_value.axis, HMI_value.Value / MINUNITMULT); }
  2099. void SetMaxJerkX() { HMI_value.axis = X_AXIS, SetFloatOnClick(MIN_MAXJERK, default_max_jerk[X_AXIS] * 2, UNITFDIGITS, planner.max_jerk[X_AXIS], ApplyMaxJerk); }
  2100. void SetMaxJerkY() { HMI_value.axis = Y_AXIS, SetFloatOnClick(MIN_MAXJERK, default_max_jerk[Y_AXIS] * 2, UNITFDIGITS, planner.max_jerk[Y_AXIS], ApplyMaxJerk); }
  2101. void SetMaxJerkZ() { HMI_value.axis = Z_AXIS, SetFloatOnClick(MIN_MAXJERK, default_max_jerk[Z_AXIS] * 2, UNITFDIGITS, planner.max_jerk[Z_AXIS], ApplyMaxJerk); }
  2102. TERN_(HAS_HOTEND, void SetMaxJerkE() { HMI_value.axis = E_AXIS; SetFloatOnClick(MIN_MAXJERK, default_max_jerk[E_AXIS] * 2, UNITFDIGITS, planner.max_jerk[E_AXIS], ApplyMaxJerk); });
  2103. #endif
  2104. void SetStepsX() { HMI_value.axis = X_AXIS, SetPFloatOnClick( MIN_STEP, MAX_STEP, UNITFDIGITS); }
  2105. void SetStepsY() { HMI_value.axis = Y_AXIS, SetPFloatOnClick( MIN_STEP, MAX_STEP, UNITFDIGITS); }
  2106. void SetStepsZ() { HMI_value.axis = Z_AXIS, SetPFloatOnClick( MIN_STEP, MAX_STEP, UNITFDIGITS); }
  2107. TERN_(HAS_HOTEND, void SetStepsE() { HMI_value.axis = E_AXIS; SetPFloatOnClick( MIN_STEP, MAX_STEP, UNITFDIGITS); });
  2108. TERN_(HAS_HOTEND, void SetHotendPidT() { SetPIntOnClick(MIN_ETEMP, MAX_ETEMP); })
  2109. TERN_(HAS_HEATED_BED, void SetBedPidT() { SetPIntOnClick(BED_MINTEMP, BED_MAX_TARGET); })
  2110. #if HAS_HOTEND || HAS_HEATED_BED
  2111. void SetPidCycles() { SetPIntOnClick(3, 50); }
  2112. void SetKp() { SetPFloatOnClick(0, 1000, 2); }
  2113. void ApplyPIDi() {
  2114. *HMI_value.P_Float = scalePID_i(HMI_value.Value / POW(10, 2));
  2115. thermalManager.updatePID();
  2116. }
  2117. void ApplyPIDd() {
  2118. *HMI_value.P_Float = scalePID_d(HMI_value.Value / POW(10, 2));
  2119. thermalManager.updatePID();
  2120. }
  2121. void SetKi() {
  2122. HMI_value.P_Float = (float*)static_cast<MenuItemPtrClass*>(CurrentMenu->SelectedItem())->value;
  2123. const float value = unscalePID_i(*HMI_value.P_Float);
  2124. SetFloatOnClick(0, 1000, 2, value, ApplyPIDi);
  2125. }
  2126. void SetKd() {
  2127. HMI_value.P_Float = (float*)static_cast<MenuItemPtrClass*>(CurrentMenu->SelectedItem())->value;
  2128. const float value = unscalePID_d(*HMI_value.P_Float);
  2129. SetFloatOnClick(0, 1000, 2, value, ApplyPIDd);
  2130. }
  2131. #endif
  2132. // Menuitem Drawing functions =================================================
  2133. void onDrawMenuItem(MenuItemClass* menuitem, int8_t line) {
  2134. if (menuitem->icon) DWINUI::Draw_Icon(menuitem->icon, ICOX, MBASE(line) - 3);
  2135. if (menuitem->frameid)
  2136. DWIN_Frame_AreaCopy(menuitem->frameid, menuitem->frame.left, menuitem->frame.top, menuitem->frame.right, menuitem->frame.bottom, LBLX, MBASE(line));
  2137. else if (menuitem->caption)
  2138. DWINUI::Draw_String(LBLX, MBASE(line) - 1, menuitem->caption);
  2139. DWIN_Draw_HLine(HMI_data.SplitLine_Color, 16, MYPOS(line + 1), 240);
  2140. }
  2141. void onDrawSubMenu(MenuItemClass* menuitem, int8_t line) {
  2142. onDrawMenuItem(menuitem, line);
  2143. DWINUI::Draw_Icon(ICON_More, VALX + 16, MBASE(line) - 3);
  2144. }
  2145. void onDrawIntMenu(MenuItemClass* menuitem, int8_t line, uint16_t value) {
  2146. onDrawMenuItem(menuitem, line);
  2147. Draw_Menu_IntValue(HMI_data.Background_Color, line, 4, value);
  2148. }
  2149. void onDrawPIntMenu(MenuItemClass* menuitem, int8_t line) {
  2150. const uint16_t value = *(uint16_t*)static_cast<MenuItemPtrClass*>(menuitem)->value;
  2151. onDrawIntMenu(menuitem, line, value);
  2152. }
  2153. void onDrawPInt8Menu(MenuItemClass* menuitem, int8_t line) {
  2154. const uint8_t value = *(uint8_t*)static_cast<MenuItemPtrClass*>(menuitem)->value;
  2155. onDrawIntMenu(menuitem, line, value);
  2156. }
  2157. void onDrawPInt32Menu(MenuItemClass* menuitem, int8_t line) {
  2158. const uint32_t value = *(uint32_t*)static_cast<MenuItemPtrClass*>(menuitem)->value;
  2159. onDrawIntMenu(menuitem, line, value);
  2160. }
  2161. void onDrawFloatMenu(MenuItemClass* menuitem, int8_t line, uint8_t dp, const float value) {
  2162. onDrawMenuItem(menuitem, line);
  2163. DWINUI::Draw_Signed_Float(HMI_data.Text_Color, HMI_data.Background_Color, 3, dp, VALX - dp * DWINUI::Get_font_width(DWIN_FONT_MENU), MBASE(line), value * POW(10, dp));
  2164. }
  2165. void onDrawPFloatMenu(MenuItemClass* menuitem, int8_t line) {
  2166. const float value = *(float*)static_cast<MenuItemPtrClass*>(menuitem)->value;
  2167. const int8_t dp = UNITFDIGITS;
  2168. onDrawFloatMenu(menuitem, line, dp, value);
  2169. }
  2170. void onDrawPFloat2Menu(MenuItemClass* menuitem, int8_t line) {
  2171. const float value = *(float*)static_cast<MenuItemPtrClass*>(menuitem)->value;
  2172. const int8_t dp = 2;
  2173. onDrawFloatMenu(menuitem, line, dp, value);
  2174. }
  2175. void onDrawChkbMenu(MenuItemClass* menuitem, int8_t line, bool checked) {
  2176. onDrawMenuItem(menuitem, line);
  2177. Draw_Chkb_Line(line, checked);
  2178. }
  2179. void onDrawBack(MenuItemClass* menuitem, int8_t line) {
  2180. if (HMI_IsChinese()) menuitem->SetFrame(1, 129, 72, 156, 84);
  2181. onDrawMenuItem(menuitem, line);
  2182. }
  2183. void onDrawTempSubMenu(MenuItemClass* menuitem, int8_t line) {
  2184. if (HMI_IsChinese()) menuitem->SetFrame(1, 57, 104, 84, 116);
  2185. onDrawSubMenu(menuitem, line);
  2186. }
  2187. void onDrawMotionSubMenu(MenuItemClass* menuitem, int8_t line) {
  2188. if (HMI_IsChinese()) menuitem->SetFrame(1, 87, 104, 114, 116);
  2189. onDrawSubMenu(menuitem, line);
  2190. }
  2191. #if ENABLED(EEPROM_SETTINGS)
  2192. void onDrawWriteEeprom(MenuItemClass* menuitem, int8_t line) {
  2193. if (HMI_IsChinese()) menuitem->SetFrame(1, 117, 104, 172, 116);
  2194. onDrawMenuItem(menuitem, line);
  2195. }
  2196. void onDrawReadEeprom(MenuItemClass* menuitem, int8_t line) {
  2197. if (HMI_IsChinese()) menuitem->SetFrame(1, 174, 103, 229, 116);
  2198. onDrawMenuItem(menuitem, line);
  2199. }
  2200. void onDrawResetEeprom(MenuItemClass* menuitem, int8_t line) {
  2201. if (HMI_IsChinese()) menuitem->SetFrame(1, 1, 118, 56, 131);
  2202. onDrawMenuItem(menuitem, line);
  2203. }
  2204. #endif
  2205. void onDrawInfoSubMenu(MenuItemClass* menuitem, int8_t line) {
  2206. if (HMI_IsChinese()) menuitem->SetFrame(1, 231, 104, 258, 116);
  2207. onDrawSubMenu(menuitem, line);
  2208. }
  2209. void onDrawMoveX(MenuItemClass* menuitem, int8_t line) {
  2210. if (HMI_IsChinese()) menuitem->SetFrame(1, 58, 118, 106, 132);
  2211. onDrawPFloatMenu(menuitem, line);
  2212. }
  2213. void onDrawMoveY(MenuItemClass* menuitem, int8_t line) {
  2214. if (HMI_IsChinese()) menuitem->SetFrame(1, 109, 118, 157, 132);
  2215. onDrawPFloatMenu(menuitem, line);
  2216. }
  2217. void onDrawMoveZ(MenuItemClass* menuitem, int8_t line) {
  2218. if (HMI_IsChinese()) menuitem->SetFrame(1, 160, 118, 209, 132);
  2219. onDrawPFloatMenu(menuitem, line);
  2220. }
  2221. #if HAS_HOTEND
  2222. void onDrawMoveE(MenuItemClass* menuitem, int8_t line) {
  2223. if (HMI_IsChinese()) menuitem->SetFrame(1, 212, 118, 253, 131);
  2224. onDrawPFloatMenu(menuitem, line);
  2225. }
  2226. #endif
  2227. void onDrawMoveSubMenu(MenuItemClass* menuitem, int8_t line) {
  2228. if (HMI_IsChinese()) menuitem->SetFrame(1, 159, 70, 200, 84);
  2229. onDrawSubMenu(menuitem, line);
  2230. }
  2231. void onDrawDisableMotors(MenuItemClass* menuitem, int8_t line) {
  2232. if (HMI_IsChinese()) menuitem->SetFrame(1, 204, 70, 259, 82);
  2233. onDrawMenuItem(menuitem, line);
  2234. }
  2235. void onDrawAutoHome(MenuItemClass* menuitem, int8_t line) {
  2236. if (HMI_IsChinese()) menuitem->SetFrame(1, 0, 89, 41, 101);
  2237. onDrawMenuItem(menuitem, line);
  2238. }
  2239. #if HAS_ZOFFSET_ITEM
  2240. #if EITHER(HAS_BED_PROBE, BABYSTEPPING)
  2241. void onDrawZOffset(MenuItemClass* menuitem, int8_t line) {
  2242. if (HMI_IsChinese()) menuitem->SetFrame(1, 174, 164, 223, 177);
  2243. onDrawPFloat2Menu(menuitem, line);
  2244. }
  2245. #else
  2246. void onDrawHomeOffset(MenuItemClass* menuitem, int8_t line) {
  2247. if (HMI_IsChinese()) menuitem->SetFrame(1, 43, 89, 98, 101);
  2248. onDrawMenuItem(menuitem, line);
  2249. }
  2250. #endif
  2251. #endif
  2252. #if HAS_HOTEND
  2253. void onDrawPreheat1(MenuItemClass* menuitem, int8_t line) {
  2254. if (HMI_IsChinese()) menuitem->SetFrame(1, 100, 89, 151, 101);
  2255. onDrawMenuItem(menuitem, line);
  2256. }
  2257. void onDrawPreheat2(MenuItemClass* menuitem, int8_t line) {
  2258. if (HMI_IsChinese()) menuitem->SetFrame(1, 180, 89, 233, 100);
  2259. onDrawMenuItem(menuitem, line);
  2260. }
  2261. #endif
  2262. #if HAS_PREHEAT
  2263. void onDrawCooldown(MenuItemClass* menuitem, int8_t line) {
  2264. if (HMI_IsChinese()) menuitem->SetFrame(1, 1, 104, 56, 117);
  2265. onDrawMenuItem(menuitem, line);
  2266. }
  2267. #endif
  2268. void onDrawLanguage(MenuItemClass* menuitem, int8_t line) {
  2269. if (HMI_IsChinese()) menuitem->SetFrame(1, 239, 134, 266, 146);
  2270. onDrawMenuItem(menuitem, line);
  2271. DWINUI::Draw_String(VALX, MBASE(line), HMI_IsChinese() ? F("CN") : F("EN"));
  2272. }
  2273. void onDrawPwrLossR(MenuItemClass* menuitem, int8_t line) { onDrawChkbMenu(menuitem, line, recovery.enabled); }
  2274. void onDrawEnableSound(MenuItemClass* menuitem, int8_t line) { onDrawChkbMenu(menuitem, line, ui.buzzer_enabled); }
  2275. void onDrawSelColorItem(MenuItemClass* menuitem, int8_t line) {
  2276. const uint16_t color = *(uint16_t*)static_cast<MenuItemPtrClass*>(menuitem)->value;
  2277. DWIN_Draw_Rectangle(0, HMI_data.Highlight_Color, ICOX + 1, MBASE(line) - 1 + 1, ICOX + 18, MBASE(line) - 1 + 18);
  2278. DWIN_Draw_Rectangle(1, color, ICOX + 2, MBASE(line) - 1 + 2, ICOX + 17, MBASE(line) - 1 + 17);
  2279. onDrawMenuItem(menuitem, line);
  2280. }
  2281. void onDrawGetColorItem(MenuItemClass* menuitem, int8_t line) {
  2282. const uint8_t i = menuitem->icon;
  2283. uint16_t color;
  2284. switch (i) {
  2285. case 0: color = RGB(0, 0, 31); break;
  2286. case 1: color = RGB(0, 63, 0); break;
  2287. case 2: color = RGB(31, 0, 0); break;
  2288. default: color = 0; break;
  2289. }
  2290. DWIN_Draw_Rectangle(0, HMI_data.Highlight_Color, ICOX + 1, MBASE(line) - 1 + 1, ICOX + 18, MBASE(line) - 1 + 18);
  2291. DWIN_Draw_Rectangle(1, color, ICOX + 2, MBASE(line) - 1 + 2, ICOX + 17, MBASE(line) - 1 + 17);
  2292. DWINUI::Draw_String(LBLX, MBASE(line) - 1, menuitem->caption);
  2293. Draw_Menu_IntValue(HMI_data.Background_Color, line, 4, HMI_value.Color[i]);
  2294. DWIN_Draw_HLine(HMI_data.SplitLine_Color, 16, MYPOS(line + 1), 240);
  2295. }
  2296. #if HAS_FILAMENT_SENSOR
  2297. void onDrawRunoutEnable(MenuItemClass* menuitem, int8_t line) { onDrawChkbMenu(menuitem, line, runout.enabled); }
  2298. #endif
  2299. void onDrawPIDi(MenuItemClass* menuitem, int8_t line) { onDrawFloatMenu(menuitem, line, 2, unscalePID_i(*(float*)static_cast<MenuItemPtrClass*>(menuitem)->value)); }
  2300. void onDrawPIDd(MenuItemClass* menuitem, int8_t line) { onDrawFloatMenu(menuitem, line, 2, unscalePID_d(*(float*)static_cast<MenuItemPtrClass*>(menuitem)->value)); }
  2301. void onDrawSpeedItem(MenuItemClass* menuitem, int8_t line) {
  2302. if (HMI_IsChinese()) menuitem->SetFrame(1, 116, 164, 171, 176);
  2303. onDrawPIntMenu(menuitem, line);
  2304. }
  2305. #if HAS_HOTEND
  2306. void onDrawHotendTemp(MenuItemClass* menuitem, int8_t line) {
  2307. if (HMI_IsChinese()) menuitem->SetFrame(1, 1, 134, 56, 146);
  2308. onDrawPIntMenu(menuitem, line);
  2309. }
  2310. #endif
  2311. #if HAS_HEATED_BED
  2312. void onDrawBedTemp(MenuItemClass* menuitem, int8_t line) {
  2313. if (HMI_IsChinese()) menuitem->SetFrame(1, 58, 134, 113, 146);
  2314. onDrawPIntMenu(menuitem, line);
  2315. }
  2316. #endif
  2317. #if HAS_FAN
  2318. void onDrawFanSpeed(MenuItemClass* menuitem, int8_t line) {
  2319. if (HMI_IsChinese()) menuitem->SetFrame(1, 115, 134, 170, 146);
  2320. onDrawPInt8Menu(menuitem, line);
  2321. }
  2322. #endif
  2323. void onDrawSpeed(MenuItemClass* menuitem, int8_t line) {
  2324. if (HMI_IsChinese()) menuitem->SetFrame(1, 173, 133, 228, 147);
  2325. onDrawSubMenu(menuitem, line);
  2326. }
  2327. void onDrawAcc(MenuItemClass* menuitem, int8_t line) {
  2328. if (HMI_IsChinese()) {
  2329. menuitem->SetFrame(1, 173, 133, 200, 147);
  2330. DWIN_Frame_AreaCopy(1, 28, 149, 69, 161, LBLX + 27, MBASE(line) + 1); // ...Acceleration
  2331. }
  2332. onDrawSubMenu(menuitem, line);
  2333. }
  2334. #if HAS_CLASSIC_JERK
  2335. void onDrawJerk(MenuItemClass* menuitem, int8_t line) {
  2336. if (HMI_IsChinese()) {
  2337. menuitem->SetFrame(1, 173, 133, 200, 147);
  2338. DWIN_Frame_AreaCopy(1, 1, 180, 28, 192, LBLX + 27, MBASE(line) + 1); // ...
  2339. DWIN_Frame_AreaCopy(1, 202, 133, 228, 147, LBLX + 54, MBASE(line)); // ...Jerk
  2340. }
  2341. onDrawSubMenu(menuitem, line);
  2342. }
  2343. #endif
  2344. void onDrawSteps(MenuItemClass* menuitem, int8_t line) {
  2345. if (HMI_IsChinese()) menuitem->SetFrame(1, 153, 148, 194, 161);
  2346. onDrawSubMenu(menuitem, line);
  2347. }
  2348. #if ENABLED(MESH_BED_LEVELING)
  2349. void onDrawMMeshMoveZ(MenuItemClass* menuitem, int8_t line) {
  2350. if (HMI_IsChinese()) menuitem->SetFrame(1, 160, 118, 209, 132);
  2351. onDrawPFloatMenu(menuitem, line);
  2352. }
  2353. #endif
  2354. #if HAS_PREHEAT
  2355. #if HAS_HOTEND
  2356. void onDrawSetPreheatHotend(MenuItemClass* menuitem, int8_t line) {
  2357. if (HMI_IsChinese()) menuitem->SetFrame(1, 1, 134, 56, 146);
  2358. onDrawPIntMenu(menuitem, line);
  2359. }
  2360. #endif
  2361. #if HAS_HEATED_BED
  2362. void onDrawSetPreheatBed(MenuItemClass* menuitem, int8_t line) {
  2363. if (HMI_IsChinese()) menuitem->SetFrame(1, 58, 134, 113, 146);
  2364. onDrawPIntMenu(menuitem, line);
  2365. }
  2366. #endif
  2367. #if HAS_FAN
  2368. void onDrawSetPreheatFan(MenuItemClass* menuitem, int8_t line) {
  2369. if (HMI_IsChinese()) menuitem->SetFrame(1, 115, 134, 170, 146);
  2370. onDrawPIntMenu(menuitem, line);
  2371. }
  2372. #endif
  2373. void onDrawPLAPreheatSubMenu(MenuItemClass* menuitem, int8_t line) {
  2374. if (HMI_IsChinese()) menuitem->SetFrame(1, 100, 89, 178, 101);
  2375. onDrawSubMenu(menuitem,line);
  2376. }
  2377. void onDrawABSPreheatSubMenu(MenuItemClass* menuitem, int8_t line) {
  2378. if (HMI_IsChinese()) menuitem->SetFrame(1, 180, 89, 260, 100);
  2379. onDrawSubMenu(menuitem,line);
  2380. }
  2381. #endif // HAS_HOTEND
  2382. void onDrawMaxSpeedX(MenuItemClass* menuitem, int8_t line) {
  2383. if (HMI_IsChinese()) {
  2384. menuitem->SetFrame(1, 173, 133, 228, 147);
  2385. DWIN_Frame_AreaCopy(1, 229, 133, 236, 147, LBLX + 58, MBASE(line)); // X
  2386. }
  2387. onDrawPFloatMenu(menuitem, line);
  2388. }
  2389. void onDrawMaxSpeedY(MenuItemClass* menuitem, int8_t line) {
  2390. if (HMI_IsChinese()) {
  2391. menuitem->SetFrame(1, 173, 133, 228, 147);
  2392. DWIN_Frame_AreaCopy(1, 1, 150, 7, 160, LBLX + 58, MBASE(line)); // Y
  2393. }
  2394. onDrawPFloatMenu(menuitem, line);
  2395. }
  2396. void onDrawMaxSpeedZ(MenuItemClass* menuitem, int8_t line) {
  2397. if (HMI_IsChinese()) {
  2398. menuitem->SetFrame(1, 173, 133, 228, 147);
  2399. DWIN_Frame_AreaCopy(1, 9, 150, 16, 160, LBLX + 58, MBASE(line) + 3); // Z
  2400. }
  2401. onDrawPFloatMenu(menuitem, line);
  2402. }
  2403. #if HAS_HOTEND
  2404. void onDrawMaxSpeedE(MenuItemClass* menuitem, int8_t line) {
  2405. if (HMI_IsChinese()) {
  2406. menuitem->SetFrame(1, 173, 133, 228, 147);
  2407. DWIN_Frame_AreaCopy(1, 18, 150, 25, 160, LBLX + 58, MBASE(line)); // E
  2408. }
  2409. onDrawPFloatMenu(menuitem, line);
  2410. }
  2411. #endif
  2412. void onDrawMaxAccelX(MenuItemClass* menuitem, int8_t line) {
  2413. if (HMI_IsChinese()) {
  2414. menuitem->SetFrame (1, 173, 133, 200, 147);
  2415. DWIN_Frame_AreaCopy(1, 28, 149, 69, 161, LBLX + 27, MBASE(line));
  2416. DWIN_Frame_AreaCopy(1, 229, 133, 236, 147, LBLX + 71, MBASE(line)); // X
  2417. }
  2418. onDrawPInt32Menu(menuitem, line);
  2419. }
  2420. void onDrawMaxAccelY(MenuItemClass* menuitem, int8_t line) {
  2421. if (HMI_IsChinese()) {
  2422. menuitem->SetFrame (1, 173, 133, 200, 147);
  2423. DWIN_Frame_AreaCopy(1, 28, 149, 69, 161, LBLX + 27, MBASE(line));
  2424. DWIN_Frame_AreaCopy(1, 1, 150, 7, 160, LBLX + 71, MBASE(line)); // Y
  2425. }
  2426. onDrawPInt32Menu(menuitem, line);
  2427. }
  2428. void onDrawMaxAccelZ(MenuItemClass* menuitem, int8_t line) {
  2429. if (HMI_IsChinese()) {
  2430. menuitem->SetFrame (1, 173, 133, 200, 147);
  2431. DWIN_Frame_AreaCopy(1, 28, 149, 69, 161, LBLX + 27, MBASE(line));
  2432. DWIN_Frame_AreaCopy(1, 9, 150, 16, 160, LBLX + 71, MBASE(line)); // Z
  2433. }
  2434. onDrawPInt32Menu(menuitem, line);
  2435. }
  2436. #if HAS_HOTEND
  2437. void onDrawMaxAccelE(MenuItemClass* menuitem, int8_t line) {
  2438. if (HMI_IsChinese()) {
  2439. menuitem->SetFrame (1, 173, 133, 200, 147);
  2440. DWIN_Frame_AreaCopy(1, 28, 149, 69, 161, LBLX + 27, MBASE(line));
  2441. DWIN_Frame_AreaCopy(1, 18, 150, 25, 160, LBLX + 71, MBASE(line)); // E
  2442. }
  2443. onDrawPInt32Menu(menuitem, line);
  2444. }
  2445. #endif
  2446. #if HAS_CLASSIC_JERK
  2447. void onDrawMaxJerkX(MenuItemClass* menuitem, int8_t line) {
  2448. if (HMI_IsChinese()) {
  2449. menuitem->SetFrame (1, 173, 133, 200, 147);
  2450. DWIN_Frame_AreaCopy(1, 1, 180, 28, 192, LBLX + 27, MBASE(line));
  2451. DWIN_Frame_AreaCopy(1, 202, 133, 228, 147, LBLX + 53, MBASE(line));
  2452. DWIN_Frame_AreaCopy(1, 229, 133, 236, 147, LBLX + 83, MBASE(line));
  2453. }
  2454. onDrawPFloatMenu(menuitem, line);
  2455. }
  2456. void onDrawMaxJerkY(MenuItemClass* menuitem, int8_t line) {
  2457. if (HMI_IsChinese()) {
  2458. menuitem->SetFrame (1, 173, 133, 200, 147);
  2459. DWIN_Frame_AreaCopy(1, 1, 180, 28, 192, LBLX + 27, MBASE(line));
  2460. DWIN_Frame_AreaCopy(1, 202, 133, 228, 147, LBLX + 53, MBASE(line));
  2461. DWIN_Frame_AreaCopy(1, 1, 150, 7, 160, LBLX + 83, MBASE(line));
  2462. }
  2463. onDrawPFloatMenu(menuitem, line);
  2464. }
  2465. void onDrawMaxJerkZ(MenuItemClass* menuitem, int8_t line) {
  2466. if (HMI_IsChinese()) {
  2467. menuitem->SetFrame (1, 173, 133, 200, 147);
  2468. DWIN_Frame_AreaCopy(1, 1, 180, 28, 192, LBLX + 27, MBASE(line));
  2469. DWIN_Frame_AreaCopy(1, 202, 133, 228, 147, LBLX + 53, MBASE(line));
  2470. DWIN_Frame_AreaCopy(1, 9, 150, 16, 160, LBLX + 83, MBASE(line));
  2471. }
  2472. onDrawPFloatMenu(menuitem, line);
  2473. }
  2474. #if HAS_HOTEND
  2475. void onDrawMaxJerkE(MenuItemClass* menuitem, int8_t line) {
  2476. if (HMI_IsChinese()) {
  2477. menuitem->SetFrame (1, 173, 133, 200, 147);
  2478. DWIN_Frame_AreaCopy(1, 1, 180, 28, 192, LBLX + 27, MBASE(line));
  2479. DWIN_Frame_AreaCopy(1, 202, 133, 228, 147, LBLX + 53, MBASE(line));
  2480. DWIN_Frame_AreaCopy(1, 18, 150, 25, 160, LBLX + 83, MBASE(line));
  2481. }
  2482. onDrawPFloatMenu(menuitem, line);
  2483. }
  2484. #endif
  2485. #endif // HAS_CLASSIC_JERK
  2486. void onDrawStepsX(MenuItemClass* menuitem, int8_t line) {
  2487. if (HMI_IsChinese()) {
  2488. menuitem->SetFrame (1, 153, 148, 194, 161);
  2489. DWIN_Frame_AreaCopy(1, 229, 133, 236, 147, LBLX + 44, MBASE(line)); // X
  2490. }
  2491. onDrawPFloatMenu(menuitem, line);
  2492. }
  2493. void onDrawStepsY(MenuItemClass* menuitem, int8_t line) {
  2494. if (HMI_IsChinese()) {
  2495. menuitem->SetFrame (1, 153, 148, 194, 161);
  2496. DWIN_Frame_AreaCopy(1, 1, 150, 7, 160, LBLX + 44, MBASE(line)); // Y
  2497. }
  2498. onDrawPFloatMenu(menuitem, line);
  2499. }
  2500. void onDrawStepsZ(MenuItemClass* menuitem, int8_t line) {
  2501. if (HMI_IsChinese()) {
  2502. menuitem->SetFrame (1, 153, 148, 194, 161);
  2503. DWIN_Frame_AreaCopy(1, 9, 150, 16, 160, LBLX + 44, MBASE(line)); // Z
  2504. }
  2505. onDrawPFloatMenu(menuitem, line);
  2506. }
  2507. #if HAS_HOTEND
  2508. void onDrawStepsE(MenuItemClass* menuitem, int8_t line) {
  2509. if (HMI_IsChinese()) {
  2510. menuitem->SetFrame (1, 153, 148, 194, 161);
  2511. DWIN_Frame_AreaCopy(1, 18, 150, 25, 160, LBLX + 44, MBASE(line)); // E
  2512. }
  2513. onDrawPFloatMenu(menuitem, line);
  2514. }
  2515. #endif
  2516. // HMI Control functions ======================================================
  2517. // Generic menu control using the encoder
  2518. void HMI_Menu() {
  2519. ENCODER_DiffState encoder_diffState = get_encoder_state();
  2520. if (encoder_diffState == ENCODER_DIFF_NO) return;
  2521. if (encoder_diffState == ENCODER_DIFF_ENTER) {
  2522. if (CurrentMenu != nullptr) CurrentMenu->onClick();
  2523. } else if (CurrentMenu != nullptr) CurrentMenu->onScroll(encoder_diffState == ENCODER_DIFF_CW);
  2524. }
  2525. // Get an integer value using the encoder without draw anything
  2526. // lo: low limit
  2527. // hi: high limit
  2528. // Return value:
  2529. // 0 : no change
  2530. // 1 : live change
  2531. // 2 : apply change
  2532. int8_t HMI_GetIntNoDraw(const int32_t lo, const int32_t hi) {
  2533. ENCODER_DiffState encoder_diffState = Encoder_ReceiveAnalyze();
  2534. if (encoder_diffState != ENCODER_DIFF_NO) {
  2535. if (Apply_Encoder(encoder_diffState, HMI_value.Value)) {
  2536. EncoderRate.enabled = false;
  2537. checkkey = last_checkkey;
  2538. return 2;
  2539. }
  2540. LIMIT(HMI_value.Value, lo, hi);
  2541. return 1;
  2542. }
  2543. return 0;
  2544. }
  2545. // Get an integer value using the encoder
  2546. // lo: low limit
  2547. // hi: high limit
  2548. // Return value:
  2549. // 0 : no change
  2550. // 1 : live change
  2551. // 2 : apply change
  2552. int8_t HMI_GetInt(const int32_t lo, const int32_t hi) {
  2553. ENCODER_DiffState encoder_diffState = Encoder_ReceiveAnalyze();
  2554. if (encoder_diffState != ENCODER_DIFF_NO) {
  2555. if (Apply_Encoder(encoder_diffState, HMI_value.Value)) {
  2556. EncoderRate.enabled = false;
  2557. DWINUI::Draw_Int(HMI_data.Text_Color, HMI_data.Background_Color, 4 , VALX, MBASE(CurrentMenu->line()) - 1, HMI_value.Value);
  2558. checkkey = last_checkkey;
  2559. return 2;
  2560. }
  2561. LIMIT(HMI_value.Value, lo, hi);
  2562. DWINUI::Draw_Int(HMI_data.Text_Color, HMI_data.Selected_Color, 4 , VALX, MBASE(CurrentMenu->line()) - 1, HMI_value.Value);
  2563. return 1;
  2564. }
  2565. return 0;
  2566. }
  2567. // Set an integer using the encoder
  2568. void HMI_SetInt() {
  2569. int8_t val = HMI_GetInt(HMI_value.MinValue, HMI_value.MaxValue);
  2570. switch (val) {
  2571. case 0: return; break;
  2572. case 1: if (HMI_value.LiveUpdate != nullptr) HMI_value.LiveUpdate(); break;
  2573. case 2: if (HMI_value.Apply != nullptr) HMI_value.Apply(); break;
  2574. }
  2575. }
  2576. // Set an integer without drawing
  2577. void HMI_SetIntNoDraw() {
  2578. int8_t val = HMI_GetIntNoDraw(HMI_value.MinValue, HMI_value.MaxValue);
  2579. switch (val) {
  2580. case 0: return; break;
  2581. case 1: if (HMI_value.LiveUpdate != nullptr) HMI_value.LiveUpdate(); break;
  2582. case 2: if (HMI_value.Apply != nullptr) HMI_value.Apply(); break;
  2583. }
  2584. }
  2585. // Set an integer pointer variable using the encoder
  2586. void HMI_SetPInt() {
  2587. int8_t val = HMI_GetInt(HMI_value.MinValue, HMI_value.MaxValue);
  2588. if (!val) return;
  2589. else if (val == 2) { // Apply
  2590. *HMI_value.P_Int = HMI_value.Value;
  2591. if (HMI_value.Apply != nullptr) HMI_value.Apply();
  2592. } else if (HMI_value.LiveUpdate != nullptr) HMI_value.LiveUpdate();
  2593. }
  2594. // Get an scaled float value using the encoder
  2595. // dp: decimal places
  2596. // lo: scaled low limit
  2597. // hi: scaled high limit
  2598. // Return value:
  2599. // 0 : no change
  2600. // 1 : live change
  2601. // 2 : apply change
  2602. int8_t HMI_GetFloat(uint8_t dp, int32_t lo, int32_t hi) {
  2603. ENCODER_DiffState encoder_diffState = Encoder_ReceiveAnalyze();
  2604. if (encoder_diffState != ENCODER_DIFF_NO) {
  2605. if (Apply_Encoder(encoder_diffState, HMI_value.Value)) {
  2606. EncoderRate.enabled = false;
  2607. DWINUI::Draw_Signed_Float(HMI_data.Text_Color, HMI_data.Background_Color, 3, dp, VALX - dp * DWINUI::Get_font_width(DWIN_FONT_MENU), MBASE(CurrentMenu->line()), HMI_value.Value);
  2608. checkkey = last_checkkey;
  2609. return 2;
  2610. }
  2611. LIMIT(HMI_value.Value, lo, hi);
  2612. DWINUI::Draw_Signed_Float(HMI_data.Text_Color, HMI_data.Selected_Color, 3, dp, VALX - dp * DWINUI::Get_font_width(DWIN_FONT_MENU), MBASE(CurrentMenu->line()), HMI_value.Value);
  2613. return 1;
  2614. }
  2615. return 0;
  2616. }
  2617. // Set an scaled float using the encoder
  2618. void HMI_SetFloat() {
  2619. int8_t val = HMI_GetFloat(HMI_value.dp, HMI_value.MinValue, HMI_value.MaxValue);
  2620. switch (val) {
  2621. case 0: return; break;
  2622. case 1: if (HMI_value.LiveUpdate != nullptr) HMI_value.LiveUpdate(); break;
  2623. case 2: if (HMI_value.Apply != nullptr) HMI_value.Apply(); break;
  2624. }
  2625. }
  2626. // Set an scaled float pointer variable using the encoder
  2627. void HMI_SetPFloat() {
  2628. int8_t val = HMI_GetFloat(HMI_value.dp, HMI_value.MinValue, HMI_value.MaxValue);
  2629. if (!val) return;
  2630. else if (val == 2) { // Apply
  2631. *HMI_value.P_Float = HMI_value.Value / POW(10, HMI_value.dp);
  2632. if (HMI_value.Apply != nullptr) HMI_value.Apply();
  2633. } else if (HMI_value.LiveUpdate != nullptr) HMI_value.LiveUpdate();
  2634. }
  2635. // Menu Creation and Drawing functions ======================================================
  2636. void SetMenuTitle(frame_rect_t cn, frame_rect_t en, const __FlashStringHelper* text) {
  2637. if (HMI_IsChinese() && (cn.w != 0))
  2638. CurrentMenu->MenuTitle.SetFrame(cn.x, cn.y, cn.w, cn.h);
  2639. else {
  2640. #ifdef USE_STRING_HEADINGS
  2641. CurrentMenu->MenuTitle.SetCaption(text);
  2642. #else
  2643. if (en.w != 0) CurrentMenu->MenuTitle.SetFrame(en.x, en.y, en.w, en.h);
  2644. #endif
  2645. }
  2646. }
  2647. void Draw_Prepare_Menu() {
  2648. checkkey = Menu;
  2649. if (PrepareMenu == nullptr) PrepareMenu = new MenuClass();
  2650. if (CurrentMenu != PrepareMenu) {
  2651. CurrentMenu = PrepareMenu;
  2652. SetMenuTitle({133, 1, 28, 13}, {179, 0, 48, 14}, GET_TEXT_F(MSG_PREPARE));
  2653. DWINUI::MenuItemsPrepare(13);
  2654. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Goto_Main_Menu);
  2655. TERN_(ADVANCED_PAUSE_FEATURE, ADDMENUITEM(ICON_FilMan, GET_TEXT(MSG_FILAMENT_MAN), onDrawSubMenu, Draw_FilamentMan_Menu));
  2656. ADDMENUITEM(ICON_Axis, GET_TEXT(MSG_MOVE_AXIS), onDrawMoveSubMenu, Draw_Move_Menu);
  2657. ADDMENUITEM(ICON_LevBed, GET_TEXT(MSG_BED_LEVELING), onDrawSubMenu, Draw_LevBedCorners_Menu);
  2658. ADDMENUITEM(ICON_CloseMotor, GET_TEXT(MSG_DISABLE_STEPPERS), onDrawDisableMotors, DisableMotors);
  2659. ADDMENUITEM(ICON_Homing, GET_TEXT(MSG_AUTO_HOME), onDrawAutoHome, AutoHome);
  2660. TERN_(MESH_BED_LEVELING, ADDMENUITEM(ICON_ManualMesh, GET_TEXT(MSG_MANUAL_MESH), onDrawSubMenu, Draw_ManualMesh_Menu));
  2661. #if HAS_ZOFFSET_ITEM
  2662. #if EITHER(HAS_BED_PROBE, BABYSTEPPING)
  2663. ADDMENUITEM(ICON_SetZOffset, GET_TEXT(MSG_PROBE_WIZARD), onDrawSubMenu, Draw_ZOffsetWiz_Menu);
  2664. #else
  2665. ADDMENUITEM(ICON_SetHome, GET_TEXT(MSG_SET_HOME_OFFSETS), onDrawHomeOffset, SetHome);
  2666. #endif
  2667. #endif
  2668. #if HAS_HOTEND
  2669. ADDMENUITEM(ICON_PLAPreheat, GET_TEXT(MSG_PREHEAT_1), onDrawPreheat1, SetPreheat0);
  2670. ADDMENUITEM(ICON_ABSPreheat, PSTR("Preheat " PREHEAT_2_LABEL), onDrawPreheat2, SetPreheat1);
  2671. ADDMENUITEM(ICON_CustomPreheat, GET_TEXT(MSG_PREHEAT_CUSTOM), onDrawMenuItem, SetPreheat2);
  2672. #endif
  2673. TERN_(HAS_PREHEAT, ADDMENUITEM(ICON_Cool, GET_TEXT(MSG_COOLDOWN), onDrawCooldown, SetCoolDown));
  2674. ADDMENUITEM(ICON_Language, PSTR("UI Language"), onDrawLanguage, SetLanguage);
  2675. }
  2676. CurrentMenu->Draw();
  2677. }
  2678. void Draw_LevBedCorners_Menu() {
  2679. DWINUI::ClearMenuArea();
  2680. checkkey = Menu;
  2681. if (LevBedMenu == nullptr) LevBedMenu = new MenuClass();
  2682. if (CurrentMenu != LevBedMenu) {
  2683. CurrentMenu = LevBedMenu;
  2684. SetMenuTitle({0}, {0}, GET_TEXT_F(MSG_BED_TRAMMING)); // TODO: Chinese, English "Bed Tramming" JPG
  2685. DWINUI::MenuItemsPrepare(6);
  2686. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Draw_Prepare_Menu);
  2687. ADDMENUITEM(ICON_Axis, GET_TEXT(MSG_LEVBED_FL), onDrawMenuItem, LevBedFL);
  2688. ADDMENUITEM(ICON_Axis, GET_TEXT(MSG_LEVBED_FR), onDrawMenuItem, LevBedFR);
  2689. ADDMENUITEM(ICON_Axis, GET_TEXT(MSG_LEVBED_BR), onDrawMenuItem, LevBedBR);
  2690. ADDMENUITEM(ICON_Axis, GET_TEXT(MSG_LEVBED_BL), onDrawMenuItem, LevBedBL);
  2691. ADDMENUITEM(ICON_Axis, GET_TEXT(MSG_LEVBED_C ), onDrawMenuItem, LevBedC );
  2692. }
  2693. CurrentMenu->Draw();
  2694. }
  2695. void Draw_Control_Menu() {
  2696. checkkey = Menu;
  2697. if (ControlMenu == nullptr) ControlMenu = new MenuClass();
  2698. if (CurrentMenu != ControlMenu) {
  2699. CurrentMenu = ControlMenu;
  2700. SetMenuTitle({103, 1, 28, 14}, {128, 2, 49, 11}, GET_TEXT_F(MSG_CONTROL));
  2701. DWINUI::MenuItemsPrepare(9);
  2702. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Goto_Main_Menu);
  2703. ADDMENUITEM(ICON_Temperature, GET_TEXT(MSG_TEMPERATURE), onDrawTempSubMenu, Draw_Temperature_Menu);
  2704. ADDMENUITEM(ICON_Motion, GET_TEXT(MSG_MOTION), onDrawMotionSubMenu, Draw_Motion_Menu);
  2705. #if ENABLED(EEPROM_SETTINGS)
  2706. ADDMENUITEM(ICON_WriteEEPROM, GET_TEXT(MSG_STORE_EEPROM), onDrawWriteEeprom, WriteEeprom);
  2707. ADDMENUITEM(ICON_ReadEEPROM, GET_TEXT(MSG_LOAD_EEPROM), onDrawReadEeprom, ReadEeprom);
  2708. ADDMENUITEM(ICON_ResumeEEPROM, GET_TEXT(MSG_RESTORE_DEFAULTS), onDrawResetEeprom, ResetEeprom);
  2709. #endif
  2710. ADDMENUITEM(ICON_Reboot, GET_TEXT(MSG_RESET_PRINTER), onDrawMenuItem, RebootPrinter);
  2711. ADDMENUITEM(ICON_AdvSet, GET_TEXT(MSG_ADVANCED_SETTINGS), onDrawSubMenu, Draw_AdvancedSettings_Menu);
  2712. ADDMENUITEM(ICON_Info, GET_TEXT(MSG_INFO_SCREEN), onDrawInfoSubMenu, Goto_InfoMenu);
  2713. }
  2714. CurrentMenu->Draw();
  2715. }
  2716. void Draw_AdvancedSettings_Menu() {
  2717. checkkey = Menu;
  2718. if (AdvancedSettings == nullptr) AdvancedSettings = new MenuClass();
  2719. if (CurrentMenu != AdvancedSettings) {
  2720. CurrentMenu = AdvancedSettings;
  2721. SetMenuTitle({0}, {0}, GET_TEXT_F(MSG_ADVANCED_SETTINGS)); // TODO: Chinese, English "Advanced Settings" JPG
  2722. DWINUI::MenuItemsPrepare(11);
  2723. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Draw_Control_Menu);
  2724. TERN_(HAS_HOME_OFFSET, ADDMENUITEM(ICON_HomeOffset, GET_TEXT(MSG_SET_HOME_OFFSETS), onDrawSubMenu, Draw_HomeOffset_Menu));
  2725. TERN_(HAS_BED_PROBE, ADDMENUITEM(ICON_ProbeSet, GET_TEXT(MSG_ZPROBE_SETTINGS), onDrawSubMenu, Draw_ProbeSet_Menu));
  2726. TERN_(HAS_HOTEND, ADDMENUITEM(ICON_PIDNozzle, F("Hotend PID Settings"), onDrawSubMenu, Draw_HotendPID_Menu));
  2727. TERN_(HAS_HEATED_BED, ADDMENUITEM(ICON_PIDbed, F("Bed PID Settings"), onDrawSubMenu, Draw_BedPID_Menu));
  2728. TERN_(HAS_FILAMENT_SENSOR, ADDMENUITEM(ICON_FilSet, GET_TEXT(MSG_FILAMENT_SET), onDrawSubMenu, Draw_FilSet_Menu));
  2729. TERN_(POWER_LOSS_RECOVERY, ADDMENUITEM(ICON_Pwrlossr, F("Power-loss recovery"), onDrawPwrLossR, SetPwrLossr));
  2730. TERN_(HAS_LCD_BRIGHTNESS, ADDMENUITEM_P(ICON_Brightness, F("LCD Brightness"), onDrawPInt8Menu, SetBrightness, &ui.brightness));
  2731. ADDMENUITEM(ICON_Scolor, F("Select Colors"), onDrawSubMenu, Draw_SelectColors_Menu);
  2732. TERN_(SOUND_MENU_ITEM, ADDMENUITEM(ICON_Sound, F("Enable Sound"), onDrawEnableSound, SetEnableSound));
  2733. ADDMENUITEM(ICON_Lock, F("Lock Screen"), onDrawMenuItem, Goto_LockScreen);
  2734. }
  2735. CurrentMenu->Draw();
  2736. }
  2737. void Draw_Move_Menu() {
  2738. checkkey = Menu;
  2739. if (MoveMenu == nullptr) MoveMenu = new MenuClass();
  2740. if (CurrentMenu != MoveMenu) {
  2741. CurrentMenu = MoveMenu;
  2742. SetMenuTitle({192, 1, 42, 14}, {231, 2, 35, 11}, GET_TEXT_F(MSG_MOVE_AXIS));
  2743. DWINUI::MenuItemsPrepare(5);
  2744. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Draw_Prepare_Menu);
  2745. ADDMENUITEM_P(ICON_MoveX, GET_TEXT(MSG_MOVE_X), onDrawMoveX, SetMoveX, &current_position.x);
  2746. ADDMENUITEM_P(ICON_MoveY, GET_TEXT(MSG_MOVE_Y), onDrawMoveY, SetMoveY, &current_position.y);
  2747. ADDMENUITEM_P(ICON_MoveZ, GET_TEXT(MSG_MOVE_Z), onDrawMoveZ, SetMoveZ, &current_position.z);
  2748. TERN_(HAS_HOTEND, ADDMENUITEM_P(ICON_Extruder, GET_TEXT(MSG_MOVE_E), onDrawMoveE, SetMoveE, &current_position.e));
  2749. }
  2750. CurrentMenu->Draw();
  2751. if (!all_axes_trusted()) ui.set_status_P(PSTR("WARNING: position is unknow"));
  2752. }
  2753. #if HAS_HOME_OFFSET
  2754. void Draw_HomeOffset_Menu() {
  2755. checkkey = Menu;
  2756. if (HomeOffMenu == nullptr) HomeOffMenu = new MenuClass();
  2757. if (CurrentMenu != HomeOffMenu) {
  2758. CurrentMenu = HomeOffMenu;
  2759. SetMenuTitle({0}, {0}, GET_TEXT_F(MSG_SET_HOME_OFFSETS)); // TODO: Chinese, English "Set Home Offsets" JPG
  2760. DWINUI::MenuItemsPrepare(4);
  2761. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Draw_AdvancedSettings_Menu);
  2762. ADDMENUITEM_P(ICON_HomeOffsetX, GET_TEXT(MSG_HOME_OFFSET_X), onDrawPFloatMenu, SetHomeOffsetX, &home_offset[X_AXIS]);
  2763. ADDMENUITEM_P(ICON_HomeOffsetY, GET_TEXT(MSG_HOME_OFFSET_Y), onDrawPFloatMenu, SetHomeOffsetY, &home_offset[Y_AXIS]);
  2764. ADDMENUITEM_P(ICON_HomeOffsetZ, GET_TEXT(MSG_HOME_OFFSET_Z), onDrawPFloatMenu, SetHomeOffsetZ, &home_offset[Z_AXIS]);
  2765. }
  2766. CurrentMenu->Draw();
  2767. }
  2768. #endif
  2769. #if HAS_BED_PROBE
  2770. void Draw_ProbeSet_Menu() {
  2771. checkkey = Menu;
  2772. if (ProbeSetMenu == nullptr) ProbeSetMenu = new MenuClass();
  2773. if (CurrentMenu != ProbeSetMenu) {
  2774. CurrentMenu = ProbeSetMenu;
  2775. SetMenuTitle({0}, {0}, GET_TEXT_F(MSG_ZPROBE_SETTINGS)); // TODO: Chinese, English "Probe Settings" JPG
  2776. DWINUI::MenuItemsPrepare(4);
  2777. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Draw_AdvancedSettings_Menu);
  2778. ADDMENUITEM_P(ICON_ProbeOffsetX, GET_TEXT(MSG_ZPROBE_XOFFSET), onDrawPFloatMenu, SetProbeOffsetX, &probe.offset.x);
  2779. ADDMENUITEM_P(ICON_ProbeOffsetY, GET_TEXT(MSG_ZPROBE_YOFFSET), onDrawPFloatMenu, SetProbeOffsetY, &probe.offset.y);
  2780. ADDMENUITEM(ICON_ProbeTest, GET_TEXT(MSG_M48_TEST), onDrawMenuItem, ProbeTest);
  2781. }
  2782. CurrentMenu->Draw();
  2783. }
  2784. #endif
  2785. #if HAS_FILAMENT_SENSOR
  2786. void Draw_FilSet_Menu() {
  2787. checkkey = Menu;
  2788. if (FilSetMenu == nullptr) FilSetMenu = new MenuClass();
  2789. if (CurrentMenu != FilSetMenu) {
  2790. CurrentMenu = FilSetMenu;
  2791. CurrentMenu->MenuTitle.SetCaption(GET_TEXT_F(MSG_FILAMENT_SET));
  2792. DWINUI::MenuItemsPrepare(6);
  2793. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawMenuItem, Draw_AdvancedSettings_Menu);
  2794. TERN_(HAS_FILAMENT_SENSOR, ADDMENUITEM(ICON_Runout, GET_TEXT(MSG_RUNOUT_ENABLE), onDrawRunoutEnable, SetRunoutEnable));
  2795. TERN_(HAS_FILAMENT_RUNOUT_DISTANCE, ADDMENUITEM_P(ICON_Runout, F("Runout Distance"), onDrawPFloatMenu, SetRunoutDistance, &runout.runout_distance()));
  2796. TERN_(PREVENT_COLD_EXTRUSION, ADDMENUITEM_P(ICON_ExtrudeMinT, F("Extrude Min Temp."), onDrawPIntMenu, SetExtMinT, &thermalManager.extrude_min_temp));
  2797. TERN_(ADVANCED_PAUSE_FEATURE, ADDMENUITEM_P(ICON_FilLoad, GET_TEXT(MSG_FILAMENT_LOAD), onDrawPFloatMenu, SetFilLoad, &fc_settings[0].load_length));
  2798. TERN_(ADVANCED_PAUSE_FEATURE, ADDMENUITEM_P(ICON_FilUnload, GET_TEXT(MSG_FILAMENT_UNLOAD), onDrawPFloatMenu, SetFilUnload, &fc_settings[0].unload_length));
  2799. }
  2800. CurrentMenu->Draw();
  2801. }
  2802. #endif
  2803. void Draw_SelectColors_Menu() {
  2804. checkkey = Menu;
  2805. if (SelectColorMenu == nullptr) SelectColorMenu = new MenuClass();
  2806. if (CurrentMenu != SelectColorMenu) {
  2807. CurrentMenu = SelectColorMenu;
  2808. SetMenuTitle({0}, {0}, F("Select Colors")); // TODO: Chinese, English "Select Color" JPG
  2809. DWINUI::MenuItemsPrepare(20);
  2810. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Draw_AdvancedSettings_Menu);
  2811. ADDMENUITEM(ICON_StockConfiguration, GET_TEXT(MSG_RESTORE_DEFAULTS), onDrawMenuItem, RestoreDefaultsColors);
  2812. ADDMENUITEM_P(0, "Screen Background", onDrawSelColorItem, SelColor, &HMI_data.Background_Color);
  2813. ADDMENUITEM_P(0, "Cursor", onDrawSelColorItem, SelColor, &HMI_data.Cursor_color);
  2814. ADDMENUITEM_P(0, "Title Background", onDrawSelColorItem, SelColor, &HMI_data.TitleBg_color);
  2815. ADDMENUITEM_P(0, "Title Text", onDrawSelColorItem, SelColor, &HMI_data.TitleTxt_color);
  2816. ADDMENUITEM_P(0, "Text", onDrawSelColorItem, SelColor, &HMI_data.Text_Color);
  2817. ADDMENUITEM_P(0, "Selected", onDrawSelColorItem, SelColor, &HMI_data.Selected_Color);
  2818. ADDMENUITEM_P(0, "Split Line", onDrawSelColorItem, SelColor, &HMI_data.SplitLine_Color);
  2819. ADDMENUITEM_P(0, "Highlight", onDrawSelColorItem, SelColor, &HMI_data.Highlight_Color);
  2820. ADDMENUITEM_P(0, "Status Background", onDrawSelColorItem, SelColor, &HMI_data.StatusBg_Color);
  2821. ADDMENUITEM_P(0, "Status Text", onDrawSelColorItem, SelColor, &HMI_data.StatusTxt_Color);
  2822. ADDMENUITEM_P(0, "Popup Background", onDrawSelColorItem, SelColor, &HMI_data.PopupBg_color);
  2823. ADDMENUITEM_P(0, "Popup Text", onDrawSelColorItem, SelColor, &HMI_data.PopupTxt_Color);
  2824. ADDMENUITEM_P(0, "Alert Background", onDrawSelColorItem, SelColor, &HMI_data.AlertBg_Color);
  2825. ADDMENUITEM_P(0, "Alert Text", onDrawSelColorItem, SelColor, &HMI_data.AlertTxt_Color);
  2826. ADDMENUITEM_P(0, "Percent Text", onDrawSelColorItem, SelColor, &HMI_data.PercentTxt_Color);
  2827. ADDMENUITEM_P(0, "Bar Fill", onDrawSelColorItem, SelColor, &HMI_data.Barfill_Color);
  2828. ADDMENUITEM_P(0, "Indicator value", onDrawSelColorItem, SelColor, &HMI_data.Indicator_Color);
  2829. ADDMENUITEM_P(0, "Coordinate value", onDrawSelColorItem, SelColor, &HMI_data.Coordinate_Color);
  2830. }
  2831. CurrentMenu->Draw();
  2832. }
  2833. void Draw_GetColor_Menu() {
  2834. checkkey = Menu;
  2835. if (GetColorMenu == nullptr) GetColorMenu = new MenuClass();
  2836. if (CurrentMenu != GetColorMenu) {
  2837. CurrentMenu = GetColorMenu;
  2838. SetMenuTitle({0}, {0}, F("Get Color")); // TODO: Chinese, English "Get Color" JPG
  2839. DWINUI::MenuItemsPrepare(5);
  2840. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, DWIN_ApplyColor);
  2841. ADDMENUITEM(ICON_Cancel, GET_TEXT(MSG_BUTTON_CANCEL), onDrawMenuItem, Draw_SelectColors_Menu);
  2842. ADDMENUITEM(0, "Blue", onDrawGetColorItem, SetRGBColor);
  2843. ADDMENUITEM(1, "Green", onDrawGetColorItem, SetRGBColor);
  2844. ADDMENUITEM(2, "Red", onDrawGetColorItem, SetRGBColor);
  2845. }
  2846. CurrentMenu->Draw();
  2847. DWIN_Draw_Rectangle(1, *HMI_value.P_Int, 20, 315, DWIN_WIDTH - 20, 335);
  2848. }
  2849. void Draw_Tune_Menu() {
  2850. checkkey = Menu;
  2851. if (TuneMenu == nullptr) TuneMenu = new MenuClass();
  2852. if (CurrentMenu != TuneMenu) {
  2853. CurrentMenu = TuneMenu;
  2854. SetMenuTitle({73, 2, 28, 12}, {94, 2, 33, 11}, GET_TEXT_F(MSG_TUNE)); // TODO: Chinese, English "Tune" JPG
  2855. DWINUI::MenuItemsPrepare(10);
  2856. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Goto_PrintProcess);
  2857. ADDMENUITEM_P(ICON_Speed, GET_TEXT(MSG_SPEED), onDrawSpeedItem, SetSpeed, &feedrate_percentage);
  2858. TERN_(HAS_HOTEND, HotendTargetItem = ADDMENUITEM_P(ICON_HotendTemp, GET_TEXT(MSG_UBL_SET_TEMP_HOTEND), onDrawHotendTemp, SetHotendTemp, &thermalManager.temp_hotend[0].target));
  2859. TERN_(HAS_HEATED_BED, BedTargetItem = ADDMENUITEM_P(ICON_BedTemp, GET_TEXT(MSG_UBL_SET_TEMP_BED), onDrawBedTemp, SetBedTemp, &thermalManager.temp_bed.target));
  2860. TERN_(HAS_FAN, FanSpeedItem = ADDMENUITEM_P(ICON_FanSpeed, GET_TEXT(MSG_FAN_SPEED), onDrawFanSpeed, SetFanSpeed, &thermalManager.fan_speed[0]));
  2861. #if HAS_ZOFFSET_ITEM
  2862. #if EITHER(HAS_BED_PROBE, BABYSTEPPING)
  2863. ADDMENUITEM_P(ICON_Zoffset, GET_TEXT(MSG_ZPROBE_ZOFFSET), onDrawZOffset, SetZOffset, &BABY_Z_VAR);
  2864. #else
  2865. ADDMENUITEM(ICON_SetHome, GET_TEXT(MSG_SET_HOME_OFFSETS), onDrawHomeOffset, SetHome);
  2866. #endif
  2867. #endif
  2868. ADDMENUITEM_P(ICON_Flow, GET_TEXT(MSG_FLOW), onDrawPIntMenu, SetFlow, &planner.flow_percentage[0]);
  2869. TERN_(ADVANCED_PAUSE_FEATURE, ADDMENUITEM(ICON_FilMan, GET_TEXT(MSG_FILAMENTCHANGE), onDrawMenuItem, ChangeFilament));
  2870. ADDMENUITEM(ICON_Lock, PSTR("Lock Screen"), onDrawMenuItem, Goto_LockScreen);
  2871. TERN_(HAS_LCD_BRIGHTNESS, ADDMENUITEM_P(ICON_Brightness, F("LCD Brightness"), onDrawPInt8Menu, SetBrightness, &ui.brightness));
  2872. }
  2873. CurrentMenu->Draw();
  2874. }
  2875. void Draw_Motion_Menu() {
  2876. checkkey = Menu;
  2877. if (MotionMenu == nullptr) MotionMenu = new MenuClass();
  2878. if (CurrentMenu != MotionMenu) {
  2879. CurrentMenu = MotionMenu;
  2880. SetMenuTitle({1, 16, 28, 13}, {144, 16, 46, 11}, GET_TEXT_F(MSG_MOTION)); // TODO: Chinese, English "Motion" JPG
  2881. DWINUI::MenuItemsPrepare(6);
  2882. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Draw_Control_Menu);
  2883. ADDMENUITEM(ICON_MaxSpeed, GET_TEXT(MSG_SPEED), onDrawSpeed, Draw_MaxSpeed_Menu);
  2884. ADDMENUITEM(ICON_MaxAccelerated, GET_TEXT(MSG_ACCELERATION), onDrawAcc, Draw_MaxAccel_Menu);
  2885. TERN_(HAS_CLASSIC_JERK, ADDMENUITEM(ICON_MaxJerk, GET_TEXT(MSG_JERK), onDrawJerk, Draw_MaxJerk_Menu));
  2886. ADDMENUITEM(ICON_Step, GET_TEXT(MSG_STEPS_PER_MM), onDrawSteps, Draw_Steps_Menu);
  2887. ADDMENUITEM_P(ICON_Flow, GET_TEXT(MSG_FLOW), onDrawPIntMenu, SetFlow, &planner.flow_percentage[0]);
  2888. }
  2889. CurrentMenu->Draw();
  2890. DWIN_StatusChanged(nullptr);
  2891. }
  2892. #if ENABLED(ADVANCED_PAUSE_FEATURE)
  2893. void Draw_FilamentMan_Menu() {
  2894. checkkey = Menu;
  2895. if (FilamentMenu == nullptr) FilamentMenu = new MenuClass();
  2896. if (CurrentMenu != FilamentMenu) {
  2897. CurrentMenu = FilamentMenu;
  2898. SetMenuTitle({0}, {0}, GET_TEXT_F(MSG_FILAMENT_MAN)); // TODO: Chinese, English "Filament Management" JPG
  2899. DWINUI::MenuItemsPrepare(5);
  2900. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Draw_Prepare_Menu);
  2901. ADDMENUITEM(ICON_Park, GET_TEXT(MSG_FILAMENT_PARK_ENABLED), onDrawMenuItem, ParkHead);
  2902. ADDMENUITEM(ICON_FilMan, GET_TEXT(MSG_FILAMENTCHANGE), onDrawMenuItem, ChangeFilament);
  2903. TERN_(FILAMENT_LOAD_UNLOAD_GCODES, ADDMENUITEM(ICON_FilUnload, GET_TEXT(MSG_FILAMENTUNLOAD), onDrawMenuItem, UnloadFilament));
  2904. TERN_(FILAMENT_LOAD_UNLOAD_GCODES, ADDMENUITEM(ICON_FilLoad, GET_TEXT(MSG_FILAMENTLOAD), onDrawMenuItem, LoadFilament));
  2905. }
  2906. CurrentMenu->Draw();
  2907. }
  2908. #endif
  2909. #if ENABLED(MESH_BED_LEVELING)
  2910. void Draw_ManualMesh_Menu() {
  2911. checkkey = Menu;
  2912. if (ManualMesh == nullptr) ManualMesh = new MenuClass();
  2913. if (CurrentMenu != ManualMesh) {
  2914. CurrentMenu = ManualMesh;
  2915. SetMenuTitle({0}, {0}, GET_TEXT_F(MSG_MANUAL_MESH)); // TODO: Chinese, English "Manual Mesh Leveling" JPG
  2916. DWINUI::MenuItemsPrepare(5);
  2917. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Draw_Prepare_Menu);
  2918. ADDMENUITEM(ICON_ManualMesh, GET_TEXT(MSG_LEVEL_BED), onDrawMenuItem, ManualMeshStart);
  2919. MMeshMoveZItem = ADDMENUITEM_P(ICON_Zoffset, GET_TEXT(MSG_MOVE_Z), onDrawMMeshMoveZ, SetMMeshMoveZ, &current_position.z);
  2920. ADDMENUITEM(ICON_Axis, GET_TEXT(MSG_UBL_CONTINUE_MESH), onDrawMenuItem, ManualMeshContinue);
  2921. ADDMENUITEM(ICON_MeshSave, GET_TEXT(MSG_UBL_SAVE_MESH), onDrawMenuItem, ManualMeshSave);
  2922. }
  2923. CurrentMenu->Draw();
  2924. }
  2925. #endif
  2926. #if HAS_PREHEAT
  2927. void Draw_Preheat_Menu(frame_rect_t cn, frame_rect_t en, const __FlashStringHelper* text) {
  2928. checkkey = Menu;
  2929. if (CurrentMenu != PreheatMenu) {
  2930. CurrentMenu = PreheatMenu;
  2931. SetMenuTitle(cn, en, text);
  2932. DWINUI::MenuItemsPrepare(5);
  2933. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Draw_Temperature_Menu);
  2934. TERN_(HAS_HOTEND, ADDMENUITEM_P(ICON_SetEndTemp, GET_TEXT(MSG_UBL_SET_TEMP_HOTEND), onDrawSetPreheatHotend, SetPreheatEndTemp, &ui.material_preset[HMI_value.Preheat].hotend_temp));
  2935. TERN_(HAS_HEATED_BED, ADDMENUITEM_P(ICON_SetBedTemp, GET_TEXT(MSG_UBL_SET_TEMP_BED), onDrawSetPreheatBed, SetPreheatBedTemp, &ui.material_preset[HMI_value.Preheat].bed_temp));
  2936. TERN_(HAS_FAN, ADDMENUITEM_P(ICON_FanSpeed, GET_TEXT(MSG_FAN_SPEED), onDrawSetPreheatFan, SetPreheatFanSpeed, &ui.material_preset[HMI_value.Preheat].fan_speed));
  2937. TERN_(EEPROM_SETTINGS, ADDMENUITEM(ICON_WriteEEPROM, GET_TEXT(MSG_STORE_EEPROM), onDrawWriteEeprom, WriteEeprom));
  2938. }
  2939. CurrentMenu->Draw();
  2940. }
  2941. void Draw_Preheat1_Menu() {
  2942. HMI_value.Preheat = 0;
  2943. if (PreheatMenu == nullptr) PreheatMenu = new MenuClass();
  2944. Draw_Preheat_Menu({59, 16, 81, 14}, {56, 15, 85, 14}, F(PREHEAT_1_LABEL " Preheat Settings")); // TODO: English "PLA Settings" JPG
  2945. }
  2946. void Draw_Preheat2_Menu() {
  2947. HMI_value.Preheat = 1;
  2948. if (PreheatMenu == nullptr) PreheatMenu = new MenuClass();
  2949. Draw_Preheat_Menu({142, 16, 82, 14}, {56, 15, 85, 14}, F(PREHEAT_2_LABEL " Preheat Settings")); // TODO: English "ABS Settings" JPG
  2950. }
  2951. #ifdef PREHEAT_3_LABEL
  2952. void Draw_Preheat3_Menu() {
  2953. HMI_value.Preheat = 2;
  2954. if (PreheatMenu == nullptr) PreheatMenu = new MenuClass();
  2955. #define PREHEAT_3_TITLE PREHEAT_3_LABEL " Preheat Set."
  2956. Draw_Preheat_Menu({0}, {0}, F(PREHEAT_3_TITLE)); // TODO: Chinese, English "Custom Preheat Settings" JPG
  2957. }
  2958. #endif
  2959. #endif
  2960. void Draw_Temperature_Menu() {
  2961. checkkey = Menu;
  2962. if (TemperatureMenu == nullptr) TemperatureMenu = new MenuClass();
  2963. if (CurrentMenu != TemperatureMenu) {
  2964. CurrentMenu = TemperatureMenu;
  2965. SetMenuTitle({236, 2, 28, 12}, {56, 15, 85, 14}, GET_TEXT_F(MSG_TEMPERATURE));
  2966. DWINUI::MenuItemsPrepare(7);
  2967. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Draw_Control_Menu);
  2968. TERN_(HAS_HOTEND, HotendTargetItem = ADDMENUITEM_P(ICON_SetEndTemp, GET_TEXT(MSG_UBL_SET_TEMP_HOTEND), onDrawHotendTemp, SetHotendTemp, &thermalManager.temp_hotend[0].target));
  2969. TERN_(HAS_HEATED_BED, BedTargetItem = ADDMENUITEM_P(ICON_SetBedTemp, GET_TEXT(MSG_UBL_SET_TEMP_BED), onDrawBedTemp, SetBedTemp, &thermalManager.temp_bed.target));
  2970. TERN_(HAS_FAN, FanSpeedItem = ADDMENUITEM_P(ICON_FanSpeed, GET_TEXT(MSG_FAN_SPEED), onDrawFanSpeed, SetFanSpeed, &thermalManager.fan_speed[0]));
  2971. #if HAS_HOTEND
  2972. ADDMENUITEM(ICON_SetPLAPreheat, F(PREHEAT_1_LABEL " Preheat Settings"), onDrawPLAPreheatSubMenu, Draw_Preheat1_Menu);
  2973. ADDMENUITEM(ICON_SetABSPreheat, F(PREHEAT_2_LABEL " Preheat Settings"), onDrawABSPreheatSubMenu, Draw_Preheat2_Menu);
  2974. #ifdef PREHEAT_3_LABEL
  2975. ADDMENUITEM(ICON_SetCustomPreheat, PREHEAT_3_TITLE, onDrawSubMenu, Draw_Preheat3_Menu);
  2976. #endif
  2977. #endif
  2978. }
  2979. CurrentMenu->Draw();
  2980. }
  2981. void Draw_MaxSpeed_Menu() {
  2982. checkkey = Menu;
  2983. if (MaxSpeedMenu == nullptr) MaxSpeedMenu = new MenuClass();
  2984. if (CurrentMenu != MaxSpeedMenu) {
  2985. CurrentMenu = MaxSpeedMenu;
  2986. SetMenuTitle({1, 16, 28, 13}, {144, 16, 46, 11}, GET_TEXT_F(MSG_MAXSPEED));
  2987. DWINUI::MenuItemsPrepare(5);
  2988. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Draw_Motion_Menu);
  2989. ADDMENUITEM_P(ICON_MaxSpeedX, GET_TEXT(MSG_MAXSPEED_X), onDrawMaxSpeedX, SetMaxSpeedX, &planner.settings.max_feedrate_mm_s[X_AXIS]);
  2990. ADDMENUITEM_P(ICON_MaxSpeedY, GET_TEXT(MSG_MAXSPEED_Y), onDrawMaxSpeedY, SetMaxSpeedY, &planner.settings.max_feedrate_mm_s[Y_AXIS]);
  2991. ADDMENUITEM_P(ICON_MaxSpeedZ, GET_TEXT(MSG_MAXSPEED_Z), onDrawMaxSpeedZ, SetMaxSpeedZ, &planner.settings.max_feedrate_mm_s[Z_AXIS]);
  2992. TERN_(HAS_HOTEND, ADDMENUITEM_P(ICON_MaxSpeedE, GET_TEXT(MSG_MAXSPEED_E), onDrawMaxSpeedE, SetMaxSpeedE, &planner.settings.max_feedrate_mm_s[Z_AXIS]));
  2993. }
  2994. CurrentMenu->Draw();
  2995. }
  2996. void Draw_MaxAccel_Menu() {
  2997. checkkey = Menu;
  2998. if (MaxAccelMenu == nullptr) MaxAccelMenu = new MenuClass();
  2999. if (CurrentMenu != MaxAccelMenu) {
  3000. CurrentMenu = MaxAccelMenu;
  3001. SetMenuTitle({1, 16, 28, 13}, {144, 16, 46, 11}, GET_TEXT_F(MSG_ACCELERATION));
  3002. DWINUI::MenuItemsPrepare(5);
  3003. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Draw_Motion_Menu);
  3004. ADDMENUITEM_P(ICON_MaxAccX, GET_TEXT(MSG_AMAX_A), onDrawMaxAccelX, SetMaxAccelX, &planner.settings.max_acceleration_mm_per_s2[X_AXIS]);
  3005. ADDMENUITEM_P(ICON_MaxAccY, GET_TEXT(MSG_AMAX_B), onDrawMaxAccelY, SetMaxAccelY, &planner.settings.max_acceleration_mm_per_s2[Y_AXIS]);
  3006. ADDMENUITEM_P(ICON_MaxAccZ, GET_TEXT(MSG_AMAX_C), onDrawMaxAccelZ, SetMaxAccelZ, &planner.settings.max_acceleration_mm_per_s2[Z_AXIS]);
  3007. TERN_(HAS_HOTEND, ADDMENUITEM_P(ICON_MaxAccE, GET_TEXT(MSG_AMAX_E), onDrawMaxAccelE, SetMaxAccelE, &planner.settings.max_acceleration_mm_per_s2[E_AXIS]));
  3008. }
  3009. CurrentMenu->Draw();
  3010. }
  3011. #if HAS_CLASSIC_JERK
  3012. void Draw_MaxJerk_Menu() {
  3013. checkkey = Menu;
  3014. if (MaxJerkMenu == nullptr) MaxJerkMenu = new MenuClass();
  3015. if (CurrentMenu != MaxJerkMenu) {
  3016. CurrentMenu = MaxJerkMenu;
  3017. SetMenuTitle({1, 16, 28, 13}, {144, 16, 46, 11}, GET_TEXT_F(MSG_JERK));
  3018. DWINUI::MenuItemsPrepare(5);
  3019. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Draw_Motion_Menu);
  3020. ADDMENUITEM_P(ICON_MaxSpeedJerkX, GET_TEXT(MSG_VA_JERK), onDrawMaxJerkX, SetMaxJerkX, &planner.max_jerk[X_AXIS]);
  3021. ADDMENUITEM_P(ICON_MaxSpeedJerkY, GET_TEXT(MSG_VB_JERK), onDrawMaxJerkY, SetMaxJerkY, &planner.max_jerk[Y_AXIS]);
  3022. ADDMENUITEM_P(ICON_MaxSpeedJerkZ, GET_TEXT(MSG_VC_JERK), onDrawMaxJerkZ, SetMaxJerkZ, &planner.max_jerk[Z_AXIS]);
  3023. TERN_(HAS_HOTEND, ADDMENUITEM_P(ICON_MaxSpeedJerkE, GET_TEXT(MSG_VE_JERK), onDrawMaxJerkE, SetMaxJerkE, &planner.max_jerk[E_AXIS]));
  3024. }
  3025. CurrentMenu->Draw();
  3026. }
  3027. #endif
  3028. void Draw_Steps_Menu() {
  3029. checkkey = Menu;
  3030. if (StepsMenu == nullptr) StepsMenu = new MenuClass();
  3031. if (CurrentMenu != StepsMenu) {
  3032. CurrentMenu = StepsMenu;
  3033. SetMenuTitle({1, 16, 28, 13}, {144, 16, 46, 11}, GET_TEXT_F(MSG_STEPS_PER_MM));
  3034. DWINUI::MenuItemsPrepare(5);
  3035. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawBack, Draw_Motion_Menu);
  3036. ADDMENUITEM_P(ICON_StepX, GET_TEXT(MSG_A_STEPS), onDrawStepsX, SetStepsX, &planner.settings.axis_steps_per_mm[X_AXIS]);
  3037. ADDMENUITEM_P(ICON_StepY, GET_TEXT(MSG_B_STEPS), onDrawStepsY, SetStepsY, &planner.settings.axis_steps_per_mm[Y_AXIS]);
  3038. ADDMENUITEM_P(ICON_StepZ, GET_TEXT(MSG_C_STEPS), onDrawStepsZ, SetStepsZ, &planner.settings.axis_steps_per_mm[Z_AXIS]);
  3039. TERN_(HAS_HOTEND, ADDMENUITEM_P(ICON_StepE, GET_TEXT(MSG_E_STEPS), onDrawStepsE, SetStepsE, &planner.settings.axis_steps_per_mm[E_AXIS]));
  3040. }
  3041. CurrentMenu->Draw();
  3042. }
  3043. #if HAS_HOTEND
  3044. void Draw_HotendPID_Menu() {
  3045. checkkey = Menu;
  3046. if (HotendPIDMenu == nullptr) HotendPIDMenu = new MenuClass();
  3047. if (CurrentMenu != HotendPIDMenu) {
  3048. CurrentMenu = HotendPIDMenu;
  3049. CurrentMenu->MenuTitle.SetCaption(F("Hotend PID Settings"));
  3050. DWINUI::MenuItemsPrepare(8);
  3051. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawMenuItem, Draw_AdvancedSettings_Menu);
  3052. ADDMENUITEM(ICON_PIDNozzle, F("Hotend PID"), onDrawMenuItem, HotendPID);
  3053. ADDMENUITEM_P(ICON_PIDValue, F(STR_KP), onDrawPFloat2Menu, SetKp, &thermalManager.temp_hotend[0].pid.Kp);
  3054. ADDMENUITEM_P(ICON_PIDValue, F(STR_KI), onDrawPIDi, SetKi, &thermalManager.temp_hotend[0].pid.Ki);
  3055. ADDMENUITEM_P(ICON_PIDValue, F(STR_KD), onDrawPIDd, SetKd, &thermalManager.temp_hotend[0].pid.Kd);
  3056. ADDMENUITEM_P(ICON_Temperature, GET_TEXT(MSG_TEMPERATURE), onDrawPIntMenu, SetHotendPidT, &HMI_data.HotendPidT);
  3057. ADDMENUITEM_P(ICON_PIDcycles, GET_TEXT(MSG_PID_CYCLE), onDrawPIntMenu, SetPidCycles, &HMI_data.PidCycles);
  3058. TERN_(EEPROM_SETTINGS, ADDMENUITEM(ICON_WriteEEPROM, GET_TEXT(MSG_STORE_EEPROM), onDrawMenuItem, WriteEeprom));
  3059. }
  3060. CurrentMenu->Draw();
  3061. }
  3062. #endif
  3063. #if HAS_HEATED_BED
  3064. void Draw_BedPID_Menu() {
  3065. checkkey = Menu;
  3066. if (BedPIDMenu == nullptr) BedPIDMenu = new MenuClass();
  3067. if (CurrentMenu != BedPIDMenu) {
  3068. CurrentMenu = BedPIDMenu;
  3069. CurrentMenu->MenuTitle.SetCaption(F("Bed PID Settings"));
  3070. DWINUI::MenuItemsPrepare(8);
  3071. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawMenuItem, Draw_AdvancedSettings_Menu);
  3072. ADDMENUITEM(ICON_PIDNozzle, F("Bed PID"), onDrawMenuItem,BedPID);
  3073. ADDMENUITEM_P(ICON_PIDValue, F(STR_KP), onDrawPFloat2Menu, SetKp, &thermalManager.temp_bed.pid.Kp);
  3074. ADDMENUITEM_P(ICON_PIDValue, F(STR_KI), onDrawPIDi, SetKi, &thermalManager.temp_bed.pid.Ki);
  3075. ADDMENUITEM_P(ICON_PIDValue, F(STR_KD), onDrawPIDd, SetKd, &thermalManager.temp_bed.pid.Kd);
  3076. ADDMENUITEM_P(ICON_Temperature, GET_TEXT(MSG_TEMPERATURE), onDrawPIntMenu, SetBedPidT, &HMI_data.BedPidT);
  3077. ADDMENUITEM_P(ICON_PIDcycles, GET_TEXT(MSG_PID_CYCLE), onDrawPIntMenu, SetPidCycles, &HMI_data.PidCycles);
  3078. TERN_(EEPROM_SETTINGS, ADDMENUITEM(ICON_WriteEEPROM, GET_TEXT(MSG_STORE_EEPROM), onDrawMenuItem, WriteEeprom));
  3079. }
  3080. CurrentMenu->Draw();
  3081. }
  3082. #endif
  3083. #if EITHER(HAS_BED_PROBE, BABYSTEPPING)
  3084. void Draw_ZOffsetWiz_Menu() {
  3085. checkkey = Menu;
  3086. if (ZOffsetWizMenu == nullptr) ZOffsetWizMenu = new MenuClass();
  3087. if (CurrentMenu != ZOffsetWizMenu) {
  3088. CurrentMenu = ZOffsetWizMenu;
  3089. CurrentMenu->MenuTitle.SetCaption(GET_TEXT_F(MSG_PROBE_WIZARD));
  3090. DWINUI::MenuItemsPrepare(4);
  3091. ADDMENUITEM(ICON_Back, GET_TEXT(MSG_BUTTON_BACK), onDrawMenuItem, Draw_Prepare_Menu);
  3092. ADDMENUITEM(ICON_Homing, GET_TEXT(MSG_AUTO_HOME), onDrawMenuItem, AutoHome);
  3093. ADDMENUITEM(ICON_MoveZ0, F("Move Z to Home"), onDrawMenuItem, SetMoveZto0);
  3094. ADDMENUITEM_P(ICON_Zoffset, GET_TEXT(MSG_ZPROBE_ZOFFSET), onDrawPFloat2Menu, SetZOffset, &BABY_Z_VAR);
  3095. }
  3096. CurrentMenu->Draw();
  3097. }
  3098. #endif
  3099. #endif // DWIN_CREALITY_LCD_ENHANCED