Multi-view Reconstruction
A 3D object can be reconstructed from multiple 2D views. ImageDisplacements is used to determine the parallax from one view to the next. The larger the parallax shift, the closer the object behind the corresponding pixel. With this depth information, you can extrude the vertices of an object mesh obtained through ImageMesh and TriangulateMesh. The result is displayed via texture mapping on the 3D mesh object.
data:image/s3,"s3://crabby-images/04a43/04a43e4a705e474796fba86c06b7f734b123f231" alt="Click for copyable input"
Obtain an object mask of the central image.
data:image/s3,"s3://crabby-images/f4686/f4686383e5aa4fbcaad33aecc4826da9cacf1167" alt="Click for copyable input"
mask = Erosion[Binarize[imgs[[2]], 0], 1];
Determine the parallax in the left and right images with respect to the center image.
data:image/s3,"s3://crabby-images/a4793/a4793bae8f8cf7f7451e61d57541f57e7dc69a42" alt="Click for copyable input"
parallaxL = First@ImageDisplacements[imgs[[{2, 1}]]];
data:image/s3,"s3://crabby-images/df4c8/df4c8259287c21beaf9c640a28cb9f8517b25c6f" alt="Click for copyable input"
parallaxR = First@ImageDisplacements[imgs[[{2, 3}]]];
Combine these displacements in a total parallax.
data:image/s3,"s3://crabby-images/9db17/9db1794cd9d8f4b11de8320663fd03e28b5a029c" alt="Click for copyable input"
parallax = parallaxL - parallaxR;
In the given setting, the component of the parallax is roughly proportional to the depth of the respective pixel source.
data:image/s3,"s3://crabby-images/74f0e/74f0ec18d691d688e4514b23c86fb37438d28f86" alt="Click for copyable input"
depth = Blur@
Opening[ImageMultiply[ImageAdjust@Image[parallax[[All, All, 1]]],
mask], DiskMatrix[4]]
data:image/s3,"s3://crabby-images/375c3/375c3de2a4bcc678e7716bef9d17d3e6243c63e4" alt=""
Construct a depth function.
data:image/s3,"s3://crabby-images/62388/62388b9f2dc6b4b22a485ee1253f4a74bf38d6c8" alt="Click for copyable input"
depthFunction = ListInterpolation[Transpose@Reverse@ImageData[depth]];
Obtain an object mesh with a refinement that depends on the object's saliency.
data:image/s3,"s3://crabby-images/5bddf/5bddf1aa606d12f8fdc2441da4a0f7955dc783b4" alt="Click for copyable input"
resolution = ImageAdjust@ImageSaliencyFilter[imgs[[2]]];
data:image/s3,"s3://crabby-images/62b77/62b77a2c7990bb416b7ab0f3dfd3e652c460201d" alt="Click for copyable input"
resolutionFunction =
ListInterpolation[Transpose@Reverse@ImageData@resolution];
data:image/s3,"s3://crabby-images/6ab5d/6ab5d27d7ad6f9992afdffabfbd40611d6800870" alt="Click for copyable input"
\[CapitalOmega] = TriangulateMesh[
ImageMesh[Erosion[mask, DiskMatrix[2]]],
MeshRefinementFunction ->
Function[{vertices, area},
area > 32 + 512 (1 - resolutionFunction @@ Mean[vertices])^6]
]
data:image/s3,"s3://crabby-images/161a1/161a11d57c7a0beab8c91dbd9de764c69c997d70" alt=""
Extrude the object mesh with the depth function.
data:image/s3,"s3://crabby-images/5f21e/5f21ea41c5e2ca59dd33407d8218b0f0e8402867" alt="Click for copyable input"
Graphics3D[
GraphicsComplex[
Apply[{##, depthFunction[##]} &,
MeshCoordinates[\[CapitalOmega]], {1}],
{EdgeForm[], MeshCells[\[CapitalOmega], 2]}
],
PlotRange -> Append[Thread[{0, ImageDimensions[mask]}], {0, 1}],
BoxRatios -> {1, 1, 2/3},
ViewPoint -> Top
]
data:image/s3,"s3://crabby-images/b51f8/b51f899018b671fd25d91e7a809a540ba6c345d7" alt=""
Extract the object texture from the central image.
data:image/s3,"s3://crabby-images/774f5/774f5e0d527aafb8c869746ecb84dd5f1af517bb" alt="Click for copyable input"
texture = SetAlphaChannel[imgs[[2]], mask]
data:image/s3,"s3://crabby-images/6cae7/6cae7def6d78bcd5fa5030115bb31d63ecba0e54" alt=""
Map the texture onto the 3D object.
data:image/s3,"s3://crabby-images/b7691/b769176400432138eb6b1713a776458e624782d6" alt="Click for copyable input"
Graphics3D[
{Texture[texture],
GraphicsComplex[
Apply[{##, depthFunction[##]} &,
MeshCoordinates[\[CapitalOmega]], {1}],
{EdgeForm[], MeshCells[\[CapitalOmega], 2]},
VertexTextureCoordinates ->
Map[#/ImageDimensions[texture] &, MeshCoordinates[\[CapitalOmega]]]
]
},
PlotRange -> Append[Thread[{0, ImageDimensions[mask]}], {0, 1}],
BoxRatios -> {1, 1, 2/3},
Lighting -> {{"Ambient", White}},
ViewPoint -> Top,
Boxed -> False
]
data:image/s3,"s3://crabby-images/b0b7c/b0b7c81e54c8c3c0d4befdd6ff6e257748ecf9f5" alt=""