**Examples for Assignment 2** Transformations ======================================================================== Rotation ------------------------------------------------------------------------ `rotate( mesh, rotate )` Rotates a mesh ( or a selected part ) around an axis by number specified in the sliders. The unit used for rotation is radians. data:image/s3,"s3://crabby-images/05b08/05b08ff564af606e8538b7663a83b4993cbf7c26" alt="cube.obj rotateX 0.8" data:image/s3,"s3://crabby-images/acd88/acd885a88ad4743d1e50bb53467a9597b0e0d715" alt="cube.obj (selection) rotateX 0.8 " Warps ======================================================================== Twist ------------------------------------------------------------------------ `twist( mesh, factor )` Rotates all vertices `v` along the `Y` axis by `v.position.y * factor`. The units are again radians. data:image/s3,"s3://crabby-images/de738/de738e4c32636940184b1dd53c47ec0f6417a1fd" alt="large_cube.obj twist 1" data:image/s3,"s3://crabby-images/f7aa6/f7aa6e999f4fdbf3f671e63414b52616aeeba9bb" alt="large_cube.obj twist 2" Inflate ------------------------------------------------------------------------ `inflate( mesh, factor )` Moves all vertices in the direction of their respective vertex normals, proportional to a `factor`. For the cube, this operation will give a scaling effect. data:image/s3,"s3://crabby-images/b0704/b07047ba47ec8a9cc1a90995c62257a761708acf" alt="cube.obj inflate -1" data:image/s3,"s3://crabby-images/dd6af/dd6af602eac56cbb70453bc9c2018ed7c0dc62d8" alt="cube.obj inflate 1" data:image/s3,"s3://crabby-images/29ad5/29ad504ba704efb9eaee013d3b3d178d0bceffc2" alt="hand.obj inflate -0.1" data:image/s3,"s3://crabby-images/20afe/20afe1c57d77b725e3c4e2cd7e6d09de2207b7b2" alt="hand.obj inflate 0.1" It is possible to augment this operation by `averageEdgeLength`. This will give nicer results for the inflated hand - the creases are better preserved, and the inflate operation is now aware of the mesh scale. data:image/s3,"s3://crabby-images/13226/13226d036e0036fafe3d5300d3a0fbf7952c7dcb" alt="cube.obj inflate -1" data:image/s3,"s3://crabby-images/c144c/c144c511bdc0f5665b40c16d9858c94535ffb995" alt="cube.obj inflate 1" data:image/s3,"s3://crabby-images/55926/55926881256fcc90c514393d0429a22770c503c4" alt="hand.obj inflate -1" data:image/s3,"s3://crabby-images/1cb85/1cb8525986615bb31028f4604cca0055149cdb5a" alt="hand.obj inflate 1" Filters ======================================================================== Noise ------------------------------------------------------------------------ `Noise( mesh, factor )` Moves all vertices in the direction of their respective vertex normals, proportional to `mesh.averageEdgeLength(v) * factor * randomIn( -1, 1 )`. data:image/s3,"s3://crabby-images/ed47d/ed47db322bc4a788d5864f914af374851cc3d556" alt="large_cube.obj noise 0.5" data:image/s3,"s3://crabby-images/27f95/27f95a2bd01be7fa14d9cec4a1e98340792ef505" alt="large_cube.obj noise 1" Uniform Laplacian Smoothing ------------------------------------------------------------------------ `smooth( mesh, iter, delta, curvFlow, scaleDep, implicit )` This operation needs to be applied `iter` number of times. For each iteration, each vertex should be moved towards the average of itself and its neighbors. In other words, for a vertex `V` with `N` neighbors, its updated position (`V_new`) should be `V_new = V + (sum(n_i)-N*V)*delta`, where `n_i` are the neighboring vertices. Be careful when updating the position. During a single iteration you want to use original positions throughout the entire computation. Update the actual positions of the vertices only after all new locations have been calculated. This uniform weighting scheme should be active if `curvFlow` is false. In this example, the weights are not normalized to sum to `1`, hence small deltas should be used, as can be seen in the images below. data:image/s3,"s3://crabby-images/1ff00/1ff00d64b3d2b0f0f8398445c68252fc57795720" alt="cheetah.obj i=1 d=1" data:image/s3,"s3://crabby-images/a8605/a8605da55c45f00fa1f06af8753f5474c3f6693f" alt="cheetah.obj i=36 d=0.15" data:image/s3,"s3://crabby-images/fd45f/fd45f63312e0d6bb9c6916b9fdbc5d9a7cdedfea" alt="cheetah.obj i=3000 d=0.01" data:image/s3,"s3://crabby-images/ba1e1/ba1e11571fdffbca6652569ea7274637c2048b88" alt="sphere2res.obj (with halfedges visualized) i=35 d=0.15" Curvature-flow Laplacian Smoothing ------------------------------------------------------------------------ `smooth( mesh, iter, delta, curvFlow, scaleDep, implicit )` The same as uniform sampling, but with a different weighting scheme, which can be applied only to triangles. In this case, the update rule is `V_new = V + (sum(w_i * n_i) - sum(w_i)*V)*delta`, where the weights `w_i` are the cotangent weights explained in the slides. Use when `curvFlow` is true. As can be seen (especially in the cut between the two resolutions of the sphere example), this scheme preserves triangle shapes well, and thus preserves features better, but it is still sensitive to triangle sizes. Note that this scheme is even more sesitive to `delta` values then the uniform one. Also note that this scheme is sensitive to long and thin triangles. To overcome this problem on bad meshes, either add checks in the code, or run a few uniform iterations before using this scheme, as was used in the hand example. data:image/s3,"s3://crabby-images/57d61/57d610bd311fbb43dd858c844043e031972d6e92" alt="cheetah.obj i=30 d=0.4" data:image/s3,"s3://crabby-images/12b80/12b80a8e115d4040967509de020a6d5f3a0c086b" alt="cheetah.obj i=3000 d=0.02" data:image/s3,"s3://crabby-images/59f43/59f433c95b2ba6680a3dcbae0c3de20a29ad0d76" alt="hand.obj i=10 d=0.02 (uniform) i=100 d=0.04 (cotan)" data:image/s3,"s3://crabby-images/d42a5/d42a5d6fe6c7e4819eb1a953a1f262cfcb4ad35d" alt="sphere2res.obj (with halfedges visualized) i=35 d=0.3" Scale-Dependent Smoothing ------------------------------------------------------------------------ `smooth( mesh, iter, delta, curvFlow, scaleDep, implicit )` No matter the weighting scheme, when `scaleDep` is true you should multiply the vertex offset (or matrix row) by the *scale dependent* term. That is,the update rule is `V_new = V + M*(sum(w_i * n_i) - sum(w_i)*V)*delta`, where M is `average(A_v)/A_v' (average of the 1ring area of all vertices over the 1ring of `V`: the sum of areas of the faces adjacent to it). For convenience, you may want to scale the mesh to its original size after each iteration (could be done by keepying the bounding box diagonal to be the same length). data:image/s3,"s3://crabby-images/03b5d/03b5d736ea31c2e199ebf17b97111a2c5489e4e7" alt="cheetah.obj i=300 d=0.01 (uniform)" data:image/s3,"s3://crabby-images/95aa0/95aa0c4c3269b6f63c102d3bbde3de680124bfd6" alt="cheetah.obj i=300 d=0.02 (cotan)" data:image/s3,"s3://crabby-images/ff161/ff161a5d23346f564f4bf224e12318e3d16f3d17" alt="sphere2res.obj (with halfedges visualized)i=1000 d=0.01 (uniform)" data:image/s3,"s3://crabby-images/22d4b/22d4bbbe61dad5426cfabbf3a57a346779083c42" alt="sphere2res.obj (with halfedges visualized) i=1000 d=0.02 (cotan)" Implicit Smoothing ------------------------------------------------------------------------ `smooth( mesh, iter, delta, curvFlow, scaleDep, implicit )` Regardless of the choice of the actual Laplacian, if `implicit` is true, you should formulate the previous computations in a matricial form and solve the linear system 'V = V_new - M*L*delta*V_new'. For further reading refer to [this paper](http://w.multires.caltech.edu/pubs/ImplicitFairing.pdf). An implicit solution is completely robust to `delta` sizes, so you typically don't need more than one iteration. Use the `.lup()` method of 'math.js' to decompose the matrix, and reuse the decomposition 3 times, to solve for the `x`, `y` and `z` axes. Note that the decomposition is slow, so you may want to stick to low vertex-count meshes such as `hand-simple`, `teapot` and `cheetah'. Using better optimized packages (or languages) would render this operation immediate. data:image/s3,"s3://crabby-images/9f9bc/9f9bc2c40971505ff86c547ae63d19c770c2659c" alt="cheetah.obj i=1 d=30 (uniform, scale-dependent)" data:image/s3,"s3://crabby-images/6d7ea/6d7ea46ed5c53adc250affb1f3c0b8d8919a8a61" alt="cheetah.obj i=1 d=30 (cotan, scale-dependent)" data:image/s3,"s3://crabby-images/341da/341da421528f1c745fcdb5d4214da80a628c18bc" alt="hand-simple.obj i=1 d=30 (uniform, scale-dependent)" data:image/s3,"s3://crabby-images/53b58/53b589b4b70cc96dbd91371a5a2d6723e78721b7" alt="hand-simple.obj i=1 d=30 (cotan, not scale-dependent)" Sharpen ------------------------------------------------------------------------ `sharpen( mesh, iter, delta )` This operation needs to be applied `iter` number of times, similarly to Uniform Laplacian Smoothing. It is the exact same behavior, only in the opposite direction. Note that this operation is very sesitive to `delta` values. data:image/s3,"s3://crabby-images/6e27f/6e27feaac0a55346250ef419bfa2f5c78617e187" alt="cheetah.obj i=5 d=0.04" data:image/s3,"s3://crabby-images/64404/64404b6db63723a73a60f4058379515d1ca9afa9" alt="cheetah.obj i=5 d=0.02" data:image/s3,"s3://crabby-images/69bbf/69bbff189db0d085df29ba5ddb04713a46382caa" alt="cheetah.obj i=12 d=0.02" Curvature ------------------------------------------------------------------------ `curvature( mesh )` Depending on your choice of visualization, your results might vary greatly from this. However for the cheetah, what you should be seeing is high curvature values (bright yellow in our visualization) in places like the claws, nostrils or ears, and low curvature values (blue in our visualization) around the claws. This is due to the fact that Gaussian curvature is defined as product of principal curvatures. Your final colors will vary depending on what method you use to map the curvature to a suitable range for the colors. Notice that for the cube, all the vertices have the same value so they are all mapped to mid gray. data:image/s3,"s3://crabby-images/fb8b2/fb8b2ee5dba8ec6ef8ba8807eb1f6e152947c20b" alt="cube.obj curvature" data:image/s3,"s3://crabby-images/18909/18909495fe0ccc19c3afa079af16e10b1d0bd5fd" alt="cheetah.obj curvature" data:image/s3,"s3://crabby-images/07b1c/07b1cadf7019d7b3dedd625af4153537170a4031" alt="cow.obj curvature" Topology ======================================================================== Truncate ------------------------------------------------------------------------ `truncate( mesh, factor )` Creates an effect where each vertex is "cut off". It is possible to accomplish this using the `splitEdge()` and `splitFace()` functions. Topologically we are only adding vertices, so there is no need to remove any vertices. Think about the necessary topological and geometrical changes. data:image/s3,"s3://crabby-images/3cbaa/3cbaa0c907c1b9ab1a055377dbbf50fbed2d06f5" alt="cube.obj truncate 0.2" data:image/s3,"s3://crabby-images/d5213/d521364c35cb265c61045eea380dc3b4680ef5dd" alt="cube.obj truncate 0.4" data:image/s3,"s3://crabby-images/a34a9/a34a90a676a0ff96b156a95110fea954ef5a3c10" alt="cube.obj (selected) truncate 0.4" Extrude ------------------------------------------------------------------------ `extrude( mesh, factor )` The vertices of each face should be duplicated and moved along the normal of the face. The original face should be attached to the duplicated vertices. The set of original vertices and duplicated vertices should be connected together by new faces. This feature requires you to modify the half edge data structure by using the `splitFaceMakeEdge(...)`, `joinEdgeKillVertex( v1, v2, v3 )`, and `splitEdgeMakeVertex( v1, v2, f )` functions. Make sure you familiarize yourself with these functions before starting to code this one! data:image/s3,"s3://crabby-images/d249e/d249ebb84b4ebf66ca5cdbd569aa44cf4ce1b5d0" alt="cube.obj extrude 1" data:image/s3,"s3://crabby-images/10805/10805e43744b4d300384c1a9226ac13da527d66b" alt="cube.obj (selected) extrude 1" Split Long Edges ------------------------------------------------------------------------ `splitLong( mesh, factor )` Splits the longest edge in the mesh. Depending on the choice of which vertices you selected to connect to a newly created one, you can obtain different results. Notice that as you go through the iterations one of the newly added edges might be the longest one. data:image/s3,"s3://crabby-images/f38d0/f38d0ac6f7088779c5921022bba2928afc82c573" alt="cube.obj splitLong 0.1" data:image/s3,"s3://crabby-images/20be3/20be36d1b108b6e9ae139c7101e96985cd525726" alt="cube.obj splitLong 0.5" data:image/s3,"s3://crabby-images/6763a/6763a74a123c81c4dc411f317fc8dcd27a0f6018" alt="cube.obj splitLong 1.0" Triangle Topology ------------------------------------------------------------------------ `triSubdiv( mesh, levels )` Splits each face into triangles. This effect is applied `levels` number of times. The mesh should be triangulated before using this function. data:image/s3,"s3://crabby-images/ae11f/ae11f98a13df1fc12203c605f3ab70dc686ec144" alt="tetrahedron.obj triSubdiv 1" data:image/s3,"s3://crabby-images/6d340/6d340f3fe32b3a1b8b854e1b33e194398c8a6340" alt="tetrahedron.obj triSubdiv 3" Subdivision ======================================================================== Loop Subdivision ------------------------------------------------------------------------ `loop( mesh, levels )` Splits each face into triangles. This effect is applied `levels` number of times. The mesh should be triangulated before using this function. In the example images, we have used Warren weights. data:image/s3,"s3://crabby-images/0feb2/0feb2ff4d3ac459fd1737ab6048607875365037e" alt="tetrahedron.obj triSubdiv 1" data:image/s3,"s3://crabby-images/23f01/23f01eacc7ad4c21076b5d2a911450c1220e80ef" alt="tetrahedron.obj triSubdiv 3" data:image/s3,"s3://crabby-images/73abf/73abfccbcc4dd5451e8603c5e73ed400a149b2c1" alt="cheetah.obj triSubdiv 1" data:image/s3,"s3://crabby-images/24760/24760122bfde74f71ff1aeed1fa30bd62930d8dd" alt="cheetah.obj triSubdiv 3" data:image/s3,"s3://crabby-images/7be32/7be326377801cd8c5e0818835a3aeac85469c526" alt="tetrahedron.obj triSubdiv 1" data:image/s3,"s3://crabby-images/2bc5e/2bc5ed5d8bfb443ae5303074b02aad0ff3a12821" alt="tetrahedron.obj triSubdiv 3" data:image/s3,"s3://crabby-images/bfe78/bfe78eda80715fbfb984dc2fa675ce39645cafa0" alt="cheetah.obj triSubdiv 1" data:image/s3,"s3://crabby-images/0f9ef/0f9ef1cb0537d78e74771a43dd491e573fb7d56d" alt="cheetah.obj triSubdiv 3" Quad Topology ------------------------------------------------------------------------ `quadSubdiv( mesh, levels )` Splits each face into quads. This effect is applied `levels` number of times. data:image/s3,"s3://crabby-images/fe1bc/fe1bcdee3b98401a0c6fc40f5ee1474451f8870a" alt="cube.obj quadSubdiv 1" data:image/s3,"s3://crabby-images/8b038/8b038fbe26ae69b9aa3094bb14cc263000743338" alt="cube.obj quadSubdiv 2" data:image/s3,"s3://crabby-images/bf01a/bf01ad0d3cf1432880aec7339edf87c6340be44a" alt="cube.obj quadSubdiv 3" data:image/s3,"s3://crabby-images/2fef4/2fef4a59f5224db8b5b08c79a7a57deecbfd636f" alt="cube.obj (selected) quadSubdiv 1" data:image/s3,"s3://crabby-images/7570a/7570a96210f9a7d31e13730e6739de07e1a45a0c" alt="cube.obj (selected) quadSubdiv 2" data:image/s3,"s3://crabby-images/40853/40853a2a418add1e9c8187301b2aa77db0436c32" alt="cube.obj (selected) quadSubdiv 3" Catmull-Clark Subdivision ------------------------------------------------------------------------ `catmullClark( mesh, levels )` Splits each face into quads. This effect is applied `levels` number of times. Users update the rules as described in the lecture and precept slides. The order in which you should apply the geometrical changes is: modify the positions of the new edge midpoints, modify the positions of the newly created face centroids, modify the positions of the old vertices. *Tip*: When calculating the locations for the edge midpoints, take the average of the vertices' positions of that edge, and the average of centroids of adjacent faces. data:image/s3,"s3://crabby-images/3bc5c/3bc5cce8b89dd19863e71d47fa7d8f4a2a9a19b7" alt="cube.obj catmullClark 1" data:image/s3,"s3://crabby-images/9600c/9600c8177151fb11da9746761a8f4ea81852540f" alt="cube.obj catmullClark 2" data:image/s3,"s3://crabby-images/ab916/ab916fcaf36a16dfc277e57b2c631603c25d0f97" alt="cube.obj catmullClark 3" data:image/s3,"s3://crabby-images/f2adc/f2adc333d0f0a4f113ae7bdfd5208ad36209844b" alt="dodacahedron.obj catmullClark 1" data:image/s3,"s3://crabby-images/fec39/fec3930f2ecbb2fb203fcd8f0d224768edd3dc67" alt="dodacahedron.obj catmullClark 2" data:image/s3,"s3://crabby-images/dee57/dee571a774569453f05cdda964c83f7b6834f0ee" alt="dodacahedron.obj catmullClark 3" data:image/s3,"s3://crabby-images/dd03c/dd03cb9b64c205be6842e635a29eb9004156d20c" alt="cube.obj catmullClark 1" data:image/s3,"s3://crabby-images/a881e/a881ec4c48c6ccc292e8c3568aa9c61da7f60bea" alt="cube.obj catmullClark 2" data:image/s3,"s3://crabby-images/1769b/1769b48938dd1e3dd0bec78cfe298d6ad636cec2" alt="cube.obj catmullClark 3" data:image/s3,"s3://crabby-images/8d0d7/8d0d777fcef4559c502a696d6ee94cec1d691a3e" alt="dodacahedron.obj catmullClark 1" data:image/s3,"s3://crabby-images/5e012/5e012c0b5ca57e17e103d4d3ac960e45a4aa6ce1" alt="dodacahedron.obj catmullClark 2" data:image/s3,"s3://crabby-images/fc912/fc9125c765ea4c9564fffc8844f8479f9703fcbe" alt="dodacahedron.obj catmullClark 3"