My Marlin configs for Fabrikator Mini and CTC i3 Pro B
您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

dwin.cpp 134KB

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