mirror of
https://github.com/Ultimaker/Cura.git
synced 2025-07-16 11:17:49 -06:00
added creating snapshot if not created till 10 attempts
refractored the snapshot.py CURA-11650
This commit is contained in:
parent
dc031c4674
commit
ebe5da7f0e
1 changed files with 45 additions and 21 deletions
|
@ -21,23 +21,31 @@ from UM.Scene.SceneNode import SceneNode
|
||||||
from UM.Qt.QtRenderer import QtRenderer
|
from UM.Qt.QtRenderer import QtRenderer
|
||||||
|
|
||||||
class Snapshot:
|
class Snapshot:
|
||||||
|
|
||||||
|
DEFAULT_WIDTH_HEIGHT = 300
|
||||||
|
MAX_RENDER_DISTANCE = 10000
|
||||||
|
BOUND_BOX_FACTOR = 1.75
|
||||||
|
CAMERA_FOVY = 30
|
||||||
|
ATTEMPTS_FOR_SNAPSHOT = 10
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def getImageBoundaries(image: QImage):
|
def getNonZeroPixels(image: QImage):
|
||||||
# Look at the resulting image to get a good crop.
|
|
||||||
# Get the pixels as byte array
|
|
||||||
pixel_array = image.bits().asarray(image.sizeInBytes())
|
pixel_array = image.bits().asarray(image.sizeInBytes())
|
||||||
width, height = image.width(), image.height()
|
width, height = image.width(), image.height()
|
||||||
# Convert to numpy array, assume it's 32 bit (it should always be)
|
|
||||||
pixels = numpy.frombuffer(pixel_array, dtype=numpy.uint8).reshape([height, width, 4])
|
pixels = numpy.frombuffer(pixel_array, dtype=numpy.uint8).reshape([height, width, 4])
|
||||||
# Find indices of non zero pixels
|
# Find indices of non zero pixels
|
||||||
nonzero_pixels = numpy.nonzero(pixels)
|
return numpy.nonzero(pixels)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def getImageBoundaries(image: QImage):
|
||||||
|
nonzero_pixels = Snapshot.getNonZeroPixels(image)
|
||||||
min_y, min_x, min_a_ = numpy.amin(nonzero_pixels, axis=1) # type: ignore
|
min_y, min_x, min_a_ = numpy.amin(nonzero_pixels, axis=1) # type: ignore
|
||||||
max_y, max_x, max_a_ = numpy.amax(nonzero_pixels, axis=1) # type: ignore
|
max_y, max_x, max_a_ = numpy.amax(nonzero_pixels, axis=1) # type: ignore
|
||||||
|
|
||||||
return min_x, max_x, min_y, max_y
|
return min_x, max_x, min_y, max_y
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def isometricSnapshot(width: int = 300, height: int = 300, *, node: Optional[SceneNode] = None) -> Optional[QImage]:
|
def isometricSnapshot(width: int = DEFAULT_WIDTH_HEIGHT, height: int = DEFAULT_WIDTH_HEIGHT, *, node: Optional[SceneNode] = None) -> Optional[QImage]:
|
||||||
"""
|
"""
|
||||||
Create an isometric snapshot of the scene.
|
Create an isometric snapshot of the scene.
|
||||||
|
|
||||||
|
@ -112,22 +120,25 @@ class Snapshot:
|
||||||
|
|
||||||
return render_pass.getOutput()
|
return render_pass.getOutput()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def isNodeRenderable(node):
|
||||||
|
return not getattr(node, "_outside_buildarea", False) and node.callDecoration(
|
||||||
|
"isSliceable") and node.getMeshData() and node.isVisible() and not node.callDecoration(
|
||||||
|
"isNonThumbnailVisibleMesh")
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def nodeBounds(root_node: SceneNode) -> Optional[AxisAlignedBox]:
|
def nodeBounds(root_node: SceneNode) -> Optional[AxisAlignedBox]:
|
||||||
axis_aligned_box = None
|
axis_aligned_box = None
|
||||||
for node in DepthFirstIterator(root_node):
|
for node in DepthFirstIterator(root_node):
|
||||||
if not getattr(node, "_outside_buildarea", False):
|
if Snapshot.isNodeRenderable(node):
|
||||||
if node.callDecoration(
|
if axis_aligned_box is None:
|
||||||
"isSliceable") and node.getMeshData() and node.isVisible() and not node.callDecoration(
|
axis_aligned_box = node.getBoundingBox()
|
||||||
"isNonThumbnailVisibleMesh"):
|
else:
|
||||||
if axis_aligned_box is None:
|
axis_aligned_box = axis_aligned_box + node.getBoundingBox()
|
||||||
axis_aligned_box = node.getBoundingBox()
|
|
||||||
else:
|
|
||||||
axis_aligned_box = axis_aligned_box + node.getBoundingBox()
|
|
||||||
return axis_aligned_box
|
return axis_aligned_box
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def snapshot(width = 300, height = 300):
|
def snapshot(width = DEFAULT_WIDTH_HEIGHT, height = DEFAULT_WIDTH_HEIGHT, number_of_attempts = ATTEMPTS_FOR_SNAPSHOT):
|
||||||
"""Return a QImage of the scene
|
"""Return a QImage of the scene
|
||||||
|
|
||||||
Uses PreviewPass that leaves out some elements Aspect ratio assumes a square
|
Uses PreviewPass that leaves out some elements Aspect ratio assumes a square
|
||||||
|
@ -163,13 +174,13 @@ class Snapshot:
|
||||||
looking_from_offset = Vector(-1, 1, 2)
|
looking_from_offset = Vector(-1, 1, 2)
|
||||||
if size > 0:
|
if size > 0:
|
||||||
# determine the watch distance depending on the size
|
# determine the watch distance depending on the size
|
||||||
looking_from_offset = looking_from_offset * size * 1.75
|
looking_from_offset = looking_from_offset * size * Snapshot.BOUND_BOX_FACTOR
|
||||||
camera.setPosition(look_at + looking_from_offset)
|
camera.setPosition(look_at + looking_from_offset)
|
||||||
camera.lookAt(look_at)
|
camera.lookAt(look_at)
|
||||||
|
|
||||||
satisfied = False
|
satisfied = False
|
||||||
size = None
|
size = None
|
||||||
fovy = 30
|
fovy = Snapshot.CAMERA_FOVY
|
||||||
|
|
||||||
while not satisfied:
|
while not satisfied:
|
||||||
if size is not None:
|
if size is not None:
|
||||||
|
@ -182,11 +193,19 @@ class Snapshot:
|
||||||
preview_pass.setCamera(camera)
|
preview_pass.setCamera(camera)
|
||||||
preview_pass.render()
|
preview_pass.render()
|
||||||
pixel_output = preview_pass.getOutput()
|
pixel_output = preview_pass.getOutput()
|
||||||
try:
|
if Snapshot._prereadSnapshot(pixel_output):
|
||||||
min_x, max_x, min_y, max_y = Snapshot.getImageBoundaries(pixel_output)
|
try:
|
||||||
except (ValueError, AttributeError):
|
min_x, max_x, min_y, max_y = Snapshot.getImageBoundaries(pixel_output)
|
||||||
Logger.logException("w", "Failed to crop the snapshot!")
|
except (ValueError, AttributeError) as e:
|
||||||
|
Logger.logException("w", f"Failed to crop the snapshot! {e}")
|
||||||
|
return None
|
||||||
|
elif number_of_attempts == 0:
|
||||||
|
Logger.warning( f"Failed to crop the snapshot even after 10 attempts!")
|
||||||
return None
|
return None
|
||||||
|
else:
|
||||||
|
number_of_attempts = number_of_attempts - 1
|
||||||
|
Logger.info("Trying to get the snapshot again.")
|
||||||
|
return Snapshot.snapshot(width, height, number_of_attempts)
|
||||||
|
|
||||||
size = max((max_x - min_x) / render_width, (max_y - min_y) / render_height)
|
size = max((max_x - min_x) / render_width, (max_y - min_y) / render_height)
|
||||||
if size > 0.5 or satisfied:
|
if size > 0.5 or satisfied:
|
||||||
|
@ -211,3 +230,8 @@ class Snapshot:
|
||||||
transformMode = QtCore.Qt.TransformationMode.SmoothTransformation)
|
transformMode = QtCore.Qt.TransformationMode.SmoothTransformation)
|
||||||
|
|
||||||
return scaled_image
|
return scaled_image
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _prereadSnapshot(snap = QImage):
|
||||||
|
nonzero_pixels = Snapshot.getNonZeroPixels(snap)
|
||||||
|
return all(len(nonzero_pixels[i]) != 0 for i in range(3))
|
Loading…
Add table
Add a link
Reference in a new issue