The algorithm can be easily extended to 3D and higher dimensions, and I decided to illustrate that with an example. The 3D implementation is really easy using Mathematica's geometric computation functions (i) for derivation of rotation matrices and (ii) for region manipulation. (The latter are new in version 10.)
Let us first generate some data.
npoints = 10000;
data = {RandomReal[{0, 2 \[Pi]}, npoints],
RandomReal[{-\[Pi], \[Pi]}, npoints],
RandomVariate[PoissonDistribution[4.8], npoints]};
data = MapThread[#3*{Cos[#1] Cos[#2], Sin[#1] Cos[#2], Sin[#2]} &, data];
Let us plot the data:
Block[{qs = 12},
qs = Map[Quantile[#, Range[0, 1, 1/(qs - 1)]] &, Transpose[data]];
ListPointPlot3D[data, PlotRange -> All, PlotTheme -> "Detailed",
FaceGrids -> {{{0, 0, -1}, Most[qs]}, {{0, 1, 0},
qs[[{1, 3}]]}, {{-1, 0, 0}, Rest[qs]}}, ImageSize -> 700]
]
data:image/s3,"s3://crabby-images/07d75/07d7598ff754270cd7f6cf12615b4fe3916f151a" alt="enter image description here"
(On the plot the grid lines derived from the quantiles of x, y and z coordinates of the data set.)
Using the set of uniform angles:
nqs = 20;
dirs = N@Flatten[
Table[{Cos[\[Theta]] Cos[\[Phi]], Sin[\[Theta]] Cos[\[Phi]],
Sin[\[Phi]]}, {\[Theta], 2 \[Pi]/(10 nqs), 2 \[Pi],
2 \[Pi]/nqs}, {\[Phi], -\[Pi], \[Pi], 2 \[Pi]/nqs}], 1];
data:image/s3,"s3://crabby-images/c57e0/c57e0efed4d1ee06ab7d57fa4e6f214b336464ab" alt="enter image description here"
we can compute the quantiles of the z-coordinates in each direction:
rmats = RotationMatrix[{{1, 0, 0}, #}] & /@ dirs;
qs = {0.95};
qDirPoints =
Flatten[Map[Function[{m}, Quantile[(m.Transpose[data])[[3]], qs]], rmats]];
and produce the quantile envelope surface using ImplicitRegion
and BoundaryDiscretizeRegion
:
data:image/s3,"s3://crabby-images/46388/46388e7758390ca9bb5a34f75e612289066b7981" alt="enter image description here"
More details and examples of usage are given in my blog post: "Directional quantile envelopes in 3D" . Also, see the attached notebook.
Attachments: