bed_mesh: add support for round beds
Signed-off-by: Eric Callahan <arksine.code@gmail.com>
This commit is contained in:
parent
542f941f07
commit
cdcc320710
|
@ -117,6 +117,26 @@
|
||||||
# using a probe to home the z-axis, it is recommended to define
|
# using a probe to home the z-axis, it is recommended to define
|
||||||
# a [homing_override] section in printer.cfg to home toward the
|
# a [homing_override] section in printer.cfg to home toward the
|
||||||
# center of the print area.
|
# center of the print area.
|
||||||
|
#
|
||||||
|
# Visual Examples:
|
||||||
|
# bed_shape = rectangular, probe_count = 3,3:
|
||||||
|
# x---x---x (max_point)
|
||||||
|
# |
|
||||||
|
# x---x---x
|
||||||
|
# |
|
||||||
|
# (min_point) x---x---x
|
||||||
|
#
|
||||||
|
# bed_shape = round, probe_count = 5, radius = r:
|
||||||
|
# x (0,r) end
|
||||||
|
# /
|
||||||
|
# x---x---x
|
||||||
|
# \
|
||||||
|
# (-r,0) x---x---x---x---x (r,0)
|
||||||
|
# \
|
||||||
|
# x---x---x
|
||||||
|
# /
|
||||||
|
# x (0,-r) start
|
||||||
|
#
|
||||||
#[bed_mesh]
|
#[bed_mesh]
|
||||||
#speed: 50
|
#speed: 50
|
||||||
# The speed (in mm/s) of non-probing moves during the
|
# The speed (in mm/s) of non-probing moves during the
|
||||||
|
@ -130,21 +150,34 @@
|
||||||
#sample_retract_dist: 2.0
|
#sample_retract_dist: 2.0
|
||||||
# The distance (in mm) to retract between each sample if
|
# The distance (in mm) to retract between each sample if
|
||||||
# sampling more than once. Default is 2mm.
|
# sampling more than once. Default is 2mm.
|
||||||
|
#bed_shape: rectangular
|
||||||
|
# Defines the shape of the bed for probing. May be 'rectangular',
|
||||||
|
# as is common for cartesian printers, or 'round', as is common for
|
||||||
|
# delta printers. Default is rectangular.
|
||||||
|
#radius:
|
||||||
|
# Defines the radius to probe when the bed_shape is 'round'. Note
|
||||||
|
# that the radius is relative to the nozzle's origin, if using a
|
||||||
|
# probe be sure to account for its offset. This parameter must be
|
||||||
|
# provided if the bed_shape is 'round'.
|
||||||
#min_point:
|
#min_point:
|
||||||
# An X,Y point defining the minimum coordinate to probe on
|
# Defines the minimum x,y position to probe when the bed_shape
|
||||||
# the bed. Note that this refers to the nozzle position,
|
# is 'rectangular'. Note that this refers to the nozzle position,
|
||||||
# and take care that you do not define a point that will move
|
# take care that you do not define a point that will move the
|
||||||
# the probe off of the bed. This parameter must be provided.
|
# probe off of the bed. This parameter must be provided.
|
||||||
#max_point:
|
#max_point:
|
||||||
# An X,Y point defining the maximum coordinate to probe on
|
# Defines the maximum x,y position to probe when the bed_shape
|
||||||
# the bed. Follow the same precautions as listed in min_point.
|
# is 'rectangular'. Follow the same precautions as listed in min_point.
|
||||||
# Also note that this does not necessarily define the last point
|
# Also note that this does not necessarily define the last point
|
||||||
# probed, only the maximum coordinate. This parameter must be provided.
|
# probed, only the maximum coordinate. This parameter must be provided.
|
||||||
#probe_count: 3,3
|
#probe_count: 3,3
|
||||||
# A comma separated pair of integer values (X,Y) defining the number
|
## OR ##
|
||||||
# of points to probe along each axis. A single value is also valid,
|
#probe_count: 5
|
||||||
# in which case that value will be for both axes. Default is 3,3
|
# For 'rectangular' beds, this is a comma separate pair of integer
|
||||||
# which probes a 3x3 grid.
|
# values (X,Y) defining the number of points to probe along each axis.
|
||||||
|
# A single value is also valid, in which case that value will be applied
|
||||||
|
# to both axes. 'Round' beds only accept a single integer value that is
|
||||||
|
# applied to both axes. The probe count must be odd for round beds.
|
||||||
|
# Default is 3,3 for 'rectangular' beds, and 5 for 'round' beds.
|
||||||
#fade_start: 1.0
|
#fade_start: 1.0
|
||||||
# The gcode z position in which to start phasing out z-adjustment
|
# The gcode z position in which to start phasing out z-adjustment
|
||||||
# when fade is enabled. Default is 1.0.
|
# when fade is enabled. Default is 1.0.
|
||||||
|
|
|
@ -10,6 +10,8 @@ import json
|
||||||
import probe
|
import probe
|
||||||
import collections
|
import collections
|
||||||
|
|
||||||
|
BED_SHAPES = {'rectangular': 0, 'round': 1}
|
||||||
|
|
||||||
class BedMeshError(Exception):
|
class BedMeshError(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
@ -163,6 +165,7 @@ class BedMeshCalibrate:
|
||||||
def __init__(self, config, bedmesh):
|
def __init__(self, config, bedmesh):
|
||||||
self.printer = config.get_printer()
|
self.printer = config.get_printer()
|
||||||
self.name = config.get_name()
|
self.name = config.get_name()
|
||||||
|
self.radius = None
|
||||||
self.bedmesh = bedmesh
|
self.bedmesh = bedmesh
|
||||||
self.probed_z_table = None
|
self.probed_z_table = None
|
||||||
self.build_map = False
|
self.build_map = False
|
||||||
|
@ -185,14 +188,29 @@ class BedMeshCalibrate:
|
||||||
'BED_MESH_PROFILE', self.cmd_BED_MESH_PROFILE,
|
'BED_MESH_PROFILE', self.cmd_BED_MESH_PROFILE,
|
||||||
desc=self.cmd_BED_MESH_PROFILE_help)
|
desc=self.cmd_BED_MESH_PROFILE_help)
|
||||||
def _generate_points(self, config):
|
def _generate_points(self, config):
|
||||||
x_cnt, y_cnt = parse_pair(
|
shape = config.getchoice('bed_shape', BED_SHAPES, 'rectangular')
|
||||||
config, ('probe_count', '3'), check=False, cast=int, minval=3)
|
if shape == BED_SHAPES['round']:
|
||||||
|
x_cnt = y_cnt = config.getint('probe_count', 5)
|
||||||
|
# round beds must have an odd number of points along each axis
|
||||||
|
if not x_cnt & 1:
|
||||||
|
raise config.error(
|
||||||
|
"bed_mesh: probe_count must be odd for round beds")
|
||||||
|
self.radius = config.getfloat('radius', above=0.)
|
||||||
|
# radius may have precision to .1mm
|
||||||
|
self.radius = math.floor(self.radius * 10) / 10
|
||||||
|
min_x = min_y = -self.radius
|
||||||
|
max_x = max_y = self.radius
|
||||||
|
else:
|
||||||
|
# rectangular
|
||||||
|
x_cnt, y_cnt = parse_pair(
|
||||||
|
config, ('probe_count', '3'), check=False, cast=int, minval=3)
|
||||||
|
min_x, min_y = parse_pair(config, ('min_point',))
|
||||||
|
max_x, max_y = parse_pair(config, ('max_point',))
|
||||||
|
if max_x <= min_x or max_y <= min_y:
|
||||||
|
raise config.error('bed_mesh: invalid min/max points')
|
||||||
|
|
||||||
self.probe_params['x_count'] = x_cnt
|
self.probe_params['x_count'] = x_cnt
|
||||||
self.probe_params['y_count'] = y_cnt
|
self.probe_params['y_count'] = y_cnt
|
||||||
min_x, min_y = parse_pair(config, ('min_point',))
|
|
||||||
max_x, max_y = parse_pair(config, ('max_point',))
|
|
||||||
if max_x <= min_x or max_y <= min_y:
|
|
||||||
raise config.error('bed_mesh: invalid min/max points')
|
|
||||||
x_dist = (max_x - min_x) / (x_cnt - 1)
|
x_dist = (max_x - min_x) / (x_cnt - 1)
|
||||||
y_dist = (max_y - min_y) / (y_cnt - 1)
|
y_dist = (max_y - min_y) / (y_cnt - 1)
|
||||||
# floor distances down to next hundredth
|
# floor distances down to next hundredth
|
||||||
|
@ -200,8 +218,16 @@ class BedMeshCalibrate:
|
||||||
y_dist = math.floor(y_dist * 100) / 100
|
y_dist = math.floor(y_dist * 100) / 100
|
||||||
if x_dist <= 1. or y_dist <= 1.:
|
if x_dist <= 1. or y_dist <= 1.:
|
||||||
raise config.error("bed_mesh: min/max points too close together")
|
raise config.error("bed_mesh: min/max points too close together")
|
||||||
# re-calc x_max
|
|
||||||
max_x = min_x + x_dist * (x_cnt - 1)
|
if self.radius is not None:
|
||||||
|
# round bed, min/max needs to be recalculated
|
||||||
|
y_dist = x_dist
|
||||||
|
new_r = (x_cnt / 2) * x_dist
|
||||||
|
min_x = min_y = -new_r
|
||||||
|
max_x = max_y = new_r
|
||||||
|
else:
|
||||||
|
# rectangular bed, only re-calc max_x
|
||||||
|
max_x = min_x + x_dist * (x_cnt - 1)
|
||||||
pos_y = min_y
|
pos_y = min_y
|
||||||
points = []
|
points = []
|
||||||
for i in range(y_cnt):
|
for i in range(y_cnt):
|
||||||
|
@ -212,7 +238,14 @@ class BedMeshCalibrate:
|
||||||
else:
|
else:
|
||||||
# move in negative direction
|
# move in negative direction
|
||||||
pos_x = max_x - j * x_dist
|
pos_x = max_x - j * x_dist
|
||||||
points.append((pos_x, pos_y))
|
if self.radius is None:
|
||||||
|
# rectangular bed, append
|
||||||
|
points.append((pos_x, pos_y))
|
||||||
|
else:
|
||||||
|
# round bed, check distance from origin
|
||||||
|
dist_from_origin = math.sqrt(pos_x*pos_x + pos_y*pos_y)
|
||||||
|
if dist_from_origin <= self.radius:
|
||||||
|
points.append((pos_x, pos_y))
|
||||||
pos_y += y_dist
|
pos_y += y_dist
|
||||||
logging.info('bed_mesh: generated points')
|
logging.info('bed_mesh: generated points')
|
||||||
for p in points:
|
for p in points:
|
||||||
|
@ -358,27 +391,58 @@ class BedMeshCalibrate:
|
||||||
z_offset = offsets[2]
|
z_offset = offsets[2]
|
||||||
x_cnt = self.probe_params['x_count']
|
x_cnt = self.probe_params['x_count']
|
||||||
y_cnt = self.probe_params['y_count']
|
y_cnt = self.probe_params['y_count']
|
||||||
# create a 2-D array representing the probed z-positions.
|
|
||||||
self.probed_z_table = [
|
self.probed_z_table = []
|
||||||
[0. for i in range(x_cnt)] for j in range(y_cnt)]
|
row = []
|
||||||
# Check for multi-sampled points
|
prev_pos = positions[0]
|
||||||
z_table_len = x_cnt * y_cnt
|
for pos in positions:
|
||||||
if len(positions) != z_table_len:
|
if not isclose(pos[1], prev_pos[1], abs_tol=.1):
|
||||||
raise self.gcode.error(
|
# y has changed, append row and start new
|
||||||
("bed_mesh: Invalid probe table length:\n"
|
self.probed_z_table.append(row)
|
||||||
"Sampled table length: %d") % len(positions))
|
row = []
|
||||||
# Populate the organized probed table
|
if pos[0] > prev_pos[0]:
|
||||||
for i in range(z_table_len):
|
# probed in the positive direction
|
||||||
y_position = i / x_cnt
|
row.append(pos[2] - z_offset)
|
||||||
x_position = 0
|
|
||||||
if y_position & 1 == 0:
|
|
||||||
# Even y count, x probed in positive directon
|
|
||||||
x_position = i % x_cnt
|
|
||||||
else:
|
else:
|
||||||
# Odd y count, x probed in the negative directon
|
# probed in the negative direction
|
||||||
x_position = (x_cnt - 1) - (i % x_cnt)
|
row.insert(0, pos[2] - z_offset)
|
||||||
self.probed_z_table[y_position][x_position] = \
|
prev_pos = pos
|
||||||
positions[i][2] - z_offset
|
# append last row
|
||||||
|
self.probed_z_table.append(row)
|
||||||
|
|
||||||
|
# make sure the y-axis is the correct length
|
||||||
|
if len(self.probed_z_table) != y_cnt:
|
||||||
|
raise self.gcode.error(
|
||||||
|
("bed_mesh: Invalid y-axis table length\n"
|
||||||
|
"Probed table length: %d Probed Table:\n%s") %
|
||||||
|
(len(self.probed_z_table), str(self.probed_z_table)))
|
||||||
|
|
||||||
|
if self.radius is not None:
|
||||||
|
# round bed, extrapolate probed values to create a square mesh
|
||||||
|
for row in self.probed_z_table:
|
||||||
|
row_size = len(row)
|
||||||
|
if not row_size & 1:
|
||||||
|
# an even number of points in a row shouldn't be possible
|
||||||
|
msg = "bed_mesh: incorrect number of points sampled on X\n"
|
||||||
|
msg += "Probed Table:\n"
|
||||||
|
msg += str(self.probed_z_table)
|
||||||
|
raise self.gcode.error(msg)
|
||||||
|
buf_cnt = (x_cnt - row_size) / 2
|
||||||
|
if buf_cnt == 0:
|
||||||
|
continue
|
||||||
|
left_buffer = [row[0]] * buf_cnt
|
||||||
|
right_buffer = [row[row_size-1]] * buf_cnt
|
||||||
|
row[0:0] = left_buffer
|
||||||
|
row.extend(right_buffer)
|
||||||
|
|
||||||
|
# make sure that the x-axis is the correct length
|
||||||
|
for row in self.probed_z_table:
|
||||||
|
if len(row) != x_cnt:
|
||||||
|
raise self.gcode.error(
|
||||||
|
("bed_mesh: invalid x-axis table length\n"
|
||||||
|
"Probed table length: %d Probed Table:\n%s") %
|
||||||
|
(len(self.probed_z_table), str(self.probed_z_table)))
|
||||||
|
|
||||||
if self.build_map:
|
if self.build_map:
|
||||||
params = self.probe_params
|
params = self.probe_params
|
||||||
outdict = {
|
outdict = {
|
||||||
|
|
Loading…
Reference in New Issue