build123d/docs/topology_selection.rst
2025-09-09 23:21:05 -04:00

432 lines
16 KiB
ReStructuredText

#####################################
Topology Selection and Exploration
#####################################
:ref:`topology` is the structure of build123d geometric features and traversing the
topology of a part is often required to specify objects for an operation or to locate a
CAD feature. :ref:`selectors` allow selection of topology objects into a |ShapeList|.
:ref:`operators` are powerful methods further explore and refine a |ShapeList| for
subsequent operations.
.. _selectors:
*********
Selectors
*********
Selectors provide methods to extract all or a subset of a feature type in the referenced
object. These methods select Edges, Faces, Solids, Vertices, or Wires in Builder objects
or from Shape objects themselves. All of these methods return a |ShapeList|,
which is a subclass of ``list`` and may be sorted, grouped, or filtered by
:ref:`operators`.
Overview
========
+--------------+----------------+-----------------------------------------------+-----------------------+
| Selector | Criteria | Applicability | Description |
+==============+================+===============================================+=======================+
| |vertices| | ALL, LAST | ``BuildLine``, ``BuildSketch``, ``BuildPart`` | ``Vertex`` extraction |
+--------------+----------------+-----------------------------------------------+-----------------------+
| |edges| | ALL, LAST, NEW | ``BuildLine``, ``BuildSketch``, ``BuildPart`` | ``Edge`` extraction |
+--------------+----------------+-----------------------------------------------+-----------------------+
| |wires| | ALL, LAST | ``BuildLine``, ``BuildSketch``, ``BuildPart`` | ``Wire`` extraction |
+--------------+----------------+-----------------------------------------------+-----------------------+
| |faces| | ALL, LAST | ``BuildSketch``, ``BuildPart`` | ``Face`` extraction |
+--------------+----------------+-----------------------------------------------+-----------------------+
| |solids| | ALL, LAST | ``BuildPart`` | ``Solid`` extraction |
+--------------+----------------+-----------------------------------------------+-----------------------+
Both shape objects and builder objects have access to selector methods to select all of
a feature as long as they can contain the feature being selected.
.. code-block:: build123d
# In context
with BuildSketch() as context:
Rectangle(1, 1)
context.edges()
# Build context implicitly has access to the selector
edges()
# Taking the sketch out of context
context.sketch.edges()
# Create sketch out of context
Rectangle(1, 1).edges()
Select In Build Context
========================
Build contexts track the last operation and their selector methods can take
:class:`~build_enums.Select` as criteria to specify a subset of
features to extract. By default, a selector will select ``ALL`` of a feature, while
``LAST`` selects features created or altered by the most recent operation. |edges| can
uniquely specify ``NEW`` to only select edges created in the last operation which neither
existed in the referenced object before the last operation, nor the modifying object.
.. important::
:class:`~build_enums.Select` as selector criteria is only valid for builder objects!
.. code-block:: build123d
# In context
with BuildPart() as context:
Box(2, 2, 1)
Cylinder(1, 2)
context.edges(Select.LAST)
# Does not work out of context!
context.part.edges(Select.LAST)
(Box(2, 2, 1) + Cylinder(1, 2)).edges(Select.LAST)
Create a simple part to demonstrate selectors. Select using the default criteria
``Select.ALL``. Specifying ``Select.ALL`` for the selector is not required.
.. code-block:: build123d
with BuildPart() as part:
Box(5, 5, 1)
Cylinder(1, 5)
part.vertices()
part.edges()
part.faces()
# Is the same as
part.vertices(Select.ALL)
part.edges(Select.ALL)
part.faces(Select.ALL)
.. figure:: assets/topology_selection/selectors_select_all.png
:align: center
The default ``Select.ALL`` features
Select features changed in the last operation with criteria ``Select.LAST``.
.. code-block:: build123d
with BuildPart() as part:
Box(5, 5, 1)
Cylinder(1, 5)
part.vertices(Select.LAST)
part.edges(Select.LAST)
part.faces(Select.LAST)
.. figure:: assets/topology_selection/selectors_select_last.png
:align: center
``Select.LAST`` features
Select only new edges from the last operation with ``Select.NEW``. This option is only
available for a ``ShapeList`` of edges!
.. code-block:: build123d
with BuildPart() as part:
Box(5, 5, 1)
Cylinder(1, 5)
part.edges(Select.NEW)
.. figure:: assets/topology_selection/selectors_select_new.png
:align: center
``Select.NEW`` edges where box and cylinder intersect
This only returns new edges which are not reused from Box or Cylinder, in this case where
the objects `intersect`. But what happens if the objects don't intersect and all the
edges are reused?
.. code-block:: build123d
with BuildPart() as part:
Box(5, 5, 1, align=(Align.CENTER, Align.CENTER, Align.MAX))
Cylinder(2, 2, align=(Align.CENTER, Align.CENTER, Align.MIN))
part.edges(Select.NEW)
.. figure:: assets/topology_selection/selectors_select_new_none.png
:align: center
``Select.NEW`` edges when box and cylinder don't intersect
No edges are selected! Unlike the previous example, the Edge between the Box and Cylinder
objects is an edge reused from the Cylinder. Think of ``Select.NEW`` as a way to select
only completely new edges created by the operation.
.. note::
Chamfer and fillet modify the current object, but do not have new edges via
``Select.NEW``.
.. code-block:: build123d
with BuildPart() as part:
Box(5, 5, 1)
Cylinder(1, 5)
edges = part.edges().filter_by(lambda a: a.length == 1)
fillet(edges, 1)
part.edges(Select.NEW)
.. figure:: assets/topology_selection/selectors_select_new_fillet.png
:align: center
Left, ``Select.NEW`` returns no edges after fillet. Right, ``Select.LAST``
Select New Edges In Algebra Mode
================================
The utility method ``new_edges`` compares one or more shape objects to a
another "combined" shape object and returns the edges new to the combined shape.
``new_edges`` is available both Algebra mode or Builder mode, but is necessary in
Algebra Mode where ``Select.NEW`` is unavailable
.. code-block:: build123d
box = Box(5, 5, 1)
circle = Cylinder(2, 5)
part = box + circle
edges = new_edges(box, circle, combined=part)
.. figure:: assets/topology_selection/selectors_new_edges.png
:align: center
``new_edges`` can also find edges created during a chamfer or fillet operation by
comparing the object before the operation to the "combined" object.
.. code-block:: build123d
box = Box(5, 5, 1)
circle = Cylinder(2, 5)
part_before = box + circle
edges = part_before.edges().filter_by(lambda a: a.length == 1)
part = fillet(edges, 1)
edges = new_edges(part_before, combined=part)
.. figure:: assets/topology_selection/operators_group_area.png
:align: center
.. _operators:
*********
Operators
*********
Operators provide methods refine a ``ShapeList`` of features isolated by a *selector* to
further specify feature(s). These methods can sort, group, or filter ``ShapeList``
objects and return a modified ``ShapeList``, or in the case of |group_by|, ``GroupBy``,
a list of ``ShapeList`` objects accessible by index or key.
Overview
========
+----------------------+------------------------------------------------------------------+-------------------------------------------------------+
| Method | Criteria | Description |
+======================+==================================================================+=======================================================+
| |sort_by| | ``Axis``, ``Edge``, ``Wire``, ``SortBy``, callable, property | Sort ``ShapeList`` by criteria |
+----------------------+------------------------------------------------------------------+-------------------------------------------------------+
| |sort_by_distance| | ``Shape``, ``VectorLike`` | Sort ``ShapeList`` by distance from criteria |
+----------------------+------------------------------------------------------------------+-------------------------------------------------------+
| |group_by| | ``Axis``, ``Edge``, ``Wire``, ``SortBy``, callable, property | Group ``ShapeList`` by criteria |
+----------------------+------------------------------------------------------------------+-------------------------------------------------------+
| |filter_by| | ``Axis``, ``Plane``, ``GeomType``, ``ShapePredicate``, property | Filter ``ShapeList`` by criteria |
+----------------------+------------------------------------------------------------------+-------------------------------------------------------+
| |filter_by_position| | ``Axis`` | Filter ``ShapeList`` by ``Axis`` & mix / max values |
+----------------------+------------------------------------------------------------------+-------------------------------------------------------+
Operator methods take criteria to refine ``ShapeList``. Broadly speaking, the criteria
fall into the following categories, though not all operators take all criteria:
- Geometric objects: ``Axis``, ``Plane``
- Topological objects: ``Edge``, ``Wire``
- Enums: :class:`~build_enums.SortBy`, :class:`~build_enums.GeomType`
- Properties, eg: ``Face.area``, ``Edge.length``
- ``ShapePredicate``, eg: ``lambda e: e.is_interior == 1``, ``lambda f: lf.edges() >= 3``
- Callable eg: ``Vertex().distance``
Sort
=======
A ``ShapeList`` can be sorted with the |sort_by| and |sort_by_distance|
methods based on a sorting criteria. Sorting is a critical step when isolating individual
features as a ``ShapeList`` from a selector is typically unordered.
Here we want to capture some vertices from the object furthest along ``X``: All the
vertices are first captured with the |vertices| selector, then sort by ``Axis.X``.
Finally, the vertices can be captured with a list slice for the last 4 list items, as the
items are sorted from least to greatest ``X`` position. Remember, ``ShapeList`` is a
subclass of ``list``, so any list slice can be used.
.. code-block:: build123d
part.vertices().sort_by(Axis.X)[-4:]
.. figure:: assets/topology_selection/operators_sort_x.png
:align: center
|
Examples
--------
.. toctree::
:maxdepth: 2
:hidden:
topology_selection/sort_examples
.. grid:: 3
:gutter: 3
.. grid-item-card:: SortBy
:img-top: assets/topology_selection/thumb_sort_sortby.png
:link: sort_sortby
:link-type: ref
.. grid-item-card:: Along Wire
:img-top: assets/topology_selection/thumb_sort_along_wire.png
:link: sort_along_wire
:link-type: ref
.. grid-item-card:: Axis
:img-top: assets/topology_selection/thumb_sort_axis.png
:link: sort_axis
:link-type: ref
.. grid-item-card:: Distance From
:img-top: assets/topology_selection/thumb_sort_distance.png
:link: sort_distance_from
:link-type: ref
Group
========
A ShapeList can be grouped and sorted with the |group_by| method based on a grouping
criteria. Grouping can be a great way to organize features without knowing the values of
specific feature properties. Rather than returning a ``ShapeList``, |group_by| returns
a ``GroupBy``, a list of ``ShapeList`` objects sorted by the grouping criteria.
``GroupBy`` can be printed to view the members of each group, indexed like a list to
retrieve a ``ShapeList``, and be accessed using a key with the ``group`` method. If the
group keys are unknown they can be discovered with ``key_to_group_index``.
If we want only the edges from the smallest faces by area we can get the faces, then
group by ``SortBy.AREA``. The ``ShapeList`` of smallest faces is available from the first
list index. Finally, a ``ShapeList`` has access to selectors, so calling |edges| will
return a new list of all edges in the previous list.
.. code-block:: build123d
part.faces().group_by(SortBy.AREA)[0].edges())
.. figure:: assets/topology_selection/operators_group_area.png
:align: center
|
Examples
--------
.. toctree::
:maxdepth: 2
:hidden:
topology_selection/group_examples
.. grid:: 3
:gutter: 3
.. grid-item-card:: Axis and Length
:img-top: assets/topology_selection/thumb_group_axis.png
:link: group_axis
:link-type: ref
.. grid-item-card:: Hole Area
:img-top: assets/topology_selection/thumb_group_hole_area.png
:link: group_hole_area
:link-type: ref
.. grid-item-card:: Properties with Keys
:img-top: assets/topology_selection/thumb_group_properties_with_keys.png
:link: group_properties_with_keys
:link-type: ref
Filter
=========
A ``ShapeList`` can be filtered with the |filter_by| and |filter_by_position| methods based
on a filtering criteria. Filters are flexible way to isolate (or exclude) features based
on known criteria.
Lets say we need all the faces with a normal in the ``+Z`` direction. One way to do this
might be with a list comprehension, however |filter_by| has the capability to take a
lambda function as a filter condition on the entire list. In this case, the normal of
each face can be checked against a vector direction and filtered accordingly.
.. code-block:: build123d
part.faces().filter_by(lambda f: f.normal_at() == Vector(0, 0, 1))
.. figure:: assets/topology_selection/operators_filter_z_normal.png
:align: center
|
Examples
--------
.. toctree::
:maxdepth: 2
:hidden:
topology_selection/filter_examples
.. grid:: 3
:gutter: 3
.. grid-item-card:: GeomType
:img-top: assets/topology_selection/thumb_filter_geomtype.png
:link: filter_geomtype
:link-type: ref
.. grid-item-card:: All Edges Circle
:img-top: assets/topology_selection/thumb_filter_all_edges_circle.png
:link: filter_all_edges_circle
:link-type: ref
.. grid-item-card:: Axis and Plane
:img-top: assets/topology_selection/thumb_filter_axisplane.png
:link: filter_axis_plane
:link-type: ref
.. grid-item-card:: Inner Wire Count
:img-top: assets/topology_selection/thumb_filter_inner_wire_count.png
:link: filter_inner_wire_count
:link-type: ref
.. grid-item-card:: Nested Filters
:img-top: assets/topology_selection/thumb_filter_nested.png
:link: filter_nested
:link-type: ref
.. grid-item-card:: Shape Properties
:img-top: assets/topology_selection/thumb_filter_shape_properties.png
:link: filter_shape_properties
:link-type: ref
.. |vertices| replace:: :meth:`~topology.Shape.vertices`
.. |edges| replace:: :meth:`~topology.Shape.edges`
.. |wires| replace:: :meth:`~topology.Shape.wires`
.. |faces| replace:: :meth:`~topology.Shape.faces`
.. |solids| replace:: :meth:`~topology.Shape.solids`
.. |sort_by| replace:: :meth:`~topology.ShapeList.sort_by`
.. |sort_by_distance| replace:: :meth:`~topology.ShapeList.sort_by_distance`
.. |group_by| replace:: :meth:`~topology.ShapeList.group_by`
.. |filter_by| replace:: :meth:`~topology.ShapeList.filter_by`
.. |filter_by_position| replace:: :meth:`~topology.ShapeList.filter_by_position`
.. |ShapeList| replace:: :class:`~topology.ShapeList`