123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318 |
- /**
- * Marlin 3D Printer Firmware
- * Copyright (c) 2020 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
- *
- * Based on Sprinter and grbl.
- * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
- *
- * This program is free software: you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation, either version 3 of the License, or
- * (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program. If not, see <http://www.gnu.org/licenses/>.
- *
- */
-
- //
- // Filament Change Menu
- //
-
- #include "../../inc/MarlinConfigPre.h"
-
- #if HAS_LCD_MENU && ENABLED(ADVANCED_PAUSE_FEATURE)
-
- #include "menu.h"
- #include "../../module/temperature.h"
- #include "../../feature/pause.h"
- #if HAS_FILAMENT_SENSOR
- #include "../../feature/runout.h"
- #endif
-
- //
- // Change Filament > Change/Unload/Load Filament
- //
- static PauseMode _change_filament_temp_mode; // =PAUSE_MODE_PAUSE_PRINT
- static int8_t _change_filament_temp_extruder; // =0
-
- inline PGM_P _change_filament_temp_command() {
- switch (_change_filament_temp_mode) {
- case PAUSE_MODE_LOAD_FILAMENT:
- return PSTR("M701 T%d");
- case PAUSE_MODE_UNLOAD_FILAMENT:
- return _change_filament_temp_extruder >= 0 ? PSTR("M702 T%d") : PSTR("M702 ;%d");
- case PAUSE_MODE_CHANGE_FILAMENT:
- case PAUSE_MODE_PAUSE_PRINT:
- default:
- return PSTR("M600 B0 T%d");
- }
- return GET_TEXT(MSG_FILAMENTCHANGE);
- }
-
- // Initiate Filament Load/Unload/Change at the specified temperature
- static void _change_filament_temp(const uint16_t temperature) {
- char cmd[11];
- sprintf_P(cmd, _change_filament_temp_command(), _change_filament_temp_extruder);
- thermalManager.setTargetHotend(temperature, _change_filament_temp_extruder);
- lcd_enqueue_one_now(cmd);
- }
-
- //
- // Menu to choose the temperature and start Filament Change
- //
-
- inline PGM_P change_filament_header(const PauseMode mode) {
- switch (mode) {
- case PAUSE_MODE_LOAD_FILAMENT:
- return GET_TEXT(MSG_FILAMENTLOAD);
- case PAUSE_MODE_UNLOAD_FILAMENT:
- return GET_TEXT(MSG_FILAMENTUNLOAD);
- default: break;
- }
- return GET_TEXT(MSG_FILAMENTCHANGE);
- }
-
- void _menu_temp_filament_op(const PauseMode mode, const int8_t extruder) {
- _change_filament_temp_mode = mode;
- _change_filament_temp_extruder = extruder;
- START_MENU();
- if (LCD_HEIGHT >= 4) STATIC_ITEM_P(change_filament_header(mode), SS_CENTER|SS_INVERT);
- BACK_ITEM(MSG_BACK);
- ACTION_ITEM(MSG_PREHEAT_1, []{ _change_filament_temp(ui.preheat_hotend_temp[0]); });
- ACTION_ITEM(MSG_PREHEAT_2, []{ _change_filament_temp(ui.preheat_hotend_temp[1]); });
- EDIT_ITEM_FAST(int3, MSG_PREHEAT_CUSTOM, &thermalManager.temp_hotend[_change_filament_temp_extruder].target, EXTRUDE_MINTEMP, heater_maxtemp[extruder] - 15, []{
- _change_filament_temp(thermalManager.temp_hotend[_change_filament_temp_extruder].target);
- });
- END_MENU();
- }
-
- /**
- *
- * "Change Filament" submenu
- *
- */
- #if E_STEPPERS > 1 || ENABLED(FILAMENT_LOAD_UNLOAD_GCODES)
- void menu_change_filament() {
- START_MENU();
- BACK_ITEM(MSG_MAIN);
-
- // Say "filament change" when no print is active
- editable.int8 = printingIsPaused() ? PAUSE_MODE_PAUSE_PRINT : PAUSE_MODE_CHANGE_FILAMENT;
-
- // Change filament
- #if E_STEPPERS == 1
- PGM_P const msg = GET_TEXT(MSG_FILAMENTCHANGE);
- if (thermalManager.targetTooColdToExtrude(active_extruder))
- SUBMENU_P(msg, []{ _menu_temp_filament_op(PAUSE_MODE_CHANGE_FILAMENT, 0); });
- else
- GCODES_ITEM_P(msg, PSTR("M600 B0"));
- #else
- PGM_P const msg = GET_TEXT(MSG_FILAMENTCHANGE_E);
- LOOP_L_N(s, E_STEPPERS) {
- if (thermalManager.targetTooColdToExtrude(s))
- SUBMENU_N_P(s, msg, []{ _menu_temp_filament_op(PAUSE_MODE_CHANGE_FILAMENT, MenuItemBase::itemIndex); });
- else {
- ACTION_ITEM_N_P(s, msg, []{
- char cmd[13];
- sprintf_P(cmd, PSTR("M600 B0 T%i"), int(MenuItemBase::itemIndex));
- lcd_enqueue_one_now(cmd);
- });
- }
- }
- #endif
-
- #if ENABLED(FILAMENT_LOAD_UNLOAD_GCODES)
- if (!printer_busy()) {
- // Load filament
- #if E_STEPPERS == 1
- PGM_P const msg_load = GET_TEXT(MSG_FILAMENTLOAD);
- if (thermalManager.targetTooColdToExtrude(active_extruder))
- SUBMENU_P(msg_load, []{ _menu_temp_filament_op(PAUSE_MODE_LOAD_FILAMENT, 0); });
- else
- GCODES_ITEM_P(msg_load, PSTR("M701"));
- #else
- PGM_P const msg_load = GET_TEXT(MSG_FILAMENTLOAD_E);
- LOOP_L_N(s, E_STEPPERS) {
- if (thermalManager.targetTooColdToExtrude(s))
- SUBMENU_N_P(s, msg_load, []{ _menu_temp_filament_op(PAUSE_MODE_LOAD_FILAMENT, MenuItemBase::itemIndex); });
- else {
- ACTION_ITEM_N_P(s, msg_load, []{
- char cmd[12];
- sprintf_P(cmd, PSTR("M701 T%i"), int(MenuItemBase::itemIndex));
- lcd_enqueue_one_now(cmd);
- });
- }
- }
- #endif
-
- // Unload filament
- #if E_STEPPERS == 1
- PGM_P const msg_unload = GET_TEXT(MSG_FILAMENTUNLOAD);
- if (thermalManager.targetTooColdToExtrude(active_extruder))
- SUBMENU_P(msg_unload, []{ _menu_temp_filament_op(PAUSE_MODE_UNLOAD_FILAMENT, 0); });
- else
- GCODES_ITEM_P(msg_unload, PSTR("M702"));
- #else
- #if ENABLED(FILAMENT_UNLOAD_ALL_EXTRUDERS)
- {
- bool too_cold = false;
- LOOP_L_N(s, E_STEPPERS) {
- if (thermalManager.targetTooColdToExtrude(s)) {
- too_cold = true; break;
- }
- }
- if (!too_cold)
- GCODES_ITEM(MSG_FILAMENTUNLOAD_ALL, PSTR("M702"));
- else
- SUBMENU(MSG_FILAMENTUNLOAD_ALL, []{ _menu_temp_filament_op(PAUSE_MODE_UNLOAD_FILAMENT, -1); });
- }
- #endif
- PGM_P const msg_unload = GET_TEXT(MSG_FILAMENTUNLOAD_E);
- LOOP_L_N(s, E_STEPPERS) {
- if (thermalManager.targetTooColdToExtrude(s))
- SUBMENU_N_P(s, msg_unload, []{ _menu_temp_filament_op(PAUSE_MODE_UNLOAD_FILAMENT, MenuItemBase::itemIndex); });
- else {
- ACTION_ITEM_N_P(s, msg_unload, []{
- char cmd[12];
- sprintf_P(cmd, PSTR("M702 T%i"), int(MenuItemBase::itemIndex));
- lcd_enqueue_one_now(cmd);
- });
- }
- }
- #endif
- } // !printer_busy
- #endif
-
- END_MENU();
- }
- #endif
-
- static uint8_t hotend_status_extruder = 0;
-
- static PGM_P pause_header() {
- switch (pause_mode) {
- case PAUSE_MODE_CHANGE_FILAMENT:
- return GET_TEXT(MSG_FILAMENT_CHANGE_HEADER);
- case PAUSE_MODE_LOAD_FILAMENT:
- return GET_TEXT(MSG_FILAMENT_CHANGE_HEADER_LOAD);
- case PAUSE_MODE_UNLOAD_FILAMENT:
- return GET_TEXT(MSG_FILAMENT_CHANGE_HEADER_UNLOAD);
- default: break;
- }
- return GET_TEXT(MSG_FILAMENT_CHANGE_HEADER_PAUSE);
- }
-
- // Portions from STATIC_ITEM...
- #define HOTEND_STATUS_ITEM() do { \
- if (_menuLineNr == _thisItemNr) { \
- if (ui.should_draw()) { \
- MenuItem_static::draw(_lcdLineNr, GET_TEXT(MSG_FILAMENT_CHANGE_NOZZLE), SS_INVERT); \
- ui.draw_hotend_status(_lcdLineNr, hotend_status_extruder); \
- } \
- if (_skipStatic && encoderLine <= _thisItemNr) { \
- ui.encoderPosition += ENCODER_STEPS_PER_MENU_ITEM; \
- ++encoderLine; \
- } \
- ui.refresh(LCDVIEW_CALL_REDRAW_NEXT); \
- } \
- ++_thisItemNr; \
- }while(0)
-
- void menu_pause_option() {
- START_MENU();
- #if LCD_HEIGHT > 2
- STATIC_ITEM(MSG_FILAMENT_CHANGE_OPTION_HEADER);
- #endif
- ACTION_ITEM(MSG_FILAMENT_CHANGE_OPTION_PURGE, []{ pause_menu_response = PAUSE_RESPONSE_EXTRUDE_MORE; });
- #if HAS_FILAMENT_SENSOR
- if (runout.filament_ran_out)
- EDIT_ITEM(bool, MSG_RUNOUT_SENSOR, &runout.enabled, runout.reset);
- #endif
- ACTION_ITEM(MSG_FILAMENT_CHANGE_OPTION_RESUME, []{ pause_menu_response = PAUSE_RESPONSE_RESUME_PRINT; });
- END_MENU();
- }
-
- //
- // ADVANCED_PAUSE_FEATURE message screens
- //
- // Warning: msg must have three null bytes to delimit lines!
- //
- void _lcd_pause_message(PGM_P const msg) {
- PGM_P const msg1 = msg;
- PGM_P const msg2 = msg1 + strlen_P(msg1) + 1;
- PGM_P const msg3 = msg2 + strlen_P(msg2) + 1;
- const bool has2 = msg2[0], has3 = msg3[0],
- skip1 = !has2 && (LCD_HEIGHT) >= 5;
-
- START_SCREEN();
- STATIC_ITEM_P(pause_header(), SS_CENTER|SS_INVERT); // 1: Header
- if (skip1) SKIP_ITEM(); // Move a single-line message down
- STATIC_ITEM_P(msg1); // 2: Message Line 1
- if (has2) STATIC_ITEM_P(msg2); // 3: Message Line 2
- if (has3 && (LCD_HEIGHT) >= 5) STATIC_ITEM_P(msg3); // 4: Message Line 3 (if LCD has 5 lines)
- if (skip1 + 1 + has2 + has3 < (LCD_HEIGHT) - 2) SKIP_ITEM(); // Push Hotend Status down, if needed
- HOTEND_STATUS_ITEM(); // 5: Hotend Status
- END_SCREEN();
- }
-
- void lcd_pause_pausing_message() { _lcd_pause_message(GET_TEXT(MSG_PAUSE_PRINT_INIT)); }
- void lcd_pause_changing_message() { _lcd_pause_message(GET_TEXT(MSG_FILAMENT_CHANGE_INIT)); }
- void lcd_pause_unload_message() { _lcd_pause_message(GET_TEXT(MSG_FILAMENT_CHANGE_UNLOAD)); }
- void lcd_pause_heating_message() { _lcd_pause_message(GET_TEXT(MSG_FILAMENT_CHANGE_HEATING)); }
- void lcd_pause_heat_message() { _lcd_pause_message(GET_TEXT(MSG_FILAMENT_CHANGE_HEAT)); }
- void lcd_pause_insert_message() { _lcd_pause_message(GET_TEXT(MSG_FILAMENT_CHANGE_INSERT)); }
- void lcd_pause_load_message() { _lcd_pause_message(GET_TEXT(MSG_FILAMENT_CHANGE_LOAD)); }
- void lcd_pause_waiting_message() { _lcd_pause_message(GET_TEXT(MSG_ADVANCED_PAUSE_WAITING)); }
- void lcd_pause_resume_message() { _lcd_pause_message(GET_TEXT(MSG_FILAMENT_CHANGE_RESUME)); }
-
- void lcd_pause_purge_message() {
- #if ENABLED(ADVANCED_PAUSE_CONTINUOUS_PURGE)
- _lcd_pause_message(GET_TEXT(MSG_FILAMENT_CHANGE_CONT_PURGE));
- #else
- _lcd_pause_message(GET_TEXT(MSG_FILAMENT_CHANGE_PURGE));
- #endif
- }
-
- FORCE_INLINE screenFunc_t ap_message_screen(const PauseMessage message) {
- switch (message) {
- case PAUSE_MESSAGE_PAUSING: return lcd_pause_pausing_message;
- case PAUSE_MESSAGE_CHANGING: return lcd_pause_changing_message;
- case PAUSE_MESSAGE_UNLOAD: return lcd_pause_unload_message;
- case PAUSE_MESSAGE_WAITING: return lcd_pause_waiting_message;
- case PAUSE_MESSAGE_INSERT: return lcd_pause_insert_message;
- case PAUSE_MESSAGE_LOAD: return lcd_pause_load_message;
- case PAUSE_MESSAGE_PURGE: return lcd_pause_purge_message;
- case PAUSE_MESSAGE_RESUME: return lcd_pause_resume_message;
- case PAUSE_MESSAGE_HEAT: return lcd_pause_heat_message;
- case PAUSE_MESSAGE_HEATING: return lcd_pause_heating_message;
- case PAUSE_MESSAGE_OPTION: pause_menu_response = PAUSE_RESPONSE_WAIT_FOR;
- return menu_pause_option;
- case PAUSE_MESSAGE_STATUS:
- default: break;
- }
- return nullptr;
- }
-
- void lcd_pause_show_message(
- const PauseMessage message,
- const PauseMode mode/*=PAUSE_MODE_SAME*/,
- const uint8_t extruder/*=active_extruder*/
- ) {
- if (mode != PAUSE_MODE_SAME) pause_mode = mode;
- hotend_status_extruder = extruder;
- const screenFunc_t next_screen = ap_message_screen(message);
- if (next_screen) {
- ui.defer_status_screen();
- ui.goto_screen(next_screen);
- }
- else
- ui.return_to_status();
- }
-
- #endif // HAS_LCD_MENU && ADVANCED_PAUSE_FEATURE
|