3 @brief Cluster analysis algorithm: BANG.
4 @details Implementation based on paper @cite inproceedings::bang::1.
6 @authors Andrei Novikov (pyclustering@yandex.ru)
8 @copyright BSD-3-Clause
16 import matplotlib.gridspec
as gridspec
17 import matplotlib.pyplot
as plt
18 import matplotlib.patches
as patches
19 import matplotlib.animation
as animation
31 @brief Visualizer of BANG algorithm's results.
32 @details BANG visualizer provides visualization services that are specific for BANG algorithm.
36 __maximum_density_alpha = 0.6
42 @brief Show BANG-blocks (leafs only) in data space.
43 @details BANG-blocks represents grid that was used for clustering process.
45 @param[in] directory (bang_directory): Directory that was created by BANG algorithm during clustering process.
49 dimension = len(directory.get_data()[0])
53 amount_canvases = int(dimension * (dimension - 1) / 2)
56 grid_spec = gridspec.GridSpec(1, amount_canvases)
58 pairs = list(itertools.combinations(range(dimension), 2))
59 if len(pairs) == 0: pairs = [(0, 0)]
61 for index
in range(amount_canvases):
62 ax = figure.add_subplot(grid_spec[index])
63 bang_visualizer.__draw_blocks(ax, directory.get_leafs(), pairs[index])
64 bang_visualizer.__draw_two_dimension_data(ax, directory.get_data(), pairs[index])
72 @brief Display dendrogram of BANG-blocks.
74 @param[in] dendrogram (list): List representation of dendrogram of BANG-blocks.
76 @see bang.get_dendrogram()
80 axis = plt.subplot(1, 1, 1)
83 for index_cluster
in range(len(dendrogram)):
84 densities = [ block.get_density()
for block
in dendrogram[index_cluster] ]
85 xrange = range(current_position, current_position + len(densities))
87 axis.bar(xrange, densities, 1.0, linewidth=0.0, color=color_list.get_color(index_cluster))
89 current_position += len(densities)
91 axis.set_ylabel(
"density")
92 axis.set_xlabel(
"block")
93 axis.xaxis.set_ticklabels([])
95 plt.xlim([-0.5, current_position - 0.5])
102 @brief Display BANG clustering results.
104 @param[in] data (list): Dataset that was used for clustering.
105 @param[in] clusters (array_like): Clusters that were allocated by the algorithm.
106 @param[in] noise (array_like): Noise that were allocated by the algorithm.
110 visualizer.append_clusters(clusters, data)
111 visualizer.append_cluster(noise
or [], data, marker=
'x')
116 def __draw_two_dimension_data(ax, data, pair):
118 @brief Display data in two-dimensional canvas.
120 @param[in] ax (Axis): Canvas where data should be displayed.
121 @param[in] data (list): Data points that should be displayed.
122 @param[in] pair (tuple): Pair of dimension indexes.
125 ax.set_xlabel(
"x%d" % pair[0])
126 ax.set_ylabel(
"x%d" % pair[1])
130 ax.plot(point[pair[0]], point[pair[1]], color=
'red', marker=
'.')
132 ax.plot(point[pair[0]], 0, color=
'red', marker=
'.')
133 ax.yaxis.set_ticklabels([])
137 def __draw_blocks(ax, blocks, pair):
139 @brief Display BANG-blocks on specified figure.
141 @param[in] ax (Axis): Axis where bang-blocks should be displayed.
142 @param[in] blocks (list): List of blocks that should be displyed.
143 @param[in] pair (tuple): Pair of coordinate index that should be displayed.
148 density_scale = blocks[-1].get_density()
150 bang_visualizer.__draw_block(ax, pair, block, density_scale)
154 def __draw_block(ax, pair, block, density_scale):
156 @brief Display BANG-block on the specified ax.
158 @param[in] ax (Axis): Axis where block should be displayed.
159 @param[in] pair (tuple): Pair of coordinate index that should be displayed.
160 @param[in] block (bang_block): BANG-block that should be displayed.
161 @param[in] density_scale (double): Max density to display density of the block by appropriate tone.
164 max_corner, min_corner = bang_visualizer.__get_rectangle_description(block, pair)
166 belong_cluster = block.get_cluster()
is not None
168 if density_scale != 0.0:
169 density_scale = bang_visualizer.__maximum_density_alpha * block.get_density() / density_scale
171 face_color = matplotlib.colors.to_rgba(
'blue', alpha=density_scale)
172 edge_color = matplotlib.colors.to_rgba(
'black', alpha=1.0)
174 rect = patches.Rectangle(min_corner, max_corner[0] - min_corner[0], max_corner[1] - min_corner[1],
176 facecolor=face_color,
177 edgecolor=edge_color,
183 def __get_rectangle_description(block, pair):
185 @brief Create rectangle description for block in specific dimension.
187 @param[in] pair (tuple): Pair of coordinate index that should be displayed.
188 @param[in] block (bang_block): BANG-block that should be displayed
190 @return (tuple) Pair of corners that describes rectangle.
193 max_corner, min_corner = block.get_spatial_block().get_corners()
195 max_corner = [max_corner[pair[0]], max_corner[pair[1]]]
196 min_corner = [min_corner[pair[0]], min_corner[pair[1]]]
199 max_corner[1], min_corner[1] = 1.0, -1.0
201 return max_corner, min_corner
206 @brief Provides service for creating 2-D animation using BANG clustering results.
207 @details The animator does not support visualization of clustering process where non 2-dimensional was used.
209 Code example of animation of BANG clustering process:
211 from pyclustering.cluster.bang import bang, bang_animator
212 from pyclustering.utils import read_sample
213 from pyclustering.samples.definitions import FCPS_SAMPLES
215 # Read data two dimensional data.
216 data = read_sample(FCPS_SAMPLES.SAMPLE_LSUN)
218 # Create instance of BANG algorithm.
219 bang_instance = bang(data, 9)
220 bang_instance.process()
222 # Obtain clustering results.
223 clusters = bang_instance.get_clusters()
224 noise = bang_instance.get_noise()
225 directory = bang_instance.get_directory()
227 # Create BANG animation using class 'bang_animator':
228 animator = bang_animator(directory, clusters)
235 @brief Creates BANG animator instance.
237 @param[in] directory (bang_directory): BANG directory that was formed during BANG clustering process.
238 @param[in] clusters (list): Allocated clusters during BANG clustering process.
256 def __validate_arguments(self):
258 @brief Check correctness of input arguments and throw exception if incorrect is found.
262 raise ValueError(
"Impossible to animate BANG clustering process for non 2D data.")
265 def __increment_block(self):
267 @brief Increment BANG block safely by updating block index, level and level block.
279 def __draw_block(self, block, block_alpha=0.0):
281 @brief Display single BANG block on axis.
283 @param[in] block (bang_block): BANG block that should be displayed.
284 @param[in] block_alpha (double): Transparency level - value of alpha.
287 max_corner, min_corner = block.get_spatial_block().get_corners()
289 face_color = matplotlib.colors.to_rgba(
'blue', alpha=block_alpha)
290 edge_color = matplotlib.colors.to_rgba(
'black', alpha=1.0)
292 rect = patches.Rectangle(min_corner, max_corner[0] - min_corner[0], max_corner[1] - min_corner[1],
294 facecolor=face_color,
295 edgecolor=edge_color,
297 self.
__ax.add_patch(rect)
300 def __draw_leaf_density(self):
302 @brief Display densities by filling blocks by appropriate colors.
306 density_scale = leafs[-1].get_density()
308 if density_scale == 0.0: density_scale = 1.0
311 alpha = 0.8 * block.get_density() / density_scale
315 def __draw_clusters(self):
317 @brief Display clusters and outliers using different colors.
321 for index_cluster
in range(len(self.
__clusters)):
322 color = color_list.get_color(index_cluster)
328 def __draw_cluster(self, data, cluster, color, marker):
330 @brief Draw 2-D single cluster on axis using specified color and marker.
334 self.
__ax.plot(data[item][0], data[item][1], color=color, marker=marker)
337 def animate(self, animation_velocity=75, movie_fps=25, movie_filename=None):
339 @brief Animates clustering process that is performed by BANG algorithm.
341 @param[in] animation_velocity (uint): Interval between frames in milliseconds (for run-time animation only).
342 @param[in] movie_fps (uint): Defines frames per second (for rendering movie only).
343 @param[in] movie_filename (string): If it is specified then animation will be stored to file that is specified in this parameter.
349 self.
__figure.suptitle(
"BANG algorithm", fontsize=18, fontweight=
'bold')
352 self.
__ax.plot(point[0], point[1], color=
'red', marker=
'.')
354 return frame_generation(0)
357 def frame_generation(index_iteration):
373 self.
__figure.suptitle(
"BANG algorithm", fontsize=18, fontweight=
'bold')
383 cluster_animation = animation.FuncAnimation(self.
__figure, frame_generation, iterations,
384 interval=animation_velocity,
385 init_func=init_frame,
388 if movie_filename
is not None:
389 cluster_animation.save(movie_filename, writer =
'ffmpeg', fps = movie_fps, bitrate = 3500)
397 @brief BANG directory stores BANG-blocks that represents grid in data space.
398 @details The directory build BANG-blocks in binary tree manner. Leafs of the tree stored separately to provide
399 a direct access to the leafs that should be analysed. Leafs cache data-points.
404 @brief Create BANG directory - basically tree structure with direct access to leafs.
406 @param[in] data (list): Input data that is clustered.
407 @param[in] levels (uint): Height of the tree of blocks.
408 @param[in] **kwargs: Arbitrary keyword arguments (available arguments: 'observe').
410 <b>Keyword Args:</b><br>
411 - observe (bool): If 'True' then blocks on each level are stored.
412 - density_threshold (double): The lowest level of density when contained data in bang-block is
413 considered as a noise and there is no need to split it till the last level. Be aware that this
414 parameter is used with 'amount_threshold' parameter.
415 - amount_threshold (uint): Amount of points in the block when it contained data in bang-block is
416 considered as a noise and there is no need to split it till the last level.
427 self.
__observe = kwargs.get(
'observe',
True)
434 @brief Returns amount of blocks that is stored in the directory
436 @return (uint) Amount of blocks in the BANG directory.
444 @brief Return data that is stored in the directory.
446 @return (list) List of points that represents stored data.
454 @brief Return leafs - the smallest blocks.
455 @details Some leafs can be bigger than others because splitting is not performed for blocks whose density is
458 @return (list) List of blocks that are leafs of BANG directory.
466 @brief Returns BANG blocks on the specific level.
468 @param[in] level (uint): Level of tree where BANG blocks are located.
470 @return (list) List of BANG blocks on the specific level.
478 @brief Returns height of BANG tree where blocks are stored.
480 @return (uint) Height of BANG tree.
486 def __create_directory(self):
488 @brief Create BANG directory as a tree with separate storage for leafs.
492 min_corner, max_corner = data_corners(self.
__data)
495 cache_require = (self.
__levels == 1)
505 def __store_level_blocks(self, level_blocks):
507 @brief Store level blocks if observing is enabled.
509 @param[in] level_blocks (list): Created blocks on a new level.
512 self.
__size += len(level_blocks)
518 def __build_directory_levels(self):
520 @brief Build levels of direction if amount of level is greater than one.
524 previous_level_blocks = [ self.
__root ]
526 for level
in range(1, self.
__levels):
527 previous_level_blocks = self.
__build_level(previous_level_blocks, level)
530 self.
__leafs = sorted(self.
__leafs, key=
lambda block: block.get_density())
533 def __build_level(self, previous_level_blocks, level):
535 @brief Build new level of directory.
537 @param[in] previous_level_blocks (list): BANG-blocks on the previous level.
538 @param[in] level (uint): Level number that should be built.
540 @return (list) New block on the specified level.
543 current_level_blocks = []
545 split_dimension = level % len(self.
__data[0])
546 cache_require = (level == self.
__levels - 1)
548 for block
in previous_level_blocks:
549 self.
__split_block(block, split_dimension, cache_require, current_level_blocks)
552 self.
__leafs += current_level_blocks
554 return current_level_blocks
557 def __split_block(self, block, split_dimension, cache_require, current_level_blocks):
559 @brief Split specific block in specified dimension.
560 @details Split is not performed for block whose density is lower than threshold value, such blocks are putted to
563 @param[in] block (bang_block): BANG-block that should be split.
564 @param[in] split_dimension (uint): Dimension at which splitting should be performed.
565 @param[in] cache_require (bool): Defines when points in cache should be stored during density calculation.
566 @param[in|out] current_level_blocks (list): Block storage at the current level where new blocks should be added.
573 left, right = block.split(split_dimension, cache_require)
574 current_level_blocks.append(left)
575 current_level_blocks.append(right)
580 @brief Geometrical description of BANG block in data space.
581 @details Provides services related to spatial functionality and used by bang_block
589 @brief Creates spatial block in data space.
591 @param[in] max_corner (array_like): Maximum corner coordinates of the block.
592 @param[in] min_corner (array_like): Minimal corner coordinates of the block.
602 @brief Returns string block description.
604 @return String representation of the block.
612 @brief Point is considered as contained if it lies in block (belong to it).
614 @return (bool) True if point is in block, otherwise False.
617 for i
in range(len(point)):
626 @brief Return spatial description of current block.
628 @return (tuple) Pair of maximum and minimum corners (max_corner, min_corner).
636 @brief Returns volume of current block.
637 @details Volume block has uncommon mining here: for 1D is length of a line, for 2D is square of rectangle,
638 for 3D is volume of 3D figure, and for ND is volume of ND figure.
640 @return (double) Volume of current block.
648 @brief Split current block into two spatial blocks in specified dimension.
650 @param[in] dimension (uint): Dimension where current block should be split.
652 @return (tuple) Pair of new split blocks from current block.
660 first_max_corner[dimension] = split_border
661 second_min_corner[dimension] = split_border
668 @brief Performs calculation to identify whether specified block is neighbor of current block.
669 @details It also considers diagonal blocks as neighbors.
671 @param[in] block (spatial_block): Another block that is check whether it is neighbor.
673 @return (bool) True is blocks are neighbors, False otherwise.
676 if block
is not self:
677 block_max_corner, _ = block.get_corners()
678 dimension = len(block_max_corner)
681 if neighborhood_score == dimension:
687 def __calculate_neighborhood(self, block_max_corner):
689 @brief Calculates neighborhood score that defined whether blocks are neighbors.
691 @param[in] block_max_corner (list): Maximum coordinates of other block.
693 @return (uint) Neighborhood score.
696 dimension = len(block_max_corner)
700 neighborhood_score = 0
701 for i
in range(dimension):
704 if diff <= length_edges[i] + length_edges[i] * 0.0001:
705 neighborhood_score += 1
707 return neighborhood_score
710 def __calculate_volume(self):
712 @brief Calculates volume of current spatial block.
713 @details If empty dimension is detected (where all points has the same value) then such dimension is ignored
714 during calculation of volume.
716 @return (double) Volume of current spatial block.
724 if side_length != 0.0:
725 if volume == 0.0: volume = side_length
726 else: volume *= side_length
733 @brief BANG-block that represent spatial region in data space.
736 def __init__(self, data, region, level, space_block, cache_points=False):
738 @brief Create BANG-block.
740 @param[in] data (list): List of points that are processed.
741 @param[in] region (uint): Region number - unique value on a level.
742 @param[in] level (uint): Level number where block is created.
743 @param[in] space_block (spatial_block): Spatial block description in data space.
744 @param[in] cache_points (bool): if True then points are stored in memory (used for leaf blocks).
761 @brief Returns string representation of BANG-block using region number and level where block is located.
769 @brief Returns block size defined by amount of points that are contained by this block.
777 @brief Returns region number of BANG-block.
778 @details Region number is unique on among region numbers on a directory level. Pair of region number and level
779 is unique for all directory.
781 @return (uint) Region number.
789 @brief Returns density of the BANG-block.
791 @return (double) BANG-block density.
799 @brief Return index of cluster to which the BANG-block belongs to.
800 @details Index of cluster may have None value if the block was not assigned to any cluster.
802 @return (uint) Index of cluster or None if the block does not belong to any cluster.
810 @brief Return spatial block - BANG-block description in data space.
812 @return (spatial_block) Spatial block of the BANG-block.
820 @brief Return points that covers by the BANG-block.
822 @return (list) List of point indexes that are covered by the block.
833 @brief Assign cluster to the BANG-block by index.
835 @param[in] index (uint): Index cluster that is assigned to BANG-block.
843 @brief Performs calculation to check whether specified block is neighbor to the current.
845 @param[in] block (bang_block): Other BANG-block that should be checked for neighborhood.
847 @return (bool) True if blocks are neighbors, False if blocks are not neighbors.
853 def split(self, split_dimension, cache_points):
855 @brief Split BANG-block into two new blocks in specified dimension.
857 @param[in] split_dimension (uint): Dimension where block should be split.
858 @param[in] cache_points (bool): If True then covered points are cached. Used for leaf blocks.
860 @return (tuple) Pair of BANG-block that were formed from the current.
874 def __calculate_density(self, amount_points):
876 @brief Calculates BANG-block density.
878 @param[in] amount_points (uint): Amount of points in block.
880 @return (double) BANG-block density.
885 return amount_points / volume
890 def __get_amount_points(self):
892 @brief Count covered points by the BANG-block and if cache is enable then covered points are stored.
894 @return (uint) Amount of covered points.
898 for index
in range(len(self.
__data)):
906 def __cache_covered_data(self):
908 @brief Cache covered data.
914 for index_point
in range(len(self.
__data)):
919 def __cache_point(self, index):
921 @brief Store index points.
923 @param[in] index (uint): Index point that should be stored.
936 @brief Class implements BANG grid based clustering algorithm.
937 @details BANG clustering algorithms uses a multidimensional grid structure to organize the value space surrounding
938 the pattern values. The patterns are grouped into blocks and clustered with respect to the blocks by
939 a topological neighbor search algorithm @cite inproceedings::bang::1.
941 Code example of BANG usage:
943 from pyclustering.cluster.bang import bang, bang_visualizer
944 from pyclustering.utils import read_sample
945 from pyclustering.samples.definitions import FCPS_SAMPLES
947 # Read data three dimensional data.
948 data = read_sample(FCPS_SAMPLES.SAMPLE_CHAINLINK)
950 # Prepare algorithm's parameters.
953 # Create instance of BANG algorithm.
954 bang_instance = bang(data, levels)
955 bang_instance.process()
957 # Obtain clustering results.
958 clusters = bang_instance.get_clusters()
959 noise = bang_instance.get_noise()
960 directory = bang_instance.get_directory()
961 dendrogram = bang_instance.get_dendrogram()
963 # Visualize BANG clustering results.
964 bang_visualizer.show_blocks(directory)
965 bang_visualizer.show_dendrogram(dendrogram)
966 bang_visualizer.show_clusters(data, clusters, noise)
969 There is visualization of BANG-clustering of three-dimensional data 'chainlink'. BANG-blocks that were formed during
970 processing are shown on following figure. The darkest color means highest density, blocks that does not cover points
972 @image html bang_blocks_chainlink.png "Fig. 1. BANG-blocks that cover input data."
974 Here is obtained dendrogram that can be used for further analysis to improve clustering results:
975 @image html bang_dendrogram_chainlink.png "Fig. 2. BANG dendrogram where the X-axis contains BANG-blocks, the Y-axis contains density."
977 BANG clustering result of 'chainlink' data:
978 @image html bang_clustering_chainlink.png "Fig. 3. BANG clustering result. Data: 'chainlink'."
982 def __init__(self, data, levels, ccore=False, **kwargs):
984 @brief Create BANG clustering algorithm.
986 @param[in] data (list): Input data (list of points) that should be clustered.
987 @param[in] levels (uint): Amount of levels in tree that is used for splitting (how many times block should be
988 split). For example, if amount of levels is two then surface will be divided into two blocks and
989 each obtained block will be divided into blocks also.
990 @param[in] ccore (bool): Reserved positional argument - not used yet.
991 @param[in] **kwargs: Arbitrary keyword arguments (available arguments: 'observe').
993 <b>Keyword Args:</b><br>
994 - density_threshold (double): If block density is smaller than this value then contained data by this
995 block is considered as a noise and its points as outliers. Block density is defined by amount of
996 points in block divided by block volume: <i>amount_block_points</i>/<i>block_volume</i>. By default
997 it is 0.0 - means than only empty blocks are considered as noise. Be aware that this parameter is used
998 with parameter 'amount_threshold' - the maximum threshold is considered during processing.
999 - amount_threshold (uint): Amount of points in the block when it contained data in bang-block is
1000 considered as a noise and there is no need to split it till the last level. Be aware that this parameter
1001 is used with parameter 'density_threshold' - the maximum threshold is considered during processing.
1020 @brief Performs clustering process in line with rules of BANG clustering algorithm.
1022 @return (bang) Returns itself (BANG instance).
1026 @see get_directory()
1027 @see get_dendrogram()
1040 @brief Returns allocated clusters.
1042 @remark Allocated clusters are returned only after data processing (method process()). Otherwise empty list is returned.
1044 @return (list) List of allocated clusters, each cluster contains indexes of objects in list of data.
1055 @brief Returns allocated noise.
1057 @remark Allocated noise is returned only after data processing (method process()). Otherwise empty list is returned.
1059 @return (list) List of indexes that are marked as a noise.
1070 @brief Returns grid directory that describes grid of the processed data.
1072 @remark Grid directory is returned only after data processing (method process()). Otherwise None value is returned.
1074 @return (bang_directory) BANG directory that describes grid of process data.
1084 @brief Returns dendrogram of clusters.
1085 @details Dendrogram is created in following way: the density indices of all regions are calculated and sorted
1086 in decreasing order for each cluster during clustering process.
1088 @remark Dendrogram is returned only after data processing (method process()). Otherwise empty list is returned.
1096 @brief Returns clustering result representation type that indicate how clusters are encoded.
1098 @return (type_encoding) Clustering result representation.
1104 return type_encoding.CLUSTER_INDEX_LIST_SEPARATION
1107 def __validate_arguments(self):
1109 @brief Check input arguments of BANG algorithm and if one of them is not correct then appropriate exception
1113 if len(self.
__data) == 0:
1114 raise ValueError(
"Input data is empty (size: '%d')." % len(self.
__data))
1117 raise ValueError(
"Height of the tree should be greater than 0 (current value: '%d')." % self.
__levels)
1120 raise ValueError(
"Density threshold should be greater or equal to 0 (current value: '%d')." %
1124 raise ValueError(
"Amount of points threshold should be greater than 0 (current value: '%d')" %
1128 def __allocate_clusters(self):
1130 @brief Performs cluster allocation using leafs of tree in BANG directory (the smallest cells).
1134 unhandled_block_indexes = set([i
for i
in range(len(leaf_blocks))
if leaf_blocks[i].get_density() > self.
__density_threshold])
1139 while current_block
is not None:
1151 def __expand_cluster_block(self, block, cluster_index, leaf_blocks, unhandled_block_indexes):
1153 @brief Expand cluster from specific block that is considered as a central block.
1155 @param[in] block (bang_block): Block that is considered as a central block for cluster.
1156 @param[in] cluster_index (uint): Index of cluster that is assigned to blocks that forms new cluster.
1157 @param[in] leaf_blocks (list): Leaf BANG-blocks that are considered during cluster formation.
1158 @param[in] unhandled_block_indexes (set): Set of candidates (BANG block indexes) to become a cluster member. The
1159 parameter helps to reduce traversing among BANG-block providing only restricted set of block that
1160 should be considered.
1164 block.set_cluster(cluster_index)
1170 for neighbor
in neighbors:
1171 neighbor.set_cluster(cluster_index)
1175 neighbors += neighbor_neighbors
1178 def __store_clustering_results(self, amount_clusters, leaf_blocks):
1180 @brief Stores clustering results in a convenient way.
1182 @param[in] amount_clusters (uint): Amount of cluster that was allocated during processing.
1183 @param[in] leaf_blocks (list): Leaf BANG-blocks (the smallest cells).
1186 self.
__clusters = [[]
for _
in range(amount_clusters)]
1187 for block
in leaf_blocks:
1188 index = block.get_cluster()
1190 if index
is not None:
1193 self.
__noise += block.get_points()
1199 def __find_block_center(self, level_blocks, unhandled_block_indexes):
1201 @brief Search block that is cluster center for new cluster.
1203 @return (bang_block) Central block for new cluster, if cluster is not found then None value is returned.
1206 for i
in reversed(range(len(level_blocks))):
1210 if level_blocks[i].get_cluster()
is None:
1211 unhandled_block_indexes.remove(i)
1212 return level_blocks[i]
1217 def __find_block_neighbors(self, block, level_blocks, unhandled_block_indexes):
1219 @brief Search block neighbors that are parts of new clusters (density is greater than threshold and that are
1220 not cluster members yet), other neighbors are ignored.
1222 @param[in] block (bang_block): BANG-block for which neighbors should be found (which can be part of cluster).
1223 @param[in] level_blocks (list): BANG-blocks on specific level.
1224 @param[in] unhandled_block_indexes (set): Blocks that have not been processed yet.
1226 @return (list) Block neighbors that can become part of cluster.
1231 handled_block_indexes = []
1232 for unhandled_index
in unhandled_block_indexes:
1233 if block.is_neighbor(level_blocks[unhandled_index]):
1234 handled_block_indexes.append(unhandled_index)
1235 neighbors.append(level_blocks[unhandled_index])
1238 if len(neighbors) == 8:
1241 for handled_index
in handled_block_indexes:
1242 unhandled_block_indexes.remove(handled_index)
1247 def __update_cluster_dendrogram(self, index_cluster, blocks):
1249 @brief Append clustered blocks to dendrogram.
1251 @param[in] index_cluster (uint): Cluster index that was assigned to blocks.
1252 @param[in] blocks (list): Blocks that were clustered.
1258 blocks = sorted(blocks, key=
lambda block: block.get_density(), reverse=
True)