Skip to main content

regions - Computing the intersection area of two disks in 3D


Closely related to this question about highlighting intersection of two disks, I am trying to figure out if one can do so similarly for disks embedded in 3D (e.g. in a bounding box). The difference is that, in 3D the orientation of the disks matters in how much of overlap/orthogonal-projection there is between them. The orientation of a disk is simply the vector normal to its surface and centered at its center. Therefore, each disk has a center vector (for its position) v and a normal vector n for its orientation. As an example, 2 disks i,j have maximal overlap if ni∥nj and the difference vector of their center positions vj−vi also being parallel to their normal, then the overlap area is exactly πr2, r being the radius of the disks.


Intuitively, computing such projection is as if we computed the shadow two drawn particles (here disks) create onto one another when visualizing them.




  • But is there a way we can quantify the overlap area between two 3D-embedded disks in Mathematica? Can RegionIntersection be made use of for such application?




Additional clarifications after comments:


To clarify how the overlap between the disks is defined or at least what I mean by it, the idea is to compute the orthogonal projection of their respective surfaces onto one another. For instance given 2 disks i,j with their position and normal vectors vi,ni and vj,nj, we can take the average of the orthogonal-surface-projection of disk i onto plane of disk j with that of disk j onto plane of disk i which yields a symmetrized definition of overlap or intersection between the disks, taking into account not only their orientations but also relative positions.




Stealing from J. M.'s answer here (its first part), here's an image of one such disk within its plane and its orientation vector visualised (the normal to the plane centered at center of disk):


circle on a tilted plane




An attempt to visualize DaveH's suggestion which was very briefly put in their answer:



Say we have one disk centered at v1 and with normal vector n1 and another with v2,n2 as given by (both with diameter d):


v1 = {0.5, 0.5, 0.5}
n1 = {1, 1, 1}
v2 = {1, 1.5, 0}
n2 = {1, 1, 0}
d = 4

then we create cylinders out of the disks, with end-points of each ceylinder given by ±5ni to respective center position of disk i:


cyl1 = Cylinder[{v1 - 5*n1, v1 + 5*n1}, d/2]
cyl2 = Cylinder[{v2 - 5*n2, v2 + 5*n2}, d/2]


and visualizing Graphics3D[{Opacity[.5], cyl1, cyl2}]:


enter image description here


But I don't know how much this approach helps in computing the overlap area of interest (and if computationally feasible).



Answer



Here's my take on solving it algebraically:


DiskRadius[Disk3D[_, _, radius_]] := radius;
RotateZToNormal[Disk3D[_, n_, _]] := RotationTransform[{{0, 0, 1}, n}];
MoveToDiskCenter[Disk3D[p_, _, _]] := TranslationTransform[p];
TransformUnitDiskTo[d_Disk3D] := RightComposition[RotateZToNormal[d], MoveToDiskCenter[d]]

Project2D = Most;(*leave out z component to project into 2D*)
CartesianFromPolar = (# /. {r -> Sqrt[x^2 + y^2], \[Phi] -> ArcTan[x, y]} &);
UnitDiskToProjectedEllipseTransform[to_Disk3D] := Function[from,
Composition[
AffineTransform[Reverse[##]] &, (*construct 2d affine transform unitdisk -> projected disk/ellipse*)
CoefficientArrays[#, {x, y}] &, (*extract 2d ellipse linear transformation coefficients*)
Simplify, CartesianFromPolar, Project2D,
InverseFunction[TransformUnitDiskTo[to]],
TransformUnitDiskTo[from]
][r {Cos[\[Phi]], Sin[\[Phi]], 0}]

]
ProjectDiskOnto[to_Disk3D] := Function[from,
Composition[
#.# <= DiskRadius[from]^2 &,
InverseFunction[UnitDiskToProjectedEllipseTransform[to][from]]
][{x, y}]
]
ProjectedDiskRegion[to_Disk3D] := Function[from,
RegionIntersection[
(ImplicitRegion[#, {x, y}] & @* ProjectDiskOnto[to]) /@ {to, from}

]
]
DiskIntersectionArea[disk1_Disk3D, disk2_Disk3D] :=
Mean[Area /@ {ProjectedDiskRegion[disk1][disk2],
ProjectedDiskRegion[disk2][disk1]}]

Let's have a look at an example:


d1 = Disk3D[{-2, 3, -3}, {2, -3, 6}/7, 1]
d2 = Disk3D[{-1, 1, 1}, {-1, 2, -2}/3, 4/5]


Here we encoded the disks by their center point, their normal and their radius with a custom Disk3D head. We can plot these to get an idea


PlotDisk3D[d_Disk3D] := ParametricPlot3D[
TransformUnitDiskTo[d][r {Cos[\[Phi]], Sin[\[Phi]], 0}],
{r, 0, DiskRadius[d]}, {\[Phi], 0, 2 \[Pi]}, Mesh -> None
]
Show[PlotDisk3D /@ {d1, d2}]

Plot of both disks together in 3D


The idea of the solution is to first get an implicit 2d equation of each disk transformed into the reference frame of the other disk and then project it into the xy plane. We do that by creating the function TransformUnitDiskTo which produces an AffineTransform that would transform a unit disk sitting in the xy-plane into any given to disk. Next we start with a parametric polar representation of a unit disk, which we first transform into our (from) disk which we want to project, and then follow it by an inverse affine transform to get it into the reference frame of our to disk. In this reference frame we can project it into 2D and after that convert back to cartesian coordinates and into an implicit representation instead of parametric. Our two example disks in the other reference frame now look like this:


ProjectDiskOnto[d1][d2]


(43x52−8y13+14)2+(37x65+11y13+15)2≤1625


ProjectDiskOnto[d2][d1]

(11x13−8y13+1813)2+(37x65+43y52−4765)2≤1


Projecting a disk onto itself naturally always gives back the disk unaltered:


ProjectDiskOnto[d1][d1]

x2+y2≤1


ProjectDiskOnto[d2][d2]


x2+y2≤1625


Now we can perform the region intersection inside ImplicitRegions


Resulting projected and intersected disk regions.


and finally take the average of the Region Areas, which Mathematica happily performs for us symbolically and we end up with an exact expression, which we can either simplify a bit via RootReduce on the algebraic parts or just get a numeric approximation with desired accuracy:


DiskIntersectionArea[d1, d2] // N
(* 0.9875 *)




  • Added support for arbitrary disk radii.

  • Fixed a bug in the construction of the implicit ellipse representation.


Comments

Popular posts from this blog

functions - Get leading series expansion term?

Given a function f[x] , I would like to have a function leadingSeries that returns just the leading term in the series around x=0 . For example: leadingSeries[(1/x + 2)/(4 + 1/x^2 + x)] x and leadingSeries[(1/x + 2 + (1 - 1/x^3)/4)/(4 + x)] -(1/(16 x^3)) Is there such a function in Mathematica? Or maybe one can implement it efficiently? EDIT I finally went with the following implementation, based on Carl Woll 's answer: lds[ex_,x_]:=( (ex/.x->(x+O[x]^2))/.SeriesData[U_,Z_,L_List,Mi_,Ma_,De_]:>SeriesData[U,Z,{L[[1]]},Mi,Mi+1,De]//Quiet//Normal) The advantage is, that this one also properly works with functions whose leading term is a constant: lds[Exp[x],x] 1 Answer Update 1 Updated to eliminate SeriesData and to not return additional terms Perhaps you could use: leadingSeries[expr_, x_] := Normal[expr /. x->(x+O[x]^2) /. a_List :> Take[a, 1]] Then for your examples: leadingSeries[(1/x + 2)/(4 + 1/x^2 + x), x] leadingSeries[Exp[x], x] leadingSeries[(1/x + 2 + (1 - 1/x...

mathematical optimization - Minimizing using indices, error: Part::pkspec1: The expression cannot be used as a part specification

I want to use Minimize where the variables to minimize are indices pointing into an array. Here a MWE that hopefully shows what my problem is. vars = u@# & /@ Range[3]; cons = Flatten@ { Table[(u[j] != #) & /@ vars[[j + 1 ;; -1]], {j, 1, 3 - 1}], 1 vec1 = {1, 2, 3}; vec2 = {1, 2, 3}; Minimize[{Total@((vec1[[#]] - vec2[[u[#]]])^2 & /@ Range[1, 3]), cons}, vars, Integers] The error I get: Part::pkspec1: The expression u[1] cannot be used as a part specification. >> Answer Ok, it seems that one can get around Mathematica trying to evaluate vec2[[u[1]]] too early by using the function Indexed[vec2,u[1]] . The working MWE would then look like the following: vars = u@# & /@ Range[3]; cons = Flatten@{ Table[(u[j] != #) & /@ vars[[j + 1 ;; -1]], {j, 1, 3 - 1}], 1 vec1 = {1, 2, 3}; vec2 = {1, 2, 3}; NMinimize[ {Total@((vec1[[#]] - Indexed[vec2, u[#]])^2 & /@ R...

How to remap graph properties?

Graph objects support both custom properties, which do not have special meanings, and standard properties, which may be used by some functions. When importing from formats such as GraphML, we usually get a result with custom properties. What is the simplest way to remap one property to another, e.g. to remap a custom property to a standard one so it can be used with various functions? Example: Let's get Zachary's karate club network with edge weights and vertex names from here: http://nexus.igraph.org/api/dataset_info?id=1&format=html g = Import[ "http://nexus.igraph.org/api/dataset?id=1&format=GraphML", {"ZIP", "karate.GraphML"}] I can remap "name" to VertexLabels and "weights" to EdgeWeight like this: sp[prop_][g_] := SetProperty[g, prop] g2 = g // sp[EdgeWeight -> (PropertyValue[{g, #}, "weight"] & /@ EdgeList[g])] // sp[VertexLabels -> (# -> PropertyValue[{g, #}, "name"]...