|
@ -457,13 +457,18 @@ uint16_t max_display_update_time = 0; |
|
|
* Synchronize safely while holding the current screen |
|
|
* Synchronize safely while holding the current screen |
|
|
* This blocks all further screen or stripe updates once called |
|
|
* This blocks all further screen or stripe updates once called |
|
|
*/ |
|
|
*/ |
|
|
|
|
|
extern uint8_t commands_in_queue; |
|
|
|
|
|
|
|
|
inline void lcd_synchronize() { |
|
|
inline void lcd_synchronize() { |
|
|
lcd_implementation_drawmenu_static(LCD_HEIGHT >= 4 ? 1 : 0, PSTR(MSG_MOVING)); |
|
|
lcd_implementation_drawmenu_static(LCD_HEIGHT >= 4 ? 1 : 0, PSTR(MSG_MOVING)); |
|
|
if (no_reentrance) return; |
|
|
if (no_reentrance) return; |
|
|
no_reentrance = true; |
|
|
no_reentrance = true; |
|
|
screenFunc_t old_screen = currentScreen; |
|
|
screenFunc_t old_screen = currentScreen; |
|
|
lcd_goto_screen(lcd_synchronize); |
|
|
lcd_goto_screen(lcd_synchronize); |
|
|
|
|
|
while (commands_in_queue) { |
|
|
|
|
|
idle(); |
|
|
stepper.synchronize(); |
|
|
stepper.synchronize(); |
|
|
|
|
|
} |
|
|
no_reentrance = false; |
|
|
no_reentrance = false; |
|
|
lcd_goto_screen(old_screen); |
|
|
lcd_goto_screen(old_screen); |
|
|
} |
|
|
} |
|
@ -982,7 +987,7 @@ void kill_screen(const char* lcd_msg) { |
|
|
MENU_ITEM_EDIT(int3, MSG_SPEED, &feedrate_percentage, 10, 999); |
|
|
MENU_ITEM_EDIT(int3, MSG_SPEED, &feedrate_percentage, 10, 999); |
|
|
|
|
|
|
|
|
// Manual bed leveling, Bed Z:
|
|
|
// Manual bed leveling, Bed Z:
|
|
|
#if ENABLED(LCD_BED_LEVELING) |
|
|
#if ENABLED(MESH_BED_LEVELING) && ENABLED(LCD_BED_LEVELING) |
|
|
MENU_ITEM_EDIT(float43, MSG_BED_Z, &mbl.z_offset, -1, 1); |
|
|
MENU_ITEM_EDIT(float43, MSG_BED_Z, &mbl.z_offset, -1, 1); |
|
|
#endif |
|
|
#endif |
|
|
|
|
|
|
|
@ -1325,14 +1330,33 @@ void kill_screen(const char* lcd_msg) { |
|
|
|
|
|
|
|
|
/**
|
|
|
/**
|
|
|
* |
|
|
* |
|
|
* "Prepare" > "Bed Leveling" handlers |
|
|
* "Prepare" > "Level Bed" handlers |
|
|
* |
|
|
* |
|
|
*/ |
|
|
*/ |
|
|
|
|
|
|
|
|
static uint8_t manual_probe_index; |
|
|
static uint8_t manual_probe_index; |
|
|
|
|
|
|
|
|
|
|
|
#if ENABLED(PROBE_MANUALLY) |
|
|
|
|
|
extern bool g29_in_progress; |
|
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
|
|
// LCD probed points are from defaults
|
|
|
|
|
|
constexpr uint8_t total_probe_points = |
|
|
|
|
|
#if ABL_GRID |
|
|
|
|
|
(ABL_GRID_MAX_POINTS_X) * (ABL_GRID_MAX_POINTS_Y) |
|
|
|
|
|
#elif ENABLED(AUTO_BED_LEVELING_3POINT) |
|
|
|
|
|
int(3) |
|
|
|
|
|
#elif ENABLED(AUTO_BED_LEVELING_UBL) |
|
|
|
|
|
(UBL_MESH_NUM_X_POINTS) * (UBL_MESH_NUM_Y_POINTS) |
|
|
|
|
|
#elif ENABLED(MESH_BED_LEVELING) |
|
|
|
|
|
(MESH_NUM_X_POINTS) * (MESH_NUM_Y_POINTS) |
|
|
|
|
|
#endif |
|
|
|
|
|
; |
|
|
|
|
|
|
|
|
|
|
|
#if ENABLED(MESH_BED_LEVELING) |
|
|
|
|
|
|
|
|
// Utility to go to the next mesh point
|
|
|
// Utility to go to the next mesh point
|
|
|
inline void _manual_probe_xy(float x, float y) { |
|
|
inline void _manual_probe_goto_xy(float x, float y) { |
|
|
if (no_reentrance) return; |
|
|
if (no_reentrance) return; |
|
|
#if MANUAL_PROBE_HEIGHT > 0 |
|
|
#if MANUAL_PROBE_HEIGHT > 0 |
|
|
current_position[Z_AXIS] = LOGICAL_Z_POSITION(Z_MIN_POS) + MANUAL_PROBE_HEIGHT; |
|
|
current_position[Z_AXIS] = LOGICAL_Z_POSITION(Z_MIN_POS) + MANUAL_PROBE_HEIGHT; |
|
@ -1348,7 +1372,11 @@ void kill_screen(const char* lcd_msg) { |
|
|
lcd_synchronize(); |
|
|
lcd_synchronize(); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
#endif // MESH_BED_LEVELING
|
|
|
|
|
|
|
|
|
|
|
|
#if ENABLED(MESH_BED_LEVELING) || ENABLED(PROBE_MANUALLY) |
|
|
void _lcd_level_goto_next_point(); |
|
|
void _lcd_level_goto_next_point(); |
|
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
void _lcd_level_bed_done() { |
|
|
void _lcd_level_bed_done() { |
|
|
if (lcdDrawUpdate) lcd_implementation_drawedit(PSTR(MSG_LEVEL_BED_DONE)); |
|
|
if (lcdDrawUpdate) lcd_implementation_drawedit(PSTR(MSG_LEVEL_BED_DONE)); |
|
@ -1356,7 +1384,19 @@ void kill_screen(const char* lcd_msg) { |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
/**
|
|
|
/**
|
|
|
* Step 7: Get the Z coordinate, then goto next point or exit |
|
|
* Step 6: Display "Next point: 1 / 9" while waiting for move to finish |
|
|
|
|
|
*/ |
|
|
|
|
|
void _lcd_level_bed_moving() { |
|
|
|
|
|
if (lcdDrawUpdate) { |
|
|
|
|
|
char msg[10]; |
|
|
|
|
|
sprintf_P(msg, PSTR("%i / %u"), (int)(manual_probe_index + 1), total_probe_points); |
|
|
|
|
|
lcd_implementation_drawedit(PSTR(MSG_LEVEL_BED_NEXT_POINT), msg); |
|
|
|
|
|
} |
|
|
|
|
|
lcdDrawUpdate = LCDVIEW_KEEP_REDRAWING; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
|
* Step 7: Get the Z coordinate, click goes to the next point or exits |
|
|
*/ |
|
|
*/ |
|
|
void _lcd_level_bed_get_z() { |
|
|
void _lcd_level_bed_get_z() { |
|
|
ENCODER_DIRECTION_NORMAL(); |
|
|
ENCODER_DIRECTION_NORMAL(); |
|
@ -1375,71 +1415,123 @@ void kill_screen(const char* lcd_msg) { |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
if (lcd_clicked) { |
|
|
if (lcd_clicked) { |
|
|
|
|
|
|
|
|
|
|
|
// Use a hook to set the probe point z
|
|
|
|
|
|
// (zigzag arranges in XY order)
|
|
|
|
|
|
#if ENABLED(AUTO_BED_LEVELING_UBL) |
|
|
|
|
|
|
|
|
|
|
|
// UBL set-z handling goes here
|
|
|
|
|
|
|
|
|
|
|
|
#elif ENABLED(PROBE_MANUALLY) |
|
|
|
|
|
|
|
|
|
|
|
// G29 helpfully records Z and goes to the next
|
|
|
|
|
|
// point (or beeps if done)
|
|
|
|
|
|
enqueue_and_echo_commands_P(PSTR("G29")); |
|
|
|
|
|
manual_probe_index++; |
|
|
|
|
|
|
|
|
|
|
|
#elif ENABLED(MESH_BED_LEVELING) |
|
|
|
|
|
|
|
|
mbl.set_zigzag_z(manual_probe_index++, current_position[Z_AXIS]); |
|
|
mbl.set_zigzag_z(manual_probe_index++, current_position[Z_AXIS]); |
|
|
if (manual_probe_index == (MESH_NUM_X_POINTS) * (MESH_NUM_Y_POINTS)) { |
|
|
|
|
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
|
|
// If done...
|
|
|
|
|
|
if (manual_probe_index == total_probe_points) { |
|
|
|
|
|
|
|
|
|
|
|
// Say "Done!"
|
|
|
lcd_goto_screen(_lcd_level_bed_done); |
|
|
lcd_goto_screen(_lcd_level_bed_done); |
|
|
|
|
|
|
|
|
|
|
|
// Raise Z to the "manual probe height"
|
|
|
#if MANUAL_PROBE_HEIGHT > 0 |
|
|
#if MANUAL_PROBE_HEIGHT > 0 |
|
|
current_position[Z_AXIS] = LOGICAL_Z_POSITION(Z_MIN_POS) + MANUAL_PROBE_HEIGHT; |
|
|
current_position[Z_AXIS] = LOGICAL_Z_POSITION(Z_MIN_POS) + MANUAL_PROBE_HEIGHT; |
|
|
line_to_current(Z_AXIS); |
|
|
line_to_current(Z_AXIS); |
|
|
lcd_synchronize(); |
|
|
lcd_synchronize(); |
|
|
#endif |
|
|
#endif |
|
|
|
|
|
|
|
|
|
|
|
// Enable leveling, if needed
|
|
|
|
|
|
#if ENABLED(MESH_BED_LEVELING) |
|
|
|
|
|
|
|
|
mbl.set_has_mesh(true); |
|
|
mbl.set_has_mesh(true); |
|
|
mbl.set_reactivate(true); |
|
|
mbl.set_reactivate(true); |
|
|
enqueue_and_echo_commands_P(PSTR("G28")); |
|
|
enqueue_and_echo_commands_P(PSTR("G28")); |
|
|
|
|
|
|
|
|
|
|
|
#elif ENABLED(AUTO_BED_LEVELING_UBL) |
|
|
|
|
|
|
|
|
|
|
|
// UBL enable goes here
|
|
|
|
|
|
|
|
|
|
|
|
#elif ENABLED(PROBE_MANUALLY) |
|
|
|
|
|
|
|
|
|
|
|
// ABL will be enabled due to "G29".
|
|
|
|
|
|
|
|
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
lcd_return_to_status(); |
|
|
lcd_return_to_status(); |
|
|
//LCD_MESSAGEPGM(MSG_LEVEL_BED_DONE);
|
|
|
//LCD_MESSAGEPGM(MSG_LEVEL_BED_DONE);
|
|
|
lcd_completion_feedback(); |
|
|
lcd_completion_feedback(); |
|
|
} |
|
|
} |
|
|
else { |
|
|
else { |
|
|
lcd_goto_screen(_lcd_level_goto_next_point); |
|
|
|
|
|
|
|
|
// Move to the next probe point, if needed
|
|
|
|
|
|
#if ENABLED(MESH_BED_LEVELING) || ENABLED(PROBE_MANUALLY) |
|
|
|
|
|
|
|
|
|
|
|
_lcd_level_goto_next_point(); |
|
|
|
|
|
|
|
|
|
|
|
#elif ENABLED(AUTO_BED_LEVELING_UBL) |
|
|
|
|
|
|
|
|
|
|
|
// UBL goto-next-point goes here
|
|
|
|
|
|
|
|
|
|
|
|
#endif |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
KeepDrawing: |
|
|
KeepDrawing: |
|
|
|
|
|
|
|
|
// Update on first display, then only on updates to Z position
|
|
|
// Update on first display, then only on updates to Z position
|
|
|
// Show message above on clicks instead
|
|
|
// Show message above on clicks instead
|
|
|
if (lcdDrawUpdate) { |
|
|
if (lcdDrawUpdate) { |
|
|
const float v = current_position[Z_AXIS]; |
|
|
const float v = current_position[Z_AXIS]; |
|
|
lcd_implementation_drawedit(PSTR(MSG_MOVE_Z), ftostr43sign(v + (v < 0 ? -0.0001 : 0.0001), '+')); |
|
|
lcd_implementation_drawedit(PSTR(MSG_MOVE_Z), ftostr43sign(v + (v < 0 ? -0.0001 : 0.0001), '+')); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
/**
|
|
|
#if ENABLED(MESH_BED_LEVELING) || ENABLED(PROBE_MANUALLY) |
|
|
* Step 6: Display "Next point: 1 / 9" while waiting for move to finish |
|
|
|
|
|
*/ |
|
|
|
|
|
void _lcd_level_bed_moving() { |
|
|
|
|
|
if (lcdDrawUpdate) { |
|
|
|
|
|
char msg[10]; |
|
|
|
|
|
#if ENABLED(MESH_BED_LEVELING) |
|
|
|
|
|
sprintf_P(msg, PSTR("%i / %u"), (int)(manual_probe_index + 1), (MESH_NUM_X_POINTS) * (MESH_NUM_Y_POINTS)); |
|
|
|
|
|
#elif ENABLED(AUTO_BED_LEVELING_UBL) |
|
|
|
|
|
sprintf_P(msg, PSTR("%i / %u"), (int)(manual_probe_index + 1), (UBL_MESH_NUM_X_POINTS) * (UBL_MESH_NUM_Y_POINTS)); |
|
|
|
|
|
#endif |
|
|
|
|
|
lcd_implementation_drawedit(PSTR(MSG_LEVEL_BED_NEXT_POINT), msg); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
lcdDrawUpdate = LCDVIEW_KEEP_REDRAWING; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
/**
|
|
|
* Step 5: Initiate a move to the next point |
|
|
* Step 5: Initiate a move to the next point |
|
|
*/ |
|
|
*/ |
|
|
void _lcd_level_goto_next_point() { |
|
|
void _lcd_level_goto_next_point() { |
|
|
|
|
|
|
|
|
// Set the menu to display ahead of blocking call
|
|
|
// Set the menu to display ahead of blocking call
|
|
|
lcd_goto_screen(_lcd_level_bed_moving); |
|
|
lcd_goto_screen(_lcd_level_bed_moving); |
|
|
|
|
|
|
|
|
// _manual_probe_xy runs the menu loop until the move is done
|
|
|
#if ENABLED(MESH_BED_LEVELING) |
|
|
|
|
|
|
|
|
int8_t px, py; |
|
|
int8_t px, py; |
|
|
mbl.zigzag(manual_probe_index, px, py); |
|
|
mbl.zigzag(manual_probe_index, px, py); |
|
|
_manual_probe_xy(mbl.index_to_xpos[px], mbl.index_to_ypos[py]); |
|
|
|
|
|
|
|
|
// Controls the loop until the move is done
|
|
|
|
|
|
_manual_probe_goto_xy( |
|
|
|
|
|
LOGICAL_X_POSITION(mbl.index_to_xpos[px]), |
|
|
|
|
|
LOGICAL_Y_POSITION(mbl.index_to_ypos[py]) |
|
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
|
|
#elif ENABLED(AUTO_BED_LEVELING_UBL) |
|
|
|
|
|
|
|
|
|
|
|
// UBL may have its own methodology
|
|
|
|
|
|
|
|
|
|
|
|
#elif ENABLED(PROBE_MANUALLY) |
|
|
|
|
|
|
|
|
|
|
|
// Just wait for the G29 move to complete
|
|
|
|
|
|
lcd_synchronize(); |
|
|
|
|
|
|
|
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
// After the blocking function returns, change menus
|
|
|
// After the blocking function returns, change menus
|
|
|
lcd_goto_screen(_lcd_level_bed_get_z); |
|
|
lcd_goto_screen(_lcd_level_bed_get_z); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
#endif // MESH_BED_LEVELING
|
|
|
|
|
|
|
|
|
/**
|
|
|
/**
|
|
|
* Step 4: Display "Click to Begin", wait for click |
|
|
* Step 4: Display "Click to Begin", wait for click |
|
|
* Move to the first probe position |
|
|
* Move to the first probe position |
|
@ -1448,7 +1540,14 @@ KeepDrawing: |
|
|
if (lcdDrawUpdate) lcd_implementation_drawedit(PSTR(MSG_LEVEL_BED_WAITING)); |
|
|
if (lcdDrawUpdate) lcd_implementation_drawedit(PSTR(MSG_LEVEL_BED_WAITING)); |
|
|
if (lcd_clicked) { |
|
|
if (lcd_clicked) { |
|
|
manual_probe_index = 0; |
|
|
manual_probe_index = 0; |
|
|
lcd_goto_screen(_lcd_level_goto_next_point); |
|
|
#if ENABLED(MESH_BED_LEVELING) |
|
|
|
|
|
_lcd_level_goto_next_point(); |
|
|
|
|
|
#elif ENABLED(AUTO_BED_LEVELING_UBL) |
|
|
|
|
|
// UBL click handling should go here
|
|
|
|
|
|
#elif ENABLED(PROBE_MANUALLY) |
|
|
|
|
|
enqueue_and_echo_commands_P(PSTR("G29")); |
|
|
|
|
|
_lcd_level_goto_next_point(); |
|
|
|
|
|
#endif |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
@ -1466,15 +1565,17 @@ KeepDrawing: |
|
|
* Step 2: Continue Bed Leveling... |
|
|
* Step 2: Continue Bed Leveling... |
|
|
*/ |
|
|
*/ |
|
|
void _lcd_level_bed_continue() { |
|
|
void _lcd_level_bed_continue() { |
|
|
|
|
|
#if PLANNER_LEVELING && DISABLED(AUTO_BED_LEVELING_UBL) |
|
|
|
|
|
reset_bed_level(); |
|
|
|
|
|
#endif |
|
|
defer_return_to_status = true; |
|
|
defer_return_to_status = true; |
|
|
axis_homed[X_AXIS] = axis_homed[Y_AXIS] = axis_homed[Z_AXIS] = false; |
|
|
axis_homed[X_AXIS] = axis_homed[Y_AXIS] = axis_homed[Z_AXIS] = false; |
|
|
mbl.reset(); |
|
|
|
|
|
enqueue_and_echo_commands_P(PSTR("G28")); |
|
|
|
|
|
lcd_goto_screen(_lcd_level_bed_homing); |
|
|
lcd_goto_screen(_lcd_level_bed_homing); |
|
|
|
|
|
enqueue_and_echo_commands_P(PSTR("G28")); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
/**
|
|
|
/**
|
|
|
* Step 1: MBL entry-point: "Cancel" or "Level Bed" |
|
|
* Step 1: Bed Level entry-point: "Cancel" or "Level Bed" |
|
|
*/ |
|
|
*/ |
|
|
void lcd_level_bed() { |
|
|
void lcd_level_bed() { |
|
|
START_MENU(); |
|
|
START_MENU(); |
|
@ -1520,12 +1621,19 @@ KeepDrawing: |
|
|
//
|
|
|
//
|
|
|
// Level Bed
|
|
|
// Level Bed
|
|
|
//
|
|
|
//
|
|
|
#if HAS_ABL |
|
|
#if ENABLED(LCD_BED_LEVELING) |
|
|
|
|
|
|
|
|
|
|
|
#if ENABLED(PROBE_MANUALLY) |
|
|
|
|
|
if (!g29_in_progress) |
|
|
|
|
|
#endif |
|
|
|
|
|
MENU_ITEM(submenu, MSG_LEVEL_BED, lcd_level_bed); |
|
|
|
|
|
|
|
|
|
|
|
#elif HAS_ABL |
|
|
|
|
|
|
|
|
MENU_ITEM(gcode, MSG_LEVEL_BED, |
|
|
MENU_ITEM(gcode, MSG_LEVEL_BED, |
|
|
axis_homed[X_AXIS] && axis_homed[Y_AXIS] ? PSTR("G29") : PSTR("G28\nG29") |
|
|
axis_homed[X_AXIS] && axis_homed[Y_AXIS] ? PSTR("G29") : PSTR("G28\nG29") |
|
|
); |
|
|
); |
|
|
#elif ENABLED(LCD_BED_LEVELING) |
|
|
|
|
|
MENU_ITEM(submenu, MSG_LEVEL_BED, lcd_level_bed); |
|
|
|
|
|
#endif |
|
|
#endif |
|
|
|
|
|
|
|
|
#if DISABLED(NO_WORKSPACE_OFFSETS) |
|
|
#if DISABLED(NO_WORKSPACE_OFFSETS) |
|
@ -2253,7 +2361,7 @@ KeepDrawing: |
|
|
MENU_ITEM_EDIT(float32, MSG_ZPROBE_ZOFFSET, &zprobe_zoffset, Z_PROBE_OFFSET_RANGE_MIN, Z_PROBE_OFFSET_RANGE_MAX); |
|
|
MENU_ITEM_EDIT(float32, MSG_ZPROBE_ZOFFSET, &zprobe_zoffset, Z_PROBE_OFFSET_RANGE_MIN, Z_PROBE_OFFSET_RANGE_MAX); |
|
|
#endif |
|
|
#endif |
|
|
// Manual bed leveling, Bed Z:
|
|
|
// Manual bed leveling, Bed Z:
|
|
|
#if ENABLED(LCD_BED_LEVELING) |
|
|
#if ENABLED(MESH_BED_LEVELING) && ENABLED(LCD_BED_LEVELING) |
|
|
MENU_ITEM_EDIT(float43, MSG_BED_Z, &mbl.z_offset, -1, 1); |
|
|
MENU_ITEM_EDIT(float43, MSG_BED_Z, &mbl.z_offset, -1, 1); |
|
|
#endif |
|
|
#endif |
|
|
MENU_ITEM_EDIT(float5, MSG_ACC, &planner.acceleration, 10, 99000); |
|
|
MENU_ITEM_EDIT(float5, MSG_ACC, &planner.acceleration, 10, 99000); |
|
|