1
+ from __future__ import annotations
2
+
3
+ import collections .abc
4
+ import numbers
1
5
from collections import Counter
2
- from collections .abc import Iterable , Sized
3
6
from itertools import chain , combinations
4
7
from math import factorial , sqrt
8
+ from typing import Any , Iterable , Iterator , List , Sequence , Tuple , Union
5
9
6
10
import scipy .spatial
7
11
from numpy import abs as np_abs
13
17
dot ,
14
18
eye ,
15
19
mean ,
20
+ ndarray ,
16
21
ones ,
17
22
square ,
18
23
subtract ,
22
27
from numpy .linalg import det as ndet
23
28
from numpy .linalg import matrix_rank , norm , slogdet , solve
24
29
30
+ from adaptive .types import Bool
31
+
32
+ try :
33
+ from typing import TypeAlias
34
+ except ImportError :
35
+ # Remove this when we drop support for Python 3.9
36
+ from typing_extensions import TypeAlias
37
+
25
38
26
- def fast_norm (v ):
39
+ SimplexPoints : TypeAlias = Union [List [Tuple [float , ...]], ndarray ]
40
+ Simplex : TypeAlias = Union [Sequence [numbers .Integral ], ndarray ]
41
+ Point : TypeAlias = Union [Tuple [float , ...], ndarray ]
42
+ Points : TypeAlias = Union [Sequence [Tuple [float , ...]], ndarray ]
43
+
44
+
45
+ def fast_norm (v : tuple [float , ...] | ndarray ) -> float :
27
46
"""Take the vector norm for len 2, 3 vectors.
28
47
Defaults to a square root of the dot product for larger vectors.
29
48
@@ -41,7 +60,9 @@ def fast_norm(v):
41
60
return sqrt (dot (v , v ))
42
61
43
62
44
- def fast_2d_point_in_simplex (point , simplex , eps = 1e-8 ):
63
+ def fast_2d_point_in_simplex (
64
+ point : Point , simplex : SimplexPoints , eps : float = 1e-8
65
+ ) -> Bool :
45
66
(p0x , p0y ), (p1x , p1y ), (p2x , p2y ) = simplex
46
67
px , py = point
47
68
@@ -55,7 +76,7 @@ def fast_2d_point_in_simplex(point, simplex, eps=1e-8):
55
76
return (t >= - eps ) and (s + t <= 1 + eps )
56
77
57
78
58
- def point_in_simplex (point , simplex , eps = 1e-8 ):
79
+ def point_in_simplex (point : Point , simplex : SimplexPoints , eps : float = 1e-8 ) -> Bool :
59
80
if len (point ) == 2 :
60
81
return fast_2d_point_in_simplex (point , simplex , eps )
61
82
@@ -66,7 +87,7 @@ def point_in_simplex(point, simplex, eps=1e-8):
66
87
return all (alpha > - eps ) and sum (alpha ) < 1 + eps
67
88
68
89
69
- def fast_2d_circumcircle (points ) :
90
+ def fast_2d_circumcircle (points : Points ) -> tuple [ tuple [ float , float ], float ] :
70
91
"""Compute the center and radius of the circumscribed circle of a triangle
71
92
72
93
Parameters
@@ -79,7 +100,7 @@ def fast_2d_circumcircle(points):
79
100
tuple
80
101
(center point : tuple(float), radius: float)
81
102
"""
82
- points = array (points )
103
+ points = array (points , dtype = float )
83
104
# transform to relative coordinates
84
105
pts = points [1 :] - points [0 ]
85
106
@@ -102,7 +123,9 @@ def fast_2d_circumcircle(points):
102
123
return (x + points [0 ][0 ], y + points [0 ][1 ]), radius
103
124
104
125
105
- def fast_3d_circumcircle (points ):
126
+ def fast_3d_circumcircle (
127
+ points : Points ,
128
+ ) -> tuple [tuple [float , float , float ], float ]:
106
129
"""Compute the center and radius of the circumscribed sphere of a simplex.
107
130
108
131
Parameters
@@ -142,7 +165,7 @@ def fast_3d_circumcircle(points):
142
165
return center , radius
143
166
144
167
145
- def fast_det (matrix ) :
168
+ def fast_det (matrix : ndarray ) -> float :
146
169
matrix = asarray (matrix , dtype = float )
147
170
if matrix .shape == (2 , 2 ):
148
171
return matrix [0 ][0 ] * matrix [1 ][1 ] - matrix [1 ][0 ] * matrix [0 ][1 ]
@@ -153,7 +176,7 @@ def fast_det(matrix):
153
176
return ndet (matrix )
154
177
155
178
156
- def circumsphere (pts ) :
179
+ def circumsphere (pts : Simplex ) -> tuple [ tuple [ float , ...], float ] :
157
180
"""Compute the center and radius of a N dimension sphere which touches each point in pts.
158
181
159
182
Parameters
@@ -201,7 +224,7 @@ def circumsphere(pts):
201
224
return tuple (center ), radius
202
225
203
226
204
- def orientation (face , origin ) :
227
+ def orientation (face : tuple | ndarray , origin : tuple | ndarray ) -> int :
205
228
"""Compute the orientation of the face with respect to a point, origin.
206
229
207
230
Parameters
@@ -224,14 +247,14 @@ def orientation(face, origin):
224
247
sign , logdet = slogdet (vectors - origin )
225
248
if logdet < - 50 : # assume it to be zero when it's close to zero
226
249
return 0
227
- return sign
250
+ return int ( sign )
228
251
229
252
230
- def is_iterable_and_sized (obj ) :
231
- return isinstance (obj , Iterable ) and isinstance ( obj , Sized )
253
+ def is_iterable_and_sized (obj : Any ) -> bool :
254
+ return isinstance (obj , collections . abc . Collection )
232
255
233
256
234
- def simplex_volume_in_embedding (vertices ) -> float :
257
+ def simplex_volume_in_embedding (vertices : Sequence [ Point ] ) -> float :
235
258
"""Calculate the volume of a simplex in a higher dimensional embedding.
236
259
That is: dim > len(vertices) - 1. For example if you would like to know the
237
260
surface area of a triangle in a 3d space.
@@ -312,7 +335,7 @@ class Triangulation:
312
335
or more simplices in the
313
336
"""
314
337
315
- def __init__ (self , coords ) :
338
+ def __init__ (self , coords : Points ) -> None :
316
339
if not is_iterable_and_sized (coords ):
317
340
raise TypeError ("Please provide a 2-dimensional list of points" )
318
341
coords = list (coords )
@@ -340,38 +363,40 @@ def __init__(self, coords):
340
363
"(the points are linearly dependent)"
341
364
)
342
365
343
- self .vertices = list (coords )
344
- self .simplices = set ()
366
+ self .vertices : list [ Point ] = list (coords )
367
+ self .simplices : set [ Simplex ] = set ()
345
368
# initialise empty set for each vertex
346
- self .vertex_to_simplices = [set () for _ in coords ]
369
+ self .vertex_to_simplices : list [ set [ Simplex ]] = [set () for _ in coords ]
347
370
348
371
# find a Delaunay triangulation to start with, then we will throw it
349
372
# away and continue with our own algorithm
350
373
initial_tri = scipy .spatial .Delaunay (coords )
351
374
for simplex in initial_tri .simplices :
352
375
self .add_simplex (simplex )
353
376
354
- def delete_simplex (self , simplex ) :
377
+ def delete_simplex (self , simplex : Simplex ) -> None :
355
378
simplex = tuple (sorted (simplex ))
356
379
self .simplices .remove (simplex )
357
380
for vertex in simplex :
358
381
self .vertex_to_simplices [vertex ].remove (simplex )
359
382
360
- def add_simplex (self , simplex ) :
383
+ def add_simplex (self , simplex : Simplex ) -> None :
361
384
simplex = tuple (sorted (simplex ))
362
385
self .simplices .add (simplex )
363
386
for vertex in simplex :
364
387
self .vertex_to_simplices [vertex ].add (simplex )
365
388
366
- def get_vertices (self , indices ) :
389
+ def get_vertices (self , indices : Iterable [ numbers . Integral ]) -> list [ Point | None ] :
367
390
return [self .get_vertex (i ) for i in indices ]
368
391
369
- def get_vertex (self , index ) :
392
+ def get_vertex (self , index : numbers . Integral | None ) -> Point | None :
370
393
if index is None :
371
394
return None
372
395
return self .vertices [index ]
373
396
374
- def get_reduced_simplex (self , point , simplex , eps = 1e-8 ) -> list :
397
+ def get_reduced_simplex (
398
+ self , point : Point , simplex : Simplex , eps : float = 1e-8
399
+ ) -> list [numbers .Integral ]:
375
400
"""Check whether vertex lies within a simplex.
376
401
377
402
Returns
@@ -396,11 +421,13 @@ def get_reduced_simplex(self, point, simplex, eps=1e-8) -> list:
396
421
397
422
return [simplex [i ] for i in result ]
398
423
399
- def point_in_simplex (self , point , simplex , eps = 1e-8 ):
424
+ def point_in_simplex (
425
+ self , point : Point , simplex : Simplex , eps : float = 1e-8
426
+ ) -> Bool :
400
427
vertices = self .get_vertices (simplex )
401
428
return point_in_simplex (point , vertices , eps )
402
429
403
- def locate_point (self , point ) :
430
+ def locate_point (self , point : Point ) -> Simplex :
404
431
"""Find to which simplex the point belongs.
405
432
406
433
Return indices of the simplex containing the point.
@@ -412,10 +439,15 @@ def locate_point(self, point):
412
439
return ()
413
440
414
441
@property
415
- def dim (self ):
442
+ def dim (self ) -> int :
416
443
return len (self .vertices [0 ])
417
444
418
- def faces (self , dim = None , simplices = None , vertices = None ):
445
+ def faces (
446
+ self ,
447
+ dim : int | None = None ,
448
+ simplices : Iterable [Simplex ] | None = None ,
449
+ vertices : Iterable [int ] | None = None ,
450
+ ) -> Iterator [tuple [numbers .Integral , ...]]:
419
451
"""Iterator over faces of a simplex or vertex sequence."""
420
452
if dim is None :
421
453
dim = self .dim
@@ -436,11 +468,11 @@ def faces(self, dim=None, simplices=None, vertices=None):
436
468
else :
437
469
return faces
438
470
439
- def containing (self , face ) :
471
+ def containing (self , face : tuple [ int , ...]) -> set [ Simplex ] :
440
472
"""Simplices containing a face."""
441
473
return set .intersection (* (self .vertex_to_simplices [i ] for i in face ))
442
474
443
- def _extend_hull (self , new_vertex , eps = 1e-8 ):
475
+ def _extend_hull (self , new_vertex : Point , eps : float = 1e-8 ) -> set [ Simplex ] :
444
476
# count multiplicities in order to get all hull faces
445
477
multiplicities = Counter (face for face in self .faces ())
446
478
hull_faces = [face for face , count in multiplicities .items () if count == 1 ]
@@ -480,7 +512,9 @@ def _extend_hull(self, new_vertex, eps=1e-8):
480
512
481
513
return new_simplices
482
514
483
- def circumscribed_circle (self , simplex , transform ):
515
+ def circumscribed_circle (
516
+ self , simplex : Simplex , transform : ndarray
517
+ ) -> tuple [tuple [float , ...], float ]:
484
518
"""Compute the center and radius of the circumscribed circle of a simplex.
485
519
486
520
Parameters
@@ -496,7 +530,9 @@ def circumscribed_circle(self, simplex, transform):
496
530
pts = dot (self .get_vertices (simplex ), transform )
497
531
return circumsphere (pts )
498
532
499
- def point_in_cicumcircle (self , pt_index , simplex , transform ):
533
+ def point_in_cicumcircle (
534
+ self , pt_index : int , simplex : Simplex , transform : ndarray
535
+ ) -> Bool :
500
536
# return self.fast_point_in_circumcircle(pt_index, simplex, transform)
501
537
eps = 1e-8
502
538
@@ -506,10 +542,15 @@ def point_in_cicumcircle(self, pt_index, simplex, transform):
506
542
return norm (center - pt ) < (radius * (1 + eps ))
507
543
508
544
@property
509
- def default_transform (self ):
545
+ def default_transform (self ) -> ndarray :
510
546
return eye (self .dim )
511
547
512
- def bowyer_watson (self , pt_index , containing_simplex = None , transform = None ):
548
+ def bowyer_watson (
549
+ self ,
550
+ pt_index : int ,
551
+ containing_simplex : Simplex | None = None ,
552
+ transform : ndarray | None = None ,
553
+ ) -> tuple [set [Simplex ], set [Simplex ]]:
513
554
"""Modified Bowyer-Watson point adding algorithm.
514
555
515
556
Create a hole in the triangulation around the new point,
@@ -569,10 +610,10 @@ def bowyer_watson(self, pt_index, containing_simplex=None, transform=None):
569
610
new_triangles = self .vertex_to_simplices [pt_index ]
570
611
return bad_triangles - new_triangles , new_triangles - bad_triangles
571
612
572
- def _simplex_is_almost_flat (self , simplex ) :
613
+ def _simplex_is_almost_flat (self , simplex : Simplex ) -> Bool :
573
614
return self ._relative_volume (simplex ) < 1e-8
574
615
575
- def _relative_volume (self , simplex ) :
616
+ def _relative_volume (self , simplex : Simplex ) -> float :
576
617
"""Compute the volume of a simplex divided by the average (Manhattan)
577
618
distance of its vertices. The advantage of this is that the relative
578
619
volume is only dependent on the shape of the simplex and not on the
@@ -583,20 +624,25 @@ def _relative_volume(self, simplex):
583
624
average_edge_length = mean (np_abs (vectors ))
584
625
return self .volume (simplex ) / (average_edge_length ** self .dim )
585
626
586
- def add_point (self , point , simplex = None , transform = None ):
627
+ def add_point (
628
+ self ,
629
+ point : Point ,
630
+ simplex : Simplex | None = None ,
631
+ transform : ndarray | None = None ,
632
+ ) -> tuple [set [Simplex ], set [Simplex ]]:
587
633
"""Add a new vertex and create simplices as appropriate.
588
634
589
635
Parameters
590
636
----------
591
637
point : float vector
592
638
Coordinates of the point to be added.
593
- transform : N*N matrix of floats
594
- Multiplication matrix to apply to the point (and neighbouring
595
- simplices) when running the Bowyer Watson method.
596
639
simplex : tuple of ints, optional
597
640
Simplex containing the point. Empty tuple indicates points outside
598
641
the hull. If not provided, the algorithm costs O(N), so this should
599
642
be used whenever possible.
643
+ transform : N*N matrix of floats
644
+ Multiplication matrix to apply to the point (and neighbouring
645
+ simplices) when running the Bowyer Watson method.
600
646
"""
601
647
point = tuple (point )
602
648
if simplex is None :
@@ -632,16 +678,16 @@ def add_point(self, point, simplex=None, transform=None):
632
678
self .vertices .append (point )
633
679
return self .bowyer_watson (pt_index , actual_simplex , transform )
634
680
635
- def volume (self , simplex ) :
681
+ def volume (self , simplex : Simplex ) -> float :
636
682
prefactor = factorial (self .dim )
637
683
vertices = array (self .get_vertices (simplex ))
638
684
vectors = vertices [1 :] - vertices [0 ]
639
685
return float (abs (fast_det (vectors )) / prefactor )
640
686
641
- def volumes (self ):
687
+ def volumes (self ) -> list [ float ] :
642
688
return [self .volume (sim ) for sim in self .simplices ]
643
689
644
- def reference_invariant (self ):
690
+ def reference_invariant (self ) -> bool :
645
691
"""vertex_to_simplices and simplices are compatible."""
646
692
for vertex in range (len (self .vertices )):
647
693
if any (vertex not in tri for tri in self .vertex_to_simplices [vertex ]):
@@ -655,26 +701,28 @@ def vertex_invariant(self, vertex):
655
701
"""Simplices originating from a vertex don't overlap."""
656
702
raise NotImplementedError
657
703
658
- def get_neighbors_from_vertices (self , simplex ) :
704
+ def get_neighbors_from_vertices (self , simplex : Simplex ) -> set [ Simplex ] :
659
705
return set .union (* [self .vertex_to_simplices [p ] for p in simplex ])
660
706
661
- def get_face_sharing_neighbors (self , neighbors , simplex ):
707
+ def get_face_sharing_neighbors (
708
+ self , neighbors : set [Simplex ], simplex : Simplex
709
+ ) -> set [Simplex ]:
662
710
"""Keep only the simplices sharing a whole face with simplex."""
663
711
return {
664
712
simpl for simpl in neighbors if len (set (simpl ) & set (simplex )) == self .dim
665
713
} # they share a face
666
714
667
- def get_simplices_attached_to_points (self , indices ) :
715
+ def get_simplices_attached_to_points (self , indices : Simplex ) -> set [ Simplex ] :
668
716
# Get all simplices that share at least a point with the simplex
669
717
neighbors = self .get_neighbors_from_vertices (indices )
670
718
return self .get_face_sharing_neighbors (neighbors , indices )
671
719
672
- def get_opposing_vertices (self , simplex ) :
720
+ def get_opposing_vertices (self , simplex : Simplex ) -> tuple [ int , ...] :
673
721
if simplex not in self .simplices :
674
722
raise ValueError ("Provided simplex is not part of the triangulation" )
675
723
neighbors = self .get_simplices_attached_to_points (simplex )
676
724
677
- def find_opposing_vertex (vertex ):
725
+ def find_opposing_vertex (vertex : int ):
678
726
# find the simplex:
679
727
simp = next ((x for x in neighbors if vertex not in x ), None )
680
728
if simp is None :
@@ -687,7 +735,7 @@ def find_opposing_vertex(vertex):
687
735
return result
688
736
689
737
@property
690
- def hull (self ):
738
+ def hull (self ) -> set [ numbers . Integral ] :
691
739
"""Compute hull from triangulation.
692
740
693
741
Parameters
0 commit comments