generated from cal-cs184/hw-webpage
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.html
459 lines (423 loc) · 19.9 KB
/
index.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
<head>
<style>
body {
padding: 100px;
width: 1000px;
margin: auto;
text-align: left;
font-weight: 300;
font-family: 'Open Sans', sans-serif;
color: #121212;
}
h1, h2, h3, h4 {
font-family: 'Source Sans Pro', sans-serif;
}
</style>
<title>CS 184 Mesh Editor</title>
<meta http-equiv="content-type" content="text/html; charset=utf-8" />
<link href="https://fonts.googleapis.com/css?family=Open+Sans|Source+Sans+Pro" rel="stylesheet">
</head>
<body>
<h1 align="middle">CS 184: Computer Graphics and Imaging, Spring 2024</h1>
<h1 align="middle">Project 4: Cloth Simulator</h1>
<h2 align="middle">Henry Khaung</h2>
<br><br>
<div>
<h2 align="middle">Overview</h2>
<p>
In this project, I simulated a piece of cloth under some constraints. For example,
I simulated how the cloth behaves when two points of it are pinned against a wall
and how it reacts when it falls onto nothing or onto another object. Since we render this based on physics,
we can change the animation of the cloth such as gravity, damping, and density. In the first part of the
project, I first have to build the cloth by using a grid of point masses and springs. Then, I added
physics of motion to it by using Verlet integration. Finally, I added shaders to add more visual effects to the cloth.
</p>
<h2 align="middle">Part I: Masses and springs</h2>
<p>In this part, we created a grid of point masses and springs for a sheet of cloth by dividing
the cloth up into evenly spaced point masses and and then connecting nearby masses with springs.
Essentially, a cloth is made up of point masses each connected to other point masses via springs.
We use 3 different type of springs: structural, shearing, and bending to further constrain the cloth
and to prevent any deformations the cloth may have.
</p>
<div align="center">
<table style="width=100%">
<tr>
<td align="middle">
<img src="part1all_constraints.png" width="480px" />
<figcaption align="middle">Wireframe with all constraints</figcaption>
</td>
</tr>
<tr>
<td align="middle">
<img src="part1no_shearing.png" width="480px" />
<figcaption align="middle">Wireframe without any shearing constraints</figcaption>
</td>
</tr>
<tr>
<td align="middle">
<img src="part1only_shearing.png" width="480px" />
<figcaption align="middle">Wireframe with only shearing constraints</figcaption>
</td>
</tr>
</table>
</div>
<h2 align="middle">Part II: Simulation Via Numerical Integration</h2>
<p>
After setting up a system of point masses and springs, we can now simulate motion by applying the forces
on our cloth's point masses to figure out how they move from one time step to another. First, I compute
the total force acting on each mass. Here, we have external forces which are the parameters we can change
in real-time during simulation such as gravity and internal forces which are the forces from the spring
acting on the point masses. After computing the force acting on each point mass, I then used Verlet
integration to compute each point mass's change in position. Verlet integration is used instead of other
methods such as Euler's method because it is fairly accurate and easy to implement since we can approximate
our current velocity by change in position (current position - position from last time step). We also add
damping to help simulate loss of energy due to friction.
</p>
<p>
Then, to prevent the springs from being deformed during each time step, I implemented a feature based on
the SIGGRAPH 1995 Provot paper which constrains each spring to be at most 10% of its rest length and
perform any corrections to the point masses connected by that spring if needed.
</p>
<div align="middle">
<table style="width=100%">
<tr align="center">
<td>
<img src="part2_defaultstart.png" width="400px" />
<figcaption align="middle">Default settings at start state</figcaption>
</td>
<td>
<img src="part2_defaultint.png" width="400px" />
<figcaption align="middle">Default settings at intermediate state</figcaption>
</td>
</tr>
</table>
<img src="part2_defaultend.png" width="400px" />
<figcaption align="middle">Default settings at end state</figcaption>
<table style="width=100%">
<h3>Below we vary ks, density, and damping percentage:</h3>
<tr align="center">
<td>
<img src="part2_ksUPend.png" width="400px" />
<figcaption align="middle">ks = 50,000 N/m at end state</figcaption>
</td>
<td>
<img src="part2_ksDOWNend.png" width="400px" />
<figcaption align="middle">ks = 500 N/m at end state</figcaption>
</td>
</tr>
</table>
</div>
<p align="middle">A high ks essentially means the cloth is more stiff and therefore, it doesn't hang as low as
a cloth with low ks. With low ks, the cloth hangs lower and has deeper creases.
</p>
<div align="middle">
<table style="width=100%">
<tr align="center">
<td>
<img src="part2_densityUPend.png" width="400px" />
<figcaption align="middle">density = 1500 g/cm^2 at end state</figcaption>
</td>
<td>
<img src="part2_densityDOWNend.png" width="400px" />
<figcaption align="middle">density = 15 g/cm^2 at end state</figcaption>
</td>
</tr>
</table>
</div>
<p align="middle"> A higher density means the cloth is heavier and therefore, it hangs lower due to the weight. There are also more
folds at the top, middle part. With lower density, the cloth is lighter and sags less and there are less folds at the top, middle part. </p>
<div align="middle">
<table style="width=100%">
<tr align="center">
<td>
<img src="part2_dampingUPend.png" width="400px" />
<figcaption align="middle">damping = 0.2% at end state</figcaption>
</td>
<td>
<img src="part2_dampingDOWNend.png" width="400px" />
<figcaption align="middle">damping = 0% at end state (but still swinging)</figcaption>
</td>
</tr>
</table>
</div>
<p align="middle">With a higher damping percentage, the cloth falls slower and there are less ripples and bounces as it falls.
On the other hand, a lower damping percentage such as 0% makes the cloth fall really fast and makes it swing back and forth. In addition, there are
more ripples and bounces as it swings. This makes sense as the damping percentage controls how fast the springs lose energy; with a higher damping percentage,
the springs lose their energy faster and therefore, the cloth does not ripple and bounce as much as a cloth with a smaller damping percentage.
</p>
<div align="middle"><img src="pinned4.png" width="400px"/>
<figcaption align="middle">Cloth pinned at all 4 ends at default settings</figcaption></div>
<h2 align="middle">Part III: Handling Collisions With Other Objects</h2>
<p>
After implementing how different forces act on the cloth, we implemented how our cloth behave
when interacting with other objects and in this case, with planes and spheres.
Essentially, with any type of object, we check if each point mass is inside the object and if so,
we adjust the point mass's position and ensure that it stays just on the surface of the object.
If the point mass intersects/is inside the object, I compute the surface point of the intersection
as the projection of the point mass position to the surface of the object. Then, I compute
the correction vector to update the point mass's last position to be the surface point.
The point mass's new position is then its last position added with the correction vector
which is scaled down by a friction factor.
</p>
<div align="middle">
<table style="width=100%">
<tr align="center">
<td>
<img src="part3_500.png" width="400px" />
<figcaption align="middle">Sphere collision at resting state with ks = 500</figcaption>
</td>
<td>
<img src="part3_5000.png" width="400px" />
<figcaption align="middle">Sphere collision at resting state with ks = 5000</figcaption>
</td>
</tr>
</table>
<img src="part3_50000.png" width="400px" />
<figcaption align="middle">Sphere collision at resting state with ks = 50,000</figcaption>
</div>
<p>
Increasing ks results in less wrapping of the cloth around the sphere while decreasing it makes the cloth wrap more around the sphere
as well as have the edges drape more.
</p>
<div align="middle">
<img src="part3plane.png" width="400px" />
<figcaption align="middle">Plane collision at resting state</figcaption>
</div>
<h2 align="middle">Part IV: Handling Self-collisions</h2>
<p>
In part III, we only accounted for collisions with other objects but not with the cloth itself.
There can be a scenario in which the cloth falls onto itself and without self-collisions implemented,
we see the cloth clipping through itself and other strange behavior. In order to prevent this from
happening, we do spatial hashing. At each time step, we build a hash map that maps a float to a list of
point masses; the float represents a 3D box volume and the list of point masses correspond to the
point masses in that 3D box volume. After this hashmap is set up, we just have to check each point mass
in each 3D box volume and apply any corrections to any point mass that are too close to each other.
Using spatial hashing is more efficient since we do not need to loop through all pairs of point masses,
compute the distance between them, and check if they are within some threshold distance apart. The hash
map cuts down the number of checks we have to do!
</p>
<div align="middle">
<img src="part4_clipping.png" width="400px" />
<figcaption align="middle">Without self-collisions implemented, we get clipping.</figcaption>
<table style="width=100%">
<tr align="center">
<td>
<img src="part4_start.png" width="400px" />
<figcaption align="middle">Start with self-collisions implemented</figcaption>
</td>
<td>
<img src="part4_intermediate.png" width="400px" />
<figcaption align="middle">Intermediate state with self-collisions implemented</figcaption>
</td>
</tr>
</table>
<img src="part4_end.png" width="400px" />
<figcaption align="middle">Rest state of cloth with self-collisions implemented</figcaption>
</div>
<div align="middle">
<h3>
Increasing and Decreasing Density:
</h3>
<p>
Increasing density of the cloth results in more folds in the cloth and therefore takes longer to uncurl. On the other hand,
decreasing density leads to less folds and therefore takes less time to unfold.
</p>
<table style="width=100%">
<tr align="center">
<td>
<img src="part4_densityUP.png" width="400px" />
<figcaption align="middle">Density = 10,000 g/cm^2</figcaption>
</td>
<td>
<img src="part4_densityDOWN.png" width="400px" />
<figcaption align="middle">Density = 1 g/cm^2</figcaption>
</td>
</tr>
</table>
<h3>
Increasing and Decreasing ks:
</h3>
<p>
Increasing ks results in faster uncurling of the cloth. This makes sense since the cloth is more stiff, it quickly straightens out.
Decreasing it takes longer for the cloth to uncurl since it is more flexible.
</p>
<table style="width=100%">
<tr align="center">
<td>
<img src="part4_ksUP.png" width="400px" />
<figcaption align="middle">ks = 50,000 N/m</figcaption>
</td>
<td>
<img src="part4_ksDOWN.png" width="400px" />
<figcaption align="middle">ks = 50 N/m</figcaption>
</td>
</tr>
</table>
</div>
<h2 align="middle">Part V: Shaders</h2>
<p>
After implementing the physics of the cloth, I added shaders to our cloth. I added 5 shaders:
diffuse shading, Blinn-Phong shading, texture mapping, displacement and bump mapping, and finally
mirror reflections. To write the shaders, I used a GLSL shader. A shader program such as GLSL shader
works by running in parallel in GPU instead of the CPU. I used two shader types: vertex shaders and
fragment shaders. Vertex shaders apply transformations to vertices such as their positions and write
the final position of the vertex to `gl_Position` and forward this data to fragment shaders.
The fragment shaders compute and write a color into `out_color` which then is the final result that
we see as a visual effect in our cloth simulator.
</p>
<p>The diffuse shader is a shader that is independent of the view direction.</p>
<div align="middle">
<img src="part5_diffuse.png" width="480px" />
<figcaption align="middle">Diffuse shading</figcaption>
</div>
<p>The Blinn-Phong shading model is an extension of the diffuse shader in that it takes into account of
view direction. The model adds ambient lighting and specular reflection in addition to the diffuse
lighting from the diffuse shader. The model now has additional k constants that can be adjusted; they
represent the weights we can give to each lighting component of the model. The Ia term is the value of the
ambient light and the p value controls how large our specular reflection is.</p>
<div align="middle">
<table style="width=100%">
<tr align="center">
<td>
<img src="part5_phongAmbient.png" width="400px" />
<figcaption align="middle">Blinn-Phong with only ambient component (Ia is set to 0.1 so there is not much light which is why this is pretty dark)</figcaption>
</td>
<td>
<img src="part5_phongDiffuse.png" width="400px" />
<figcaption align="middle">Blinn-Phong with only diffuse component. This is just diffuse shading.</figcaption>
</td>
</tr>
<tr align="center">
<td>
<img src="part5_phongSpec.png" width="400px" />
<figcaption align="middle">Blinn-Phong with only specular component</figcaption>
</td>
<td>
<img src="part5_phong.png" width="400px" />
<figcaption align="middle">Entire Blinn-Phong model</figcaption>
</td>
</tr>
</table>
<p>Texture mapping is straightforward. We map a texture onto the objects in our scene by using the built-in
function `texture`.
</p>
<div align="middle">
<img src="part5_texture.png" width="480px" />
<figcaption align="middle">Texture mapping of Yoshi</figcaption>
</div>
<p>
Displacement mapping modifies the actual geometry of an object by displacing its vertices based on a height map.
On the other hand, bump mapping does not change any actual geometry but rather simulates the appearance of surface detail by
altering the surface normals of the object. Below you can see distinctly see that bump mapping does not change the geometry; it does
not create the deeper lines that we see with displacement.
</p>
<div align="middle">
<table style="width=100%">
<tr align="center">
<td>
<img src="part5_bumpsphere1.png" width="400px" />
<figcaption align="middle">Sphere bump mapping</figcaption>
</td>
<td>
<img src="part5_bump1.png" width="400px" />
<figcaption align="middle">Cloth on sphere bump mapping</figcaption>
</td>
</tr>
<tr align="center">
<td>
<img src="part5_dispsphere1.png" width="400px" />
<figcaption align="middle">Sphere displacement mapping</figcaption>
</td>
<td>
<img src="part5_disp1.png" width="400px" />
<figcaption align="middle">Cloth on sphere displacement mapping</figcaption>
</td>
</tr>
</table>
</div>
<div align="middle">
<p>Using the following flags such as `-o 16 -a 16` changes the resolution of the rendered spheres.
-o [number] controls the vertical resolution, and -a [number] controls the horizontal resolution. It is hard to tell if the resolution
of the sphere improved the visual look with bump mapping but with displacement mapping, we can see that the sphere is more realistically rendered with higher resolution.</p>
<table align="middle">
<tr align="center">
<td>
<img src="part5_res16bump.png" width="400px" />
<figcaption align="middle">Sphere resolution `-o 16 -a 16` with bump mapping</figcaption>
</td>
<td>
<img src="part5_res16disp.png" width="400px" />
<figcaption align="middle">Sphere resolution `-o 16 -a 16` with displacement mapping</figcaption>
</td>
</tr>
<tr align="center">
<td>
<img src="part5_res128bump.png" width="400px" />
<figcaption align="middle">Sphere resolution `-o 128 -a 128` with bump mapping</figcaption>
</td>
<td>
<img src="part5_res128disp.png" width="400px" />
<figcaption align="middle">Sphere resolution `-o 128 -a 128` with displacement mapping</figcaption>
</td>
</tr>
</table>
</div>
<p>Mirror shading is similar to texture mapping except we needed to trace a ray to determine the
reflected direction. After this, we simply sampled from the environment map.
</p>
<table style="width=100%">
<tr align="center">
<td>
<img src="part5_mirrorsphere.png" width="400px" />
<figcaption align="middle">Sphere with mirror shading</figcaption>
</td>
<td>
<img src="part5_mirror.png" width="400px" />
<figcaption align="middle">Cloth on sphere with mirror shading</figcaption>
</td>
</tr>
</table>
<p>
I also added a custom shader. I noticed that in the Blinn-Phong shader, light source angle cannot be changed so
I implemented this in this custom shader. I realized that I could just use the color wheel to adjust the angle of the
light source. Since this is already in clothSimulator.cpp as `u_color`, I added this to the custom shader's fragment
file and used it to determine light direction. Instead of the vector pointing from the current vertex that is being shaded
towards the light position, we simply use `u_color` and normalize it. With this
implemented, the value in the color wheel chosen by the user will determine the direction of the light source.
</p>
<table style="width=100%">
<tr align="center">
<td>
<img src="part5_withoutdirsphere.png" width="400px" />
<figcaption align="middle">What the sphere looks like without variable light angle (default setting)</figcaption>
</td>
<td>
<img src="part5_withoutdir.png" width="400px" />
<figcaption align="middle">What the cloth on sphere looks like without variable light angle (default setting)</figcaption>
</td>
</tr>
<tr align="center">
<td>
<img src="part5_changedirsphere.png" width="400px" />
<figcaption align="middle">After using the color wheel to change the lighting angle, our sphere looks like this</figcaption>
</td>
<td>
<img src="part5_changedir.png" width="400px" />
<figcaption align="middle">Similarly, the cloth on sphere after a change in lighting angle</figcaption>
</td>
</tr>
<tr align="center">
<td>
<img src="part5_changedir1sphere.png" width="400px" />
<figcaption align="middle">Light angle at close to center of sphere</figcaption>
</td>
<td>
<img src="part5_changedir1.png" width="400px" />
<figcaption align="middle">Light angle at close to center of sphere for cloth on sphere</figcaption>
</td>
</tr>
</table>
</body>
</html>