Skip to main content

How to create blurred Graphics3D objects?



Initially I was interested in renderring a 3D analog of a blurred disk like this


DensityPlot[1 - HeavisideLambda[(x^2 + y^2)/8]/2, {x, -4, 4}, {y, -4, 4}, 
FrameTicks -> False, ColorFunctionScaling -> False,
ColorFunction -> "SunsetColors", PlotRange -> {0, 1},
PlotPoints -> 50]

enter image description here


The only idea that came to my mind was playing with Opacity, which gives hardly an impressive result:


Graphics3D[{Orange}~Join~
Table[{Opacity[i], Sphere[{0, 0, 0}, 1 - i]}, {i, 0.1, 1, 0.1}] //

Flatten]

enter image description here


So, 1) is it possible to get "true" blurred ball and 2) how to extrapolate this idea to other 3D objects?



Answer



UPDATE: latest Mathematica 9 functionality


This is very easy now with latest Mathematica 9 functionality. Just use Image3D or Raster3D functions:


data = Developer`ToPackedArray[With[{step = .03}, 
ParallelTable[Exp[-(i^2 + j^2 + k^2)^4/.99],
{k, -1.2, 1.2, step}, {i, -1.2, 1.2, step}, {j, -1.2, 1.2, step}]]];


Image3D[data, ColorFunction -> #, Axes -> True,
ImageSize -> 400] & /@ {Automatic, "XRay"}

enter image description here


-------- OLDER VERSIONS ----------------


METHOD 1 - volumetric rendering - from scratch


I will use ideas from this post by Yu-Sung. First create data for 3D texture:


data = Developer`ToPackedArray[
With[{step = .05},

ParallelTable[{1, 0, 0, Exp[-(i^2 + j^2 + k^2)^4/.8]},
{k, -1, 1, step}, {i, -1, 1, step}, {j, -1, 1, step}]]];

Next create many polygons with applied texture:


Graphics3D[{
EdgeForm[],
Opacity[.4],(*Overall transparency of the textured polygons*)
Texture[data],(*Set volumetric texture*)
With[{pts =
Table[{{0, 0, z}, {1, 0, z}, {1, 1, z}, {0, 1, z}}, {z, 0,

1, .05}]}, Polygon[pts, VertexTextureCoordinates -> pts]]},
PlotRange -> {{0, 1}, {0, 1}, {0, 1}},
Lighting -> "Neutral",
Background -> Black, RotationAction -> "Clip",
SphericalRegion -> True,
BoxStyle -> Directive[Opacity[.2], White],
ImageSize -> 4 {100, 100},
BoxRatios -> {1, 1, 1},
Axes -> False,
BaseStyle -> {RenderingOptions -> {"DepthPeelingLayers" -> 100}}]


Below are the views in different directions. It's pretty fast:


enter image description here


METHOD 2 - volumetric rendering - CUDA - on GPU from built-in interface


Again, I will use ideas from this post by Yu-Sung. We will use built-in interface CUDAVolumetricRender to render our 3D fading out texture on GPU.


Make sure you have latest CUDA paclet - read this tutorial.


Create data which are good for 3D texture understood by CUDAVolumetricRender


data = Developer`ToPackedArray[
With[{step = .05},
Table[Round[255 Exp[-(i^2 + j^2 + k^2)^1/.8]], {k, -1, 1,

step}, {i, -1, 1, step}, {j, -1, 1, step}]]];

Load CUDA package and follow Yu-Sung CUDA cooking recipes as shortly given below (see details in this post)


<< CUDALink`
Clear[prepareCUDAVolumeData];
prepareCUDAVolumeData::arg =
"The argument should be an integer array of depth 3.";
prepareCUDAVolumeData[array_] /; ArrayQ[array, 3, IntegerQ] :=
Module[{x, y, z}, {x, y, z} = Dimensions[array];
Developer`ToPackedArray[

Partition[#, x] & /@ Partition[Flatten[array], x*z]]];
prepareCUDAVolumeData[___] /; (Message[prepareCUDAVolumeData::arg];
False) := Null;
CUDAVolumetricRender[prepareCUDAVolumeData[data]]

After playing with options of the interface (see control positions) and the usual zooming of Mathematica 3D graphics you can get this beautiful view:


enter image description here


METHOD 3 - opaque spheres ---------------------------------------------


This is not ideal, but just a demonstration of a concept. We can approximate a complex volume opacity by filling the volume with small transparent spheres. The more spheres we have and the smaller and closer they are the better. In the example below I even make spheres to overlap to mask the gaps between them. Computation and rendering of 64000 transparent spheres is not short and it is tedious to rotate that 3D object. Due to cubic grid only certain viewing direction result in the image below (I set that with ViewPoint option). I suspect that cubic close packed (face-centered cubic fcc) and hexagonal close-packed (hcp) will show more uniform viewing from various perspectives. I may add the code for those later.


Block[{r = 20},

Graphics3D[
Table[{Red, Opacity[7 N@Exp[-(i^2 + j^2 + k^2)/(r/2)^2]],
Sphere[{i, j, k}, 3]}, {i, -r, r}, {j, -r, r}, {k, -r,
r}], Boxed -> False, Axes -> False, Lighting -> "Neutral",
ViewPoint -> {0, 0, 1}]]

enter image description here


Comments

Popular posts from this blog

front end - keyboard shortcut to invoke Insert new matrix

I frequently need to type in some matrices, and the menu command Insert > Table/Matrix > New... allows matrices with lines drawn between columns and rows, which is very helpful. I would like to make a keyboard shortcut for it, but cannot find the relevant frontend token command (4209405) for it. Since the FullForm[] and InputForm[] of matrices with lines drawn between rows and columns is the same as those without lines, it's hard to do this via 3rd party system-wide text expanders (e.g. autohotkey or atext on mac). How does one assign a keyboard shortcut for the menu item Insert > Table/Matrix > New... , preferably using only mathematica? Thanks! Answer In the MenuSetup.tr (for linux located in the $InstallationDirectory/SystemFiles/FrontEnd/TextResources/X/ directory), I changed the line MenuItem["&New...", "CreateGridBoxDialog"] to read MenuItem["&New...", "CreateGridBoxDialog", MenuKey["m", Modifiers-...

How to thread a list

I have data in format data = {{a1, a2}, {b1, b2}, {c1, c2}, {d1, d2}} Tableform: I want to thread it to : tdata = {{{a1, b1}, {a2, b2}}, {{a1, c1}, {a2, c2}}, {{a1, d1}, {a2, d2}}} Tableform: And I would like to do better then pseudofunction[n_] := Transpose[{data2[[1]], data2[[n]]}]; SetAttributes[pseudofunction, Listable]; Range[2, 4] // pseudofunction Here is my benchmark data, where data3 is normal sample of real data. data3 = Drop[ExcelWorkBook[[Column1 ;; Column4]], None, 1]; data2 = {a #, b #, c #, d #} & /@ Range[1, 10^5]; data = RandomReal[{0, 1}, {10^6, 4}]; Here is my benchmark code kptnw[list_] := Transpose[{Table[First@#, {Length@# - 1}], Rest@#}, {3, 1, 2}] &@list kptnw2[list_] := Transpose[{ConstantArray[First@#, Length@# - 1], Rest@#}, {3, 1, 2}] &@list OleksandrR[list_] := Flatten[Outer[List, List@First[list], Rest[list], 1], {{2}, {1, 4}}] paradox2[list_] := Partition[Riffle[list[[1]], #], 2] & /@ Drop[list, 1] RM[list_] := FoldList[Transpose[{First@li...

plotting - How to draw lines between specified dots on ListPlot?

I would like to create a plot where I have unconnected dots and some connected. So far, I have figured out how to draw the dots. My code is the following: ListPlot[{{1, 1}, {2, 2}, {3, 3}, {4, 4}, {1, 4}, {2, 5}, {3, 6}, {4, 7}, {1, 7}, {2, 8}, {3, 9}, {4, 10}, {1, 10}, {2, 11}, {3, 12}, {4,13}, {2.5, 7}}, Ticks -> {{1, 2, 3, 4}, None}, AxesStyle -> Thin, TicksStyle -> Directive[Black, Bold, 12], Mesh -> Full] I have thought using ListLinePlot command, but I don't know how to specify to the command to draw only selected lines between the dots. Do have any suggestions/hints on how to do that? Thank you. Answer One possibility would be to use Epilog with Line : ListPlot[ {{1, 1}, {2, 2}, {3, 3}, {4, 4}, {1, 4}, {2, 5}, {3, 6}, {4, 7}, {1, 7}, {2, 8}, {3, 9}, {4, 10}, {1, 10}, {2, 11}, {3, 12}, {4, 13}, {2.5, 7}}, Ticks -> {{1, 2, 3, 4}, None}, AxesStyle -> Thin, TicksStyle -> Directive[Black, Bold, 12], Mesh -> Full, Epilog -> { Line[ ...