My Marlin configs for Fabrikator Mini and CTC i3 Pro B
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

menu.h 18KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424
  1. /**
  2. * Marlin 3D Printer Firmware
  3. * Copyright (c) 2019 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
  4. *
  5. * Based on Sprinter and grbl.
  6. * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
  7. *
  8. * This program is free software: you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License as published by
  10. * the Free Software Foundation, either version 3 of the License, or
  11. * (at your option) any later version.
  12. *
  13. * This program is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU General Public License
  19. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  20. *
  21. */
  22. #pragma once
  23. #include "../ultralcd.h"
  24. #include "../../libs/numtostr.h"
  25. #include "../../inc/MarlinConfig.h"
  26. #include "limits.h"
  27. extern int8_t encoderLine, encoderTopLine, screen_items;
  28. extern bool screen_changed;
  29. #if HOTENDS
  30. constexpr int16_t heater_maxtemp[HOTENDS] = ARRAY_BY_HOTENDS(HEATER_0_MAXTEMP, HEATER_1_MAXTEMP, HEATER_2_MAXTEMP, HEATER_3_MAXTEMP, HEATER_4_MAXTEMP, HEATER_5_MAXTEMP);
  31. #endif
  32. void scroll_screen(const uint8_t limit, const bool is_menu);
  33. bool printer_busy();
  34. ////////////////////////////////////////////
  35. ////////// Menu Item Numeric Types /////////
  36. ////////////////////////////////////////////
  37. #define DECLARE_MENU_EDIT_TYPE(TYPE, NAME, STRFUNC, SCALE) \
  38. struct MenuItemInfo_##NAME { \
  39. typedef TYPE type_t; \
  40. static constexpr float scale = SCALE; \
  41. static inline char* strfunc(const float value) { return STRFUNC((TYPE) value); } \
  42. };
  43. DECLARE_MENU_EDIT_TYPE(uint8_t, percent, ui8tostr4pct, 100.0/255); // 100% right-justified
  44. DECLARE_MENU_EDIT_TYPE(int16_t, int3, i16tostr3, 1 ); // 123, -12 right-justified
  45. DECLARE_MENU_EDIT_TYPE(int16_t, int4, i16tostr4sign, 1 ); // 1234, -123 right-justified
  46. DECLARE_MENU_EDIT_TYPE(int8_t, int8, i8tostr3, 1 ); // 123, -12 right-justified
  47. DECLARE_MENU_EDIT_TYPE(uint8_t, uint8, ui8tostr3, 1 ); // 123 right-justified
  48. DECLARE_MENU_EDIT_TYPE(uint16_t, uint16_3, ui16tostr3, 1 ); // 123 right-justified
  49. DECLARE_MENU_EDIT_TYPE(uint16_t, uint16_4, ui16tostr4, 0.1 ); // 1234 right-justified
  50. DECLARE_MENU_EDIT_TYPE(uint16_t, uint16_5, ui16tostr5, 0.01 ); // 12345 right-justified
  51. DECLARE_MENU_EDIT_TYPE(float, float3, ftostr3, 1 ); // 123 right-justified
  52. DECLARE_MENU_EDIT_TYPE(float, float52, ftostr42_52, 100 ); // _2.34, 12.34, -2.34 or 123.45, -23.45
  53. DECLARE_MENU_EDIT_TYPE(float, float43, ftostr43sign, 1000 ); // 1.234
  54. DECLARE_MENU_EDIT_TYPE(float, float5, ftostr5rj, 1 ); // 12345 right-justified
  55. DECLARE_MENU_EDIT_TYPE(float, float5_25, ftostr5rj, 0.04f ); // 12345 right-justified (25 increment)
  56. DECLARE_MENU_EDIT_TYPE(float, float51, ftostr51rj, 10 ); // 1234.5 right-justified
  57. DECLARE_MENU_EDIT_TYPE(float, float51sign, ftostr51sign, 10 ); // +1234.5
  58. DECLARE_MENU_EDIT_TYPE(float, float52sign, ftostr52sign, 100 ); // +123.45
  59. DECLARE_MENU_EDIT_TYPE(uint32_t, long5, ftostr5rj, 0.01f ); // 12345 right-justified
  60. DECLARE_MENU_EDIT_TYPE(uint32_t, long5_25, ftostr5rj, 0.04f ); // 12345 right-justified (25 increment)
  61. ////////////////////////////////////////////
  62. ///////// Menu Item Draw Functions /////////
  63. ////////////////////////////////////////////
  64. typedef void (*selectFunc_t)();
  65. void draw_select_screen(PGM_P const yes, PGM_P const no, const bool yesno, PGM_P const pref, const char * const string, PGM_P const suff);
  66. void do_select_screen(PGM_P const yes, PGM_P const no, selectFunc_t yesFunc, selectFunc_t noFunc, PGM_P const pref, const char * const string=nullptr, PGM_P const suff=nullptr);
  67. inline void do_select_screen_yn(selectFunc_t yesFunc, selectFunc_t noFunc, PGM_P const pref, const char * const string=nullptr, PGM_P const suff=nullptr) {
  68. do_select_screen(PSTR(MSG_YES), PSTR(MSG_NO), yesFunc, noFunc, pref, string, suff);
  69. }
  70. #define SS_LEFT 0x00
  71. #define SS_CENTER 0x01
  72. #define SS_INVERT 0x02
  73. void draw_edit_screen(PGM_P const pstr, const char* const value=nullptr);
  74. void draw_menu_item(const bool sel, const uint8_t row, PGM_P const pstr, const char pre_char, const char post_char);
  75. void draw_menu_item_static(const uint8_t row, PGM_P const pstr, const uint8_t style=SS_CENTER, const char * const valstr=nullptr);
  76. void _draw_menu_item_edit(const bool sel, const uint8_t row, PGM_P const pstr, const char* const data, const bool pgm);
  77. FORCE_INLINE void draw_menu_item_back(const bool sel, const uint8_t row, PGM_P const pstr) { draw_menu_item(sel, row, pstr, LCD_STR_UPLEVEL[0], LCD_STR_UPLEVEL[0]); }
  78. FORCE_INLINE void draw_menu_item_edit(const bool sel, const uint8_t row, PGM_P const pstr, const char* const data) { _draw_menu_item_edit(sel, row, pstr, data, false); }
  79. FORCE_INLINE void draw_menu_item_edit_P(const bool sel, const uint8_t row, PGM_P const pstr, const char* const data) { _draw_menu_item_edit(sel, row, pstr, data, true); }
  80. #define draw_menu_item_submenu(sel, row, pstr, data) draw_menu_item(sel, row, pstr, '>', LCD_STR_ARROW_RIGHT[0])
  81. #define draw_menu_item_gcode(sel, row, pstr, gcode) draw_menu_item(sel, row, pstr, '>', ' ')
  82. #define draw_menu_item_function(sel, row, pstr, data) draw_menu_item(sel, row, pstr, '>', ' ')
  83. #define DRAW_MENU_ITEM_SETTING_EDIT_GENERIC(VAL) draw_menu_item_edit(sel, row, pstr, VAL)
  84. #define DRAW_BOOL_SETTING(sel, row, pstr, data) draw_menu_item_edit_P(sel, row, pstr, (*(data))?PSTR(MSG_LCD_ON):PSTR(MSG_LCD_OFF))
  85. #if ENABLED(SDSUPPORT)
  86. class CardReader;
  87. void draw_sd_menu_item(const bool sel, const uint8_t row, PGM_P const pstr, CardReader &theCard, const bool isDir);
  88. FORCE_INLINE void draw_menu_item_sdfile(const bool sel, const uint8_t row, PGM_P const pstr, CardReader &theCard) { draw_sd_menu_item(sel, row, pstr, theCard, false); }
  89. FORCE_INLINE void draw_menu_item_sdfolder(const bool sel, const uint8_t row, PGM_P const pstr, CardReader &theCard) { draw_sd_menu_item(sel, row, pstr, theCard, true); }
  90. #endif
  91. #if HAS_GRAPHICAL_LCD && EITHER(BABYSTEP_ZPROBE_GFX_OVERLAY, MESH_EDIT_GFX_OVERLAY)
  92. void _lcd_zoffset_overlay_gfx(const float zvalue);
  93. #endif
  94. ////////////////////////////////////////////
  95. /////// Edit Setting Draw Functions ////////
  96. ////////////////////////////////////////////
  97. #define _DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(TYPE, NAME, STRFUNC) \
  98. FORCE_INLINE void draw_menu_item_edit_##NAME (const bool sel, const uint8_t row, PGM_P const pstr, PGM_P const pstr2, TYPE * const data, ...) { \
  99. UNUSED(pstr2); \
  100. DRAW_MENU_ITEM_SETTING_EDIT_GENERIC(STRFUNC(*(data))); \
  101. } \
  102. FORCE_INLINE void draw_menu_item_edit_accessor_##NAME (const bool sel, const uint8_t row, PGM_P const pstr, PGM_P const pstr2, TYPE (*pget)(), void (*pset)(TYPE), ...) { \
  103. UNUSED(pstr2); UNUSED(pset); \
  104. DRAW_MENU_ITEM_SETTING_EDIT_GENERIC(STRFUNC(pget())); \
  105. } \
  106. typedef void NAME##_void
  107. #define DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(NAME) _DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(MenuItemInfo_##NAME::type_t, NAME, MenuItemInfo_##NAME::strfunc)
  108. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(percent); // 100% right-justified
  109. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(int3); // 123, -12 right-justified
  110. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(int4); // 1234, -123 right-justified
  111. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(int8); // 123, -12 right-justified
  112. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(uint8); // 123 right-justified
  113. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(uint16_3); // 123 right-justified
  114. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(uint16_4); // 1234 right-justified
  115. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(uint16_5); // 12345 right-justified
  116. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(float3); // 123 right-justified
  117. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(float52); // _2.34, 12.34, -2.34 or 123.45, -23.45
  118. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(float43); // 1.234
  119. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(float5); // 12345 right-justified
  120. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(float5_25); // 12345 right-justified (25 increment)
  121. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(float51); // _1234.5 right-justified
  122. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(float51sign); // +1234.5
  123. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(float52sign); // +123.45
  124. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(long5); // 12345 right-justified
  125. DEFINE_DRAW_MENU_ITEM_SETTING_EDIT(long5_25); // 12345 right-justified (25 increment)
  126. #define draw_menu_item_edit_bool(sel, row, pstr, pstr2, data, ...) DRAW_BOOL_SETTING(sel, row, pstr, data)
  127. #define draw_menu_item_edit_accessor_bool(sel, row, pstr, pstr2, pget, pset) DRAW_BOOL_SETTING(sel, row, pstr, data)
  128. ////////////////////////////////////////////
  129. /////////////// Menu Actions ///////////////
  130. ////////////////////////////////////////////
  131. class MenuItem_back {
  132. public:
  133. static inline void action() {
  134. ui.goto_previous_screen(
  135. #if ENABLED(TURBO_BACK_MENU_ITEM)
  136. true
  137. #endif
  138. );
  139. }
  140. };
  141. class MenuItem_submenu {
  142. public:
  143. static inline void action(const screenFunc_t func) { ui.save_previous_screen(); ui.goto_screen(func); }
  144. };
  145. class MenuItem_gcode {
  146. public:
  147. static void action(const char * const pgcode);
  148. };
  149. class MenuItem_function {
  150. public:
  151. static inline void action(const menuAction_t func) { (*func)(); };
  152. };
  153. ////////////////////////////////////////////
  154. /////////// Menu Editing Actions ///////////
  155. ////////////////////////////////////////////
  156. class MenuItemBase {
  157. private:
  158. static PGM_P editLabel;
  159. static void *editValue;
  160. static int32_t minEditValue, maxEditValue;
  161. static screenFunc_t callbackFunc;
  162. static bool liveEdit;
  163. protected:
  164. typedef char* (*strfunc_t)(const int32_t);
  165. typedef void (*loadfunc_t)(void *, const int32_t);
  166. static void init(PGM_P const el, void * const ev, const int32_t minv, const int32_t maxv, const uint16_t ep, const screenFunc_t cs, const screenFunc_t cb, const bool le);
  167. static void edit(strfunc_t, loadfunc_t);
  168. };
  169. template<typename NAME>
  170. class TMenuItem : MenuItemBase {
  171. private:
  172. typedef typename NAME::type_t type_t;
  173. static inline float unscale(const float value) { return value * (1.0f / NAME::scale); }
  174. static inline float scale(const float value) { return value * NAME::scale; }
  175. static void load(void *ptr, const int32_t value) { *((type_t*)ptr) = unscale(value); }
  176. static char* to_string(const int32_t value) { return NAME::strfunc(unscale(value)); }
  177. public:
  178. static void action_edit(PGM_P const pstr, type_t * const ptr, const type_t minValue, const type_t maxValue, const screenFunc_t callback=nullptr, const bool live=false) {
  179. // Make sure minv and maxv fit within int16_t
  180. const int32_t minv = _MAX(scale(minValue), INT16_MIN),
  181. maxv = _MIN(scale(maxValue), INT16_MAX);
  182. init(pstr, ptr, minv, maxv - minv, scale(*ptr) - minv, edit, callback, live);
  183. }
  184. static void edit() { MenuItemBase::edit(to_string, load); }
  185. };
  186. #define DECLARE_MENU_EDIT_ITEM(NAME) typedef TMenuItem<MenuItemInfo_##NAME> MenuItem_##NAME;
  187. DECLARE_MENU_EDIT_ITEM(percent);
  188. DECLARE_MENU_EDIT_ITEM(int3);
  189. DECLARE_MENU_EDIT_ITEM(int4);
  190. DECLARE_MENU_EDIT_ITEM(int8);
  191. DECLARE_MENU_EDIT_ITEM(uint8);
  192. DECLARE_MENU_EDIT_ITEM(uint16_3);
  193. DECLARE_MENU_EDIT_ITEM(uint16_4);
  194. DECLARE_MENU_EDIT_ITEM(uint16_5);
  195. DECLARE_MENU_EDIT_ITEM(float3);
  196. DECLARE_MENU_EDIT_ITEM(float52);
  197. DECLARE_MENU_EDIT_ITEM(float43);
  198. DECLARE_MENU_EDIT_ITEM(float5);
  199. DECLARE_MENU_EDIT_ITEM(float5_25);
  200. DECLARE_MENU_EDIT_ITEM(float51);
  201. DECLARE_MENU_EDIT_ITEM(float51sign);
  202. DECLARE_MENU_EDIT_ITEM(float52sign);
  203. DECLARE_MENU_EDIT_ITEM(long5);
  204. DECLARE_MENU_EDIT_ITEM(long5_25);
  205. class MenuItem_bool {
  206. public:
  207. static void action_edit(PGM_P const pstr, bool* ptr, const screenFunc_t callbackFunc=nullptr);
  208. };
  209. ////////////////////////////////////////////
  210. //////////// Menu System Macros ////////////
  211. ////////////////////////////////////////////
  212. /**
  213. * SCREEN_OR_MENU_LOOP generates init code for a screen or menu
  214. *
  215. * encoderTopLine is the top menu line to display
  216. * _lcdLineNr is the index of the LCD line (e.g., 0-3)
  217. * _menuLineNr is the menu item to draw and process
  218. * _thisItemNr is the index of each MENU_ITEM or STATIC_ITEM
  219. */
  220. #define SCREEN_OR_MENU_LOOP() \
  221. int8_t _menuLineNr = encoderTopLine, _thisItemNr; \
  222. for (int8_t _lcdLineNr = 0; _lcdLineNr < LCD_HEIGHT; _lcdLineNr++, _menuLineNr++) { \
  223. _thisItemNr = 0
  224. /**
  225. * START_SCREEN Opening code for a screen having only static items.
  226. * Do simplified scrolling of the entire screen.
  227. *
  228. * START_MENU Opening code for a screen with menu items.
  229. * Scroll as-needed to keep the selected line in view.
  230. */
  231. #define START_SCREEN() \
  232. scroll_screen(LCD_HEIGHT, false); \
  233. bool _skipStatic = false; \
  234. SCREEN_OR_MENU_LOOP()
  235. #define START_MENU() \
  236. scroll_screen(1, true); \
  237. bool _skipStatic = true; \
  238. SCREEN_OR_MENU_LOOP()
  239. #define END_SCREEN() \
  240. } \
  241. screen_items = _thisItemNr
  242. #define END_MENU() \
  243. } \
  244. screen_items = _thisItemNr; \
  245. UNUSED(_skipStatic)
  246. #if ENABLED(ENCODER_RATE_MULTIPLIER)
  247. #define ENCODER_RATE_MULTIPLY(F) (ui.encoderRateMultiplierEnabled = F)
  248. #define _MENU_ITEM_MULTIPLIER_CHECK(USE_MULTIPLIER) do{ if (USE_MULTIPLIER) ui.enable_encoder_multiplier(true); }while(0)
  249. //#define ENCODER_RATE_MULTIPLIER_DEBUG // If defined, output the encoder steps per second value
  250. #else
  251. #define ENCODER_RATE_MULTIPLY(F) NOOP
  252. #define _MENU_ITEM_MULTIPLIER_CHECK(USE_MULTIPLIER)
  253. #endif
  254. /**
  255. * MENU_ITEM generates draw & handler code for a menu item, potentially calling:
  256. *
  257. * draw_menu_item_<type>[_variant](sel, row, label, arg3...)
  258. * MenuItem_<type>::action[_variant](arg3...)
  259. *
  260. * Examples:
  261. * MENU_ITEM(back, MSG_WATCH, 0 [dummy parameter] )
  262. * or
  263. * MENU_BACK(MSG_WATCH)
  264. * draw_menu_item_back(sel, row, PSTR(MSG_WATCH))
  265. * MenuItem_back::action()
  266. *
  267. * MENU_ITEM(function, MSG_PAUSE_PRINT, lcd_sdcard_pause)
  268. * draw_menu_item_function(sel, row, PSTR(MSG_PAUSE_PRINT), lcd_sdcard_pause)
  269. * MenuItem_function::action(lcd_sdcard_pause)
  270. *
  271. * MENU_ITEM_EDIT(int3, MSG_SPEED, &feedrate_percentage, 10, 999)
  272. * draw_menu_item_edit_int3(sel, row, PSTR(MSG_SPEED), PSTR(MSG_SPEED), &feedrate_percentage, 10, 999)
  273. * MenuItem_int3::action_edit(PSTR(MSG_SPEED), &feedrate_percentage, 10, 999)
  274. *
  275. */
  276. #define _MENU_ITEM_VARIANT_P(TYPE, VARIANT, USE_MULTIPLIER, PLABEL, V...) do { \
  277. _skipStatic = false; \
  278. if (_menuLineNr == _thisItemNr) { \
  279. if (encoderLine == _thisItemNr && ui.use_click()) { \
  280. _MENU_ITEM_MULTIPLIER_CHECK(USE_MULTIPLIER); \
  281. MenuItem_##TYPE ::action ## VARIANT(V); \
  282. if (screen_changed) return; \
  283. } \
  284. if (ui.should_draw()) \
  285. draw_menu_item ## VARIANT ## _ ## TYPE(encoderLine == _thisItemNr, _lcdLineNr, PLABEL, ##V); \
  286. } \
  287. ++_thisItemNr; \
  288. }while(0)
  289. // Used to print static text with no visible cursor.
  290. // Parameters: label [, bool center [, bool invert [, char *value] ] ]
  291. #define STATIC_ITEM_P(PLABEL, V...) do{ \
  292. if (_menuLineNr == _thisItemNr) { \
  293. if (_skipStatic && encoderLine <= _thisItemNr) { \
  294. ui.encoderPosition += ENCODER_STEPS_PER_MENU_ITEM; \
  295. ++encoderLine; \
  296. } \
  297. if (ui.should_draw()) \
  298. draw_menu_item_static(_lcdLineNr, PLABEL, ##V); \
  299. } \
  300. ++_thisItemNr; \
  301. } while(0)
  302. #define MENU_ITEM_ADDON_START(X) do{ \
  303. if (ui.should_draw() && _menuLineNr == _thisItemNr - 1) { \
  304. SETCURSOR(X, _lcdLineNr)
  305. #define MENU_ITEM_ADDON_END() } }while(0)
  306. #define STATIC_ITEM(LABEL, V...) STATIC_ITEM_P(PSTR(LABEL), ##V)
  307. #define MENU_BACK(LABEL) MENU_ITEM(back, LABEL)
  308. #define MENU_ITEM_DUMMY() do { _thisItemNr++; }while(0)
  309. #define MENU_ITEM_P(TYPE, PLABEL, V...) _MENU_ITEM_VARIANT_P(TYPE, , false, PLABEL, ##V)
  310. #define MENU_ITEM(TYPE, LABEL, V...) _MENU_ITEM_VARIANT_P(TYPE, , false, PSTR(LABEL), ##V)
  311. #define MENU_ITEM_EDIT(TYPE, LABEL, V...) _MENU_ITEM_VARIANT_P(TYPE, _edit, false, PSTR(LABEL), PSTR(LABEL), ##V)
  312. #define MENU_ITEM_EDIT_CALLBACK(TYPE, LABEL, V...) _MENU_ITEM_VARIANT_P(TYPE, _edit, false, PSTR(LABEL), PSTR(LABEL), ##V)
  313. #define MENU_MULTIPLIER_ITEM_EDIT(TYPE, LABEL, V...) _MENU_ITEM_VARIANT_P(TYPE, _edit, true, PSTR(LABEL), PSTR(LABEL), ##V)
  314. #define MENU_MULTIPLIER_ITEM_EDIT_CALLBACK(TYPE, LABEL, V...) _MENU_ITEM_VARIANT_P(TYPE, _edit, true, PSTR(LABEL), PSTR(LABEL), ##V)
  315. ////////////////////////////////////////////
  316. /////////////// Menu Screens ///////////////
  317. ////////////////////////////////////////////
  318. void menu_main();
  319. void menu_move();
  320. #if ENABLED(SDSUPPORT)
  321. void menu_media();
  322. #endif
  323. // First Fan Speed title in "Tune" and "Control>Temperature" menus
  324. #if FAN_COUNT > 0 && HAS_FAN0
  325. #if FAN_COUNT > 1
  326. #define FAN_SPEED_1_SUFFIX " 1"
  327. #else
  328. #define FAN_SPEED_1_SUFFIX ""
  329. #endif
  330. #endif
  331. ////////////////////////////////////////////
  332. //////// Menu Item Helper Functions ////////
  333. ////////////////////////////////////////////
  334. void lcd_move_z();
  335. void _lcd_draw_homing();
  336. #define HAS_LINE_TO_Z ANY(DELTA, PROBE_MANUALLY, MESH_BED_LEVELING, LEVEL_BED_CORNERS)
  337. #if HAS_LINE_TO_Z
  338. void line_to_z(const float &z);
  339. #endif
  340. #if ANY(AUTO_BED_LEVELING_UBL, PID_AUTOTUNE_MENU, ADVANCED_PAUSE_FEATURE)
  341. void lcd_enqueue_one_now(const char * const cmd);
  342. void lcd_enqueue_one_now_P(PGM_P const cmd);
  343. #endif
  344. #if ENABLED(LEVEL_BED_CORNERS)
  345. void _lcd_level_bed_corners();
  346. #endif
  347. #if ENABLED(ENABLE_LEVELING_FADE_HEIGHT)
  348. extern float lcd_z_fade_height;
  349. void _lcd_set_z_fade_height();
  350. #endif
  351. #if ENABLED(LCD_BED_LEVELING) || (HAS_LEVELING && DISABLED(SLIM_LCD_MENUS))
  352. void _lcd_toggle_bed_leveling();
  353. #endif
  354. #if ENABLED(BABYSTEPPING)
  355. #if ENABLED(BABYSTEP_ZPROBE_OFFSET)
  356. void lcd_babystep_zoffset();
  357. #else
  358. void lcd_babystep_z();
  359. #endif
  360. #endif
  361. #if ENABLED(EEPROM_SETTINGS)
  362. void lcd_store_settings();
  363. void lcd_load_settings();
  364. #endif
  365. #if ENABLED(POWER_LOSS_RECOVERY)
  366. void menu_job_recovery();
  367. #endif