Deploying to gh-pages from @ sotrh/learn-wgpu@b6f66e0565 🚀

gh-pages
sotrh 2 years ago
parent 9df59ee02e
commit 59680370d6

@ -8,11 +8,11 @@
<meta name="description" content="">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.cbd0d45d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/39.1c5b4548.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.576af98e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/17.7a15a658.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.e74dc361.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.69a9ea7e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.e284d6f3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/22.ae08829d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.9d07c0eb.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.6982a462.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.32361a22.js"><link rel="prefetch" href="/learn-wgpu/assets/js/26.9b876ccf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.9fe49a03.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.c212a80a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.b61fde25.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.25585faa.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.acd07b95.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.a51ea3c5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.41e9897b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.2d445b19.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.9b4d649e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.e0f57de3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.155fa5a2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.d3e6544b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.2bed1539.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.9a3ae8d9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.4d927984.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.bcb97198.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.c87751e7.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/39.1c5b4548.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.e0b6e526.js"><link rel="prefetch" href="/learn-wgpu/assets/js/17.7a15a658.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.c8349761.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.2469c630.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.3972e66e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/22.ae08829d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.d649043b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.5b5cf4ba.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.a63ee8c2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/26.9b876ccf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.064a85a6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.fb933617.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.2e0b419b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.09885c9c.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.19c7b8dd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.9a017b74.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.695eaab4.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.1ae459af.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.bb1a0e6a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.34968282.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.58137882.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.c80ee966.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.740cfe90.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.5be8c6e7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.2e48975d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.ec58808b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="stylesheet" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css">
</head>
<body>
<div id="app" data-server-rendered="true"><div class="theme-container"><div class="theme-default-content"><h1>404</h1> <blockquote>How did we get here?</blockquote> <a href="/learn-wgpu/" class="router-link-active">Take me home.</a></div></div><div class="global-ui"><!----></div></div>
<script src="/learn-wgpu/assets/js/app.cbd0d45d.js" defer></script><script src="/learn-wgpu/assets/js/39.1c5b4548.js" defer></script>
<div id="app" data-server-rendered="true"><div class="theme-container"><div class="theme-default-content"><h1>404</h1> <blockquote>That's a Four-Oh-Four.</blockquote> <a href="/learn-wgpu/" class="router-link-active">Take me home.</a></div></div><div class="global-ui"><!----></div></div>
<script src="/learn-wgpu/assets/js/app.c87751e7.js" defer></script><script src="/learn-wgpu/assets/js/39.1c5b4548.js" defer></script>
</body>
</html>

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

@ -1 +1 @@
(window.webpackJsonp=window.webpackJsonp||[]).push([[45],{593:function(e,t,o){"use strict";o.r(t);var s=o(23),r=Object(s.a)({},(function(){var e=this,t=e.$createElement,o=e._self._c||t;return o("ContentSlotsDistributor",{attrs:{"slot-key":e.$parent.slotKey}},[o("h1",{attrs:{id:"foreword"}},[o("a",{staticClass:"header-anchor",attrs:{href:"#foreword"}},[e._v("#")]),e._v(" Foreword")]),e._v(" "),o("p",[e._v("The articles in this section are not meant to be tutorials. They are showcases of the various things you can do with "),o("code",[e._v("wgpu")]),e._v(". I won't go over specifics of creating "),o("code",[e._v("wgpu")]),e._v(" resources, as those will be covered elsewhere. The code for these examples is still available however, and will be accessible on Github.")])])}),[],!1,null,null,null);t.default=r.exports}}]);
(window.webpackJsonp=window.webpackJsonp||[]).push([[45],{593:function(e,t,o){"use strict";o.r(t);var s=o(23),r=Object(s.a)({},(function(){var e=this,t=e.$createElement,o=e._self._c||t;return o("ContentSlotsDistributor",{attrs:{"slot-key":e.$parent.slotKey}},[o("h1",{attrs:{id:"foreword"}},[o("a",{staticClass:"header-anchor",attrs:{href:"#foreword"}},[e._v("#")]),e._v(" Foreword")]),e._v(" "),o("p",[e._v("The articles in this section are not meant to be tutorials. They are showcases of the various things you can do with "),o("code",[e._v("wgpu")]),e._v(". I won't go over the specifics of creating "),o("code",[e._v("wgpu")]),e._v(" resources, as those will be covered elsewhere. The code for these examples is still available however and will be accessible on Github.")])])}),[],!1,null,null,null);t.default=r.exports}}]);

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

@ -7,7 +7,7 @@
<meta name="generator" content="VuePress 1.9.7">
<meta name="description" content="">
<meta property="article:modified_time" content="2022-05-14T18:41:58.000Z">
<meta property="article:modified_time" content="2022-05-14T18:44:34.000Z">
<meta property="og:site_name" content="Learn Wgpu">
<meta property="og:title" content="Dependencies and the window">
<meta property="og:type" content="website">
@ -19,7 +19,7 @@
<meta name="twitter:data1" content="Benjamin Hansen">
<meta name="twitter:creator" content="https://twitter.com/sotrh760">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.cbd0d45d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/41.d3e6544b.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/22.ae08829d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/26.9b876ccf.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.576af98e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.e74dc361.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.69a9ea7e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.e284d6f3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.9d07c0eb.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.6982a462.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.32361a22.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.9fe49a03.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.c212a80a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.b61fde25.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.25585faa.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.acd07b95.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.a51ea3c5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.41e9897b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.2d445b19.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.9b4d649e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.e0f57de3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.155fa5a2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.2bed1539.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.9a3ae8d9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.4d927984.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.bcb97198.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.c87751e7.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/41.c80ee966.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/22.ae08829d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/26.9b876ccf.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.e0b6e526.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.c8349761.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.2469c630.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.3972e66e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.d649043b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.5b5cf4ba.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.a63ee8c2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.064a85a6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.fb933617.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.2e0b419b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.09885c9c.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.19c7b8dd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.9a017b74.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.695eaab4.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.1ae459af.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.bb1a0e6a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.34968282.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.58137882.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.740cfe90.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.5be8c6e7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.2e48975d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.ec58808b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="stylesheet" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css">
</head>
<body>
@ -28,9 +28,9 @@
<span class="token key property">env_logger</span> <span class="token punctuation">=</span> <span class="token string">&quot;0.9&quot;</span>
<span class="token key property">log</span> <span class="token punctuation">=</span> <span class="token string">&quot;0.4&quot;</span>
<span class="token key property">wgpu</span> <span class="token punctuation">=</span> <span class="token string">&quot;0.12&quot;</span>
</code></pre></div><h2 id="using-rust-s-new-resolver"><a href="#using-rust-s-new-resolver" class="header-anchor">#</a> Using Rust's new resolver</h2> <p>As of version 0.10, wgpu require's cargo's <a href="https://doc.rust-lang.org/cargo/reference/resolver.html#feature-resolver-version-2" target="_blank" rel="noopener noreferrer">newest feature resolver<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a>, which is the default in the 2021 edition (any new project started with Rust version 1.56.0 or newer). However, if you are still using the 2018 edition, you must include <code>resolver = &quot;2&quot;</code> in either the <code>[package]</code> section of <code>Cargo.toml</code> if you are working on a single crate, or the <code>[workspace]</code> section of the root <code>Cargo.toml</code> in a workspace.</p> <h2 id="env-logger"><a href="#env-logger" class="header-anchor">#</a> env_logger</h2> <p>It is very important to enable logging via <code>env_logger::init();</code>.
</code></pre></div><h2 id="using-rust-s-new-resolver"><a href="#using-rust-s-new-resolver" class="header-anchor">#</a> Using Rust's new resolver</h2> <p>As of version 0.10, wgpu requires cargo's <a href="https://doc.rust-lang.org/cargo/reference/resolver.html#feature-resolver-version-2" target="_blank" rel="noopener noreferrer">newest feature resolver<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a>, which is the default in the 2021 edition (any new project started with Rust version 1.56.0 or newer). However, if you are still using the 2018 edition, you must include <code>resolver = &quot;2&quot;</code> in either the <code>[package]</code> section of <code>Cargo.toml</code> if you are working on a single crate, or the <code>[workspace]</code> section of the root <code>Cargo.toml</code> in a workspace.</p> <h2 id="env-logger"><a href="#env-logger" class="header-anchor">#</a> env_logger</h2> <p>It is very important to enable logging via <code>env_logger::init();</code>.
When wgpu hits any error it panics with a generic message, while logging the real error via the log crate.
This means if you don't include <code>env_logger::init()</code> wgpu will fail silently, leaving you very confused!</p> <h2 id="the-code"><a href="#the-code" class="header-anchor">#</a> The code</h2> <p>There's not much going on here yet, so I'm just going to post the code in full. Just paste this into your <code>lib.rs</code> or equivalent.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">use</span> <span class="token namespace">winit<span class="token punctuation">::</span></span><span class="token punctuation">{</span>
This means if you don't include <code>env_logger::init()</code>, wgpu will fail silently, leaving you very confused!</p> <h2 id="the-code"><a href="#the-code" class="header-anchor">#</a> The code</h2> <p>There's not much going on here yet, so I'm just going to post the code in full. Just paste this into your <code>lib.rs</code> or equivalent.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">use</span> <span class="token namespace">winit<span class="token punctuation">::</span></span><span class="token punctuation">{</span>
<span class="token namespace">event<span class="token punctuation">::</span></span><span class="token operator">*</span><span class="token punctuation">,</span>
<span class="token namespace">event_loop<span class="token punctuation">::</span></span><span class="token punctuation">{</span><span class="token class-name">ControlFlow</span><span class="token punctuation">,</span> <span class="token class-name">EventLoop</span><span class="token punctuation">}</span><span class="token punctuation">,</span>
<span class="token namespace">window<span class="token punctuation">::</span></span><span class="token class-name">WindowBuilder</span><span class="token punctuation">,</span>
@ -62,14 +62,14 @@ This means if you don't include <code>env_logger::init()</code> wgpu will fail s
<span class="token punctuation">}</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span>
</code></pre></div><p>All this does is create a window, and keep it open until the user closes it, or presses escape. Next we'll need a <code>main.rs</code> to actually run the code. It's quite simple it just imports <code>run()</code> and, well runs it!</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">use</span> <span class="token namespace">tutorial1_window<span class="token punctuation">::</span></span>run<span class="token punctuation">;</span>
</code></pre></div><p>All this does is create a window, and keep it open until the user closes it, or presses escape. Next, we'll need a <code>main.rs</code> to run the code. It's quite simple, it just imports <code>run()</code> and, well, runs it!</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">use</span> <span class="token namespace">tutorial1_window<span class="token punctuation">::</span></span>run<span class="token punctuation">;</span>
<span class="token keyword">fn</span> <span class="token function-definition function">main</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
<span class="token namespace">pollster<span class="token punctuation">::</span></span><span class="token function">block_on</span><span class="token punctuation">(</span><span class="token function">run</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token function">run</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span>
</code></pre></div><p>If you only want to support desktop, that's all you have to do! In the next tutorial we'll actually start using wgpu!</p> <h2 id="added-support-for-the-web"><a href="#added-support-for-the-web" class="header-anchor">#</a> Added support for the web</h2> <p>If I go through this tutorial about WebGPU and never talk about using it on the web, then I'd hardly call this tutorial complete. Fortunately getting a wgpu application running in a browser is not too difficult once you get things set up.</p> <p>Lets start with the changes we need to make to are <code>Cargo.toml</code>:</p> <div class="language-toml extra-class"><pre class="language-toml"><code><span class="token punctuation">[</span><span class="token table class-name">lib</span><span class="token punctuation">]</span>
</code></pre></div><p>If you only want to support desktops, that's all you have to do! In the next tutorial, we'll start using wgpu!</p> <h2 id="added-support-for-the-web"><a href="#added-support-for-the-web" class="header-anchor">#</a> Added support for the web</h2> <p>If I go through this tutorial about WebGPU and never talk about using it on the web, then I'd hardly call this tutorial complete. Fortunately getting a wgpu application running in a browser is not too difficult once you get things set up.</p> <p>Let's start with the changes we need to make to are <code>Cargo.toml</code>:</p> <div class="language-toml extra-class"><pre class="language-toml"><code><span class="token punctuation">[</span><span class="token table class-name">lib</span><span class="token punctuation">]</span>
<span class="token key property">crate-type</span> <span class="token punctuation">=</span> <span class="token punctuation">[</span><span class="token string">&quot;cdylib&quot;</span><span class="token punctuation">,</span> <span class="token string">&quot;rlib&quot;</span><span class="token punctuation">]</span>
</code></pre></div><p>These lines tell cargo that we want to allow our crate to build a native Rust static library (rlib) and a C/C++ compatible library (cdylib). We need the rlib if we want to run wgpu in a desktop environment. We need the cdylib to create the Web Assembly that browser will actually run.</p> <div class="note"><h2 id="web-assembly"><a href="#web-assembly" class="header-anchor">#</a> Web Assembly</h2> <p>Web Assembly ie WASM, is a binary format supported by most modern browsers that allows lower level languages such as Rust to run on a web page. This allows us to right the bulk of our application in Rust and use a few lines of Javascript to get it running in a web browser.</p></div> <p>Now all we need are some more dependencies that are specific to running in WASM:</p> <div class="language-toml extra-class"><pre class="language-toml"><code><span class="token punctuation">[</span><span class="token table class-name">dependencies</span><span class="token punctuation">]</span>
</code></pre></div><p>These lines tell cargo that we want to allow our crate to build a native Rust static library (rlib) and a C/C++ compatible library (cdylib). We need rlib if we want to run wgpu in a desktop environment. We need cdylib to create the Web Assembly that the browser will run.</p> <div class="note"><h2 id="web-assembly"><a href="#web-assembly" class="header-anchor">#</a> Web Assembly</h2> <p>Web Assembly i.e. WASM, is a binary format supported by most modern browsers that allows lower-level languages such as Rust to run on a web page. This allows us to write the bulk of our application in Rust and use a few lines of Javascript to get it running in a web browser.</p></div> <p>Now, all we need are some more dependencies that are specific to running in WASM:</p> <div class="language-toml extra-class"><pre class="language-toml"><code><span class="token punctuation">[</span><span class="token table class-name">dependencies</span><span class="token punctuation">]</span>
<span class="token key property">cfg-if</span> <span class="token punctuation">=</span> <span class="token string">&quot;1&quot;</span>
<span class="token comment"># the other regular dependencies...</span>
@ -83,24 +83,24 @@ This means if you don't include <code>env_logger::init()</code> wgpu will fail s
<span class="token string">&quot;Window&quot;</span><span class="token punctuation">,</span>
<span class="token string">&quot;Element&quot;</span><span class="token punctuation">,</span>
<span class="token punctuation">]</span><span class="token punctuation">}</span>
</code></pre></div><p>The <a href="https://docs.rs/cfg-if" target="_blank" rel="noopener noreferrer">cfg-if<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> crate adds a macro that makes using platform specific code more manageable.</p> <p>The <code>[target.'cfg(target_arch = &quot;wasm32&quot;)'.dependencies]</code> line tells cargo to only include these dependencies if we are targeting the <code>wasm32</code> architecture. The next few dependencies are just make interfacing with javascript a lot easier.</p> <ul><li><a href="https://docs.rs/console_error_panic_hook" target="_blank" rel="noopener noreferrer">console_error_panic_hook<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> configures the <code>panic!</code> macro to send errors to the javascript console. Without this when you encounter panics, you'll be left in the dark for what caused them.</li> <li><a href="https://docs.rs/console_log" target="_blank" rel="noopener noreferrer">console_log<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> implements the <a href="https://docs.rs/log" target="_blank" rel="noopener noreferrer">log<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> API. It sends all logs to the javascript console. It can be configured to only send logs of a particular log level. This is also great for debugging.</li> <li>We need to enable WebGL feature on wgpu if we want to run on most current browsers. Support is in the works for using the WebGPU api directly, but that is only possible on experimental versions of browsers such as Firefox Nightly and Chrome Canary.<br>
You're welcome to test this code on these browsers (and the wgpu devs would appreciate it as well), but for sake of simplicity I'm going to stick to using the WebGL feature until the WebGPU api gets to a more stable state.<br>
If you want more details check out the guide for compiling for the web on <a href="https://github.com/gfx-rs/wgpu/wiki/Running-on-the-Web-with-WebGPU-and-WebGL" target="_blank" rel="noopener noreferrer">wgpu's repo<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a></li> <li><a href="https://docs.rs/wasm-bindgen" target="_blank" rel="noopener noreferrer">wasm-bindgen<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> is the most important dependency in this list. It's responsible for generating the boilerplate code that will tell the browser how to use our crate. It also allows us to expose methods in Rust that will can be used in Javascript, and vice-versa.<br>
I won't get into the specifics of wasm-bindgen, so if you need a primer (or just a refresher) check out <a href="https://rustwasm.github.io/wasm-bindgen/" target="_blank" rel="noopener noreferrer">this<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a></li> <li><a href="https://docs.rs/web-sys" target="_blank" rel="noopener noreferrer">web-sys<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> is a crate that includes many methods and structures that are available in a normal javascript application: <code>get_element_by_id</code>, <code>append_child</code>. The features listed are only the bare minimum of what we need currently.</li></ul> <h2 id="more-code"><a href="#more-code" class="header-anchor">#</a> More code</h2> <p>First we need to import <code>wasm-bindgen</code> in <code>lib.rs</code>:</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token attribute attr-name">#[cfg(target_arch=<span class="token string">&quot;wasm32&quot;</span>)]</span>
</code></pre></div><p>The <a href="https://docs.rs/cfg-if" target="_blank" rel="noopener noreferrer">cfg-if<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> crate adds a macro that makes using platform-specific code more manageable.</p> <p>The <code>[target.'cfg(target_arch = &quot;wasm32&quot;)'.dependencies]</code> line tells cargo to only include these dependencies if we are targeting the <code>wasm32</code> architecture. The next few dependencies just make interfacing with javascript a lot easier.</p> <ul><li><a href="https://docs.rs/console_error_panic_hook" target="_blank" rel="noopener noreferrer">console_error_panic_hook<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> configures the <code>panic!</code> macro to send errors to the javascript console. Without this when you encounter panics, you'll be left in the dark about what caused them.</li> <li><a href="https://docs.rs/console_log" target="_blank" rel="noopener noreferrer">console_log<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> implements the <a href="https://docs.rs/log" target="_blank" rel="noopener noreferrer">log<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> API. It sends all logs to the javascript console. It can be configured to only send logs of a particular log level. This is also great for debugging.</li> <li>We need to enable WebGL feature on wgpu if we want to run on most current browsers. Support is in the works for using the WebGPU api directly, but that is only possible on experimental versions of browsers such as Firefox Nightly and Chrome Canary.<br>
You're welcome to test this code on these browsers (and the wgpu devs would appreciate it as well), but for sake of simplicity, I'm going to stick to using the WebGL feature until the WebGPU api gets to a more stable state.<br>
If you want more details check out the guide for compiling for the web on <a href="https://github.com/gfx-rs/wgpu/wiki/Running-on-the-Web-with-WebGPU-and-WebGL" target="_blank" rel="noopener noreferrer">wgpu's repo<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a></li> <li><a href="https://docs.rs/wasm-bindgen" target="_blank" rel="noopener noreferrer">wasm-bindgen<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> is the most important dependency in this list. It's responsible for generating the boilerplate code that will tell the browser how to use our crate. It also allows us to expose methods in Rust that can be used in Javascript, and vice-versa.<br>
I won't get into the specifics of wasm-bindgen, so if you need a primer (or just a refresher) check out <a href="https://rustwasm.github.io/wasm-bindgen/" target="_blank" rel="noopener noreferrer">this<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a></li> <li><a href="https://docs.rs/web-sys" target="_blank" rel="noopener noreferrer">web-sys<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> is a crate that includes many methods and structures that are available in a normal javascript application: <code>get_element_by_id</code>, <code>append_child</code>. The features listed are only the bare minimum of what we need currently.</li></ul> <h2 id="more-code"><a href="#more-code" class="header-anchor">#</a> More code</h2> <p>First, we need to import <code>wasm-bindgen</code> in <code>lib.rs</code>:</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token attribute attr-name">#[cfg(target_arch=<span class="token string">&quot;wasm32&quot;</span>)]</span>
<span class="token keyword">use</span> <span class="token namespace">wasm_bindgen<span class="token punctuation">::</span>prelude<span class="token punctuation">::</span></span><span class="token operator">*</span><span class="token punctuation">;</span>
</code></pre></div><p>Next we need to tell wasm-bindgen to run our <code>run()</code> function when the WASM is loaded:</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token attribute attr-name">#[cfg_attr(target_arch=<span class="token string">&quot;wasm32&quot;</span>, wasm_bindgen(start))]</span>
</code></pre></div><p>Next, we need to tell wasm-bindgen to run our <code>run()</code> function when the WASM is loaded:</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token attribute attr-name">#[cfg_attr(target_arch=<span class="token string">&quot;wasm32&quot;</span>, wasm_bindgen(start))]</span>
<span class="token keyword">pub</span> <span class="token keyword">async</span> <span class="token keyword">fn</span> <span class="token function-definition function">run</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
<span class="token comment">// snipped...</span>
<span class="token punctuation">}</span>
</code></pre></div><p>Then we need to toggle what logger we are using based on if we are in WASM land or not. Add the following to the top of the run function replacing the <code>env_logger::init()</code> line:</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token namespace">cfg_if<span class="token punctuation">::</span></span><span class="token macro property">cfg_if!</span> <span class="token punctuation">{</span>
<span class="token keyword">if</span> <span class="token attribute attr-name">#[cfg(target_arch = <span class="token string">&quot;wasm32&quot;</span>)]</span> <span class="token punctuation">{</span>
<span class="token namespace">std<span class="token punctuation">::</span>panic<span class="token punctuation">::</span></span><span class="token function">set_hook</span><span class="token punctuation">(</span><span class="token class-name">Box</span><span class="token punctuation">::</span><span class="token function">new</span><span class="token punctuation">(</span><span class="token namespace">console_error_panic_hook<span class="token punctuation">::</span></span>hook<span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token namespace">console_log<span class="token punctuation">::</span></span><span class="token function">init_with_level</span><span class="token punctuation">(</span><span class="token namespace">log<span class="token punctuation">::</span></span><span class="token class-name">Level</span><span class="token punctuation">::</span><span class="token class-name">Warn</span><span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">expect</span><span class="token punctuation">(</span><span class="token string">&quot;Could't initialize logger&quot;</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token namespace">console_log<span class="token punctuation">::</span></span><span class="token function">init_with_level</span><span class="token punctuation">(</span><span class="token namespace">log<span class="token punctuation">::</span></span><span class="token class-name">Level</span><span class="token punctuation">::</span><span class="token class-name">Warn</span><span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">expect</span><span class="token punctuation">(</span><span class="token string">&quot;Couldn't initialize logger&quot;</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span> <span class="token keyword">else</span> <span class="token punctuation">{</span>
<span class="token namespace">env_logger<span class="token punctuation">::</span></span><span class="token function">init</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span>
<span class="token punctuation">}</span>
</code></pre></div><p>This will setup <code>console_log</code> and <code>console_error_panic_hook</code> in a web build, and will initialize <code>env_logger</code> in a normal build. This is important as <code>env_logger</code> doesn't support Web Assembly at the moment.</p> <p>Next, after we create our event loop and window, we need to add an canvas to the html document that we will host our application:</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token attribute attr-name">#[cfg(target_arch = <span class="token string">&quot;wasm32&quot;</span>)]</span>
</code></pre></div><p>This will set up <code>console_log</code> and <code>console_error_panic_hook</code> in a web build, and will initialize <code>env_logger</code> in a normal build. This is important as <code>env_logger</code> doesn't support Web Assembly at the moment.</p> <p>Next, after we create our event loop and window, we need to add a canvas to the HTML document that we will host our application:</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token attribute attr-name">#[cfg(target_arch = <span class="token string">&quot;wasm32&quot;</span>)]</span>
<span class="token punctuation">{</span>
<span class="token comment">// Winit prevents sizing with CSS, so we have to set</span>
<span class="token comment">// the size manually when on web.</span>
@ -118,11 +118,11 @@ I won't get into the specifics of wasm-bindgen, so if you need a primer (or just
<span class="token punctuation">}</span><span class="token punctuation">)</span>
<span class="token punctuation">.</span><span class="token function">expect</span><span class="token punctuation">(</span><span class="token string">&quot;Couldn't append canvas to document body.&quot;</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span>
</code></pre></div><div class="note"><p>The <code>&quot;wasm-example&quot;</code> id is specific to my project (aka. this tutorial). You can substitute this for what ever id your using in your html. Alternatively you could add the canvas directly to the <code>&lt;body&gt;</code> as they do in the wgpu repo. This part is ultimately up to you.</p></div> <p>That's all the web specific code we need for now. Next thing we need to do is build the Web Assembly itself.</p> <h2 id="wasm-pack"><a href="#wasm-pack" class="header-anchor">#</a> Wasm Pack</h2> <p>Now you can build a wgpu application with just wasm-bindgen, but I ran into some issues doing that. For one, you need to install wasm-bindgen on your computer as well as include it as a dependency. They version you install as a dependency <strong>needs</strong> to exactly match the version you installed, otherwise your build will fail.</p> <p>In order to get around this shortcoming, and to make the lives of everyone reading this easier, I opted to add <a href="https://rustwasm.github.io/docs/wasm-pack/" target="_blank" rel="noopener noreferrer">wasm-pack<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> to the mix. Wasm-pack handles installing the correct version of wasm-bindgen for you, and it supports building for different types of web targets as well: browser, NodeJS, and bundlers such as webpack.</p> <p>In order to use wasm-pack, first you need to <a href="https://rustwasm.github.io/wasm-pack/installer/" target="_blank" rel="noopener noreferrer">install it<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a>.</p> <p>Once you've done that, we can use it to build our crate. If you only have one crate in your project, you can just use <code>wasm-pack build</code>. If your using a workspace, you'll have to specify what crate you want to build. Imagine your crate is a directory called <code>game</code>, you would use:</p> <div class="language-bash extra-class"><pre class="language-bash"><code>wasm-pack build game
</code></pre></div><div class="note"><p>The <code>&quot;wasm-example&quot;</code> id is specific to my project (aka. this tutorial). You can substitute this for whatever id you're using in your HTML. Alternatively, you could add the canvas directly to the <code>&lt;body&gt;</code> as they do in the wgpu repo. This part is ultimately up to you.</p></div> <p>That's all the web-specific code we need for now. The next thing we need to do is build the Web Assembly itself.</p> <h2 id="wasm-pack"><a href="#wasm-pack" class="header-anchor">#</a> Wasm Pack</h2> <p>Now you can build a wgpu application with just wasm-bindgen, but I ran into some issues doing that. For one, you need to install wasm-bindgen on your computer as well as include it as a dependency. The version you install as a dependency <strong>needs</strong> to exactly match the version you installed, otherwise, your build will fail.</p> <p>To get around this shortcoming, and to make the lives of everyone reading this easier, I opted to add <a href="https://rustwasm.github.io/docs/wasm-pack/" target="_blank" rel="noopener noreferrer">wasm-pack<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> to the mix. Wasm-pack handles installing the correct version of wasm-bindgen for you, and it supports building for different types of web targets as well: browser, NodeJS, and bundlers such as webpack.</p> <p>To use wasm-pack, first, you need to <a href="https://rustwasm.github.io/wasm-pack/installer/" target="_blank" rel="noopener noreferrer">install it<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a>.</p> <p>Once you've done that, we can use it to build our crate. If you only have one crate in your project, you can just use <code>wasm-pack build</code>. If you're using a workspace, you'll have to specify what crate you want to build. Imagine your crate is a directory called <code>game</code>, you would use:</p> <div class="language-bash extra-class"><pre class="language-bash"><code>wasm-pack build game
</code></pre></div><p>Once wasm-pack is done building you'll have a <code>pkg</code> directory in the same directory as your crate. This has all the javascript code needed to run the WASM code. You'd then import the WASM module in javascript:</p> <div class="language-js extra-class"><pre class="language-js"><code><span class="token keyword">const</span> init <span class="token operator">=</span> <span class="token keyword">await</span> <span class="token keyword">import</span><span class="token punctuation">(</span><span class="token string">'./pkg/game.js'</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token function">init</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">then</span><span class="token punctuation">(</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">=&gt;</span> console<span class="token punctuation">.</span><span class="token function">log</span><span class="token punctuation">(</span><span class="token string">&quot;WASM Loaded&quot;</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
</code></pre></div><p>This site uses <a href="https://vuepress.vuejs.org/" target="_blank" rel="noopener noreferrer">Vuepress<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a>, so I load the WASM in a Vue component. How you handle your WASM will depend on what you want to do. If you want to check out how I'm doing things take a look at <a href="https://github.com/sotrh/learn-wgpu/blob/master/docs/.vuepress/components/WasmExample.vue" target="_blank" rel="noopener noreferrer">this<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a>.</p> <div class="note"><p>If you intend to use your WASM module in a plain HTML website, you'll need to tell wasm-pack to target the web:</p> <div class="language-bash extra-class"><pre class="language-bash"><code>wasm-pack build --target web
</code></pre></div><p>You'll then need run the WASM code in an ES6 Module:</p> <div class="language-html extra-class"><pre class="language-html"><code><span class="token doctype"><span class="token punctuation">&lt;!</span><span class="token doctype-tag">DOCTYPE</span> <span class="token name">html</span><span class="token punctuation">&gt;</span></span>
</code></pre></div><p>You'll then need to run the WASM code in an ES6 Module:</p> <div class="language-html extra-class"><pre class="language-html"><code><span class="token doctype"><span class="token punctuation">&lt;!</span><span class="token doctype-tag">DOCTYPE</span> <span class="token name">html</span><span class="token punctuation">&gt;</span></span>
<span class="token tag"><span class="token tag"><span class="token punctuation">&lt;</span>html</span> <span class="token attr-name">lang</span><span class="token attr-value"><span class="token punctuation attr-equals">=</span><span class="token punctuation">&quot;</span>en<span class="token punctuation">&quot;</span></span><span class="token punctuation">&gt;</span></span>
<span class="token tag"><span class="token tag"><span class="token punctuation">&lt;</span>head</span><span class="token punctuation">&gt;</span></span>
@ -133,21 +133,21 @@ I won't get into the specifics of wasm-bindgen, so if you need a primer (or just
<span class="token tag"><span class="token tag"><span class="token punctuation">&lt;/</span>head</span><span class="token punctuation">&gt;</span></span>
<span class="token tag"><span class="token tag"><span class="token punctuation">&lt;</span>body</span><span class="token punctuation">&gt;</span></span>
<span class="token tag"><span class="token tag"><span class="token punctuation">&lt;</span>script</span> <span class="token attr-name">type</span><span class="token attr-value"><span class="token punctuation attr-equals">=</span><span class="token punctuation">&quot;</span>module<span class="token punctuation">&quot;</span></span><span class="token punctuation">&gt;</span></span><span class="token script"><span class="token language-javascript">
<span class="token keyword">import</span> init <span class="token keyword">from</span> <span class="token string">&quot;./pkg/pong.js&quot;</span><span class="token punctuation">;</span>
<span class="token function">init</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">then</span><span class="token punctuation">(</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">=&gt;</span> <span class="token punctuation">{</span>
console<span class="token punctuation">.</span><span class="token function">log</span><span class="token punctuation">(</span><span class="token string">&quot;WASM Loaded&quot;</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
</span></span><span class="token tag"><span class="token tag"><span class="token punctuation">&lt;/</span>script</span><span class="token punctuation">&gt;</span></span>
<span class="token tag"><span class="token tag"><span class="token punctuation">&lt;</span>style</span><span class="token punctuation">&gt;</span></span><span class="token style"><span class="token language-css">
<span class="token selector">canvas</span> <span class="token punctuation">{</span>
<span class="token property">background-color</span><span class="token punctuation">:</span> black<span class="token punctuation">;</span>
<span class="token punctuation">}</span>
</span></span><span class="token tag"><span class="token tag"><span class="token punctuation">&lt;/</span>style</span><span class="token punctuation">&gt;</span></span>
<span class="token tag"><span class="token tag"><span class="token punctuation">&lt;</span>script</span> <span class="token attr-name">type</span><span class="token attr-value"><span class="token punctuation attr-equals">=</span><span class="token punctuation">&quot;</span>module<span class="token punctuation">&quot;</span></span><span class="token punctuation">&gt;</span></span><span class="token script"><span class="token language-javascript">
<span class="token keyword">import</span> init <span class="token keyword">from</span> <span class="token string">&quot;./pkg/pong.js&quot;</span><span class="token punctuation">;</span>
<span class="token function">init</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">then</span><span class="token punctuation">(</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">=&gt;</span> <span class="token punctuation">{</span>
console<span class="token punctuation">.</span><span class="token function">log</span><span class="token punctuation">(</span><span class="token string">&quot;WASM Loaded&quot;</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
</span></span><span class="token tag"><span class="token tag"><span class="token punctuation">&lt;/</span>script</span><span class="token punctuation">&gt;</span></span>
<span class="token tag"><span class="token tag"><span class="token punctuation">&lt;</span>style</span><span class="token punctuation">&gt;</span></span><span class="token style"><span class="token language-css">
<span class="token selector">canvas</span> <span class="token punctuation">{</span>
<span class="token property">background-color</span><span class="token punctuation">:</span> black<span class="token punctuation">;</span>
<span class="token punctuation">}</span>
</span></span><span class="token tag"><span class="token tag"><span class="token punctuation">&lt;/</span>style</span><span class="token punctuation">&gt;</span></span>
<span class="token tag"><span class="token tag"><span class="token punctuation">&lt;/</span>body</span><span class="token punctuation">&gt;</span></span>
<span class="token tag"><span class="token tag"><span class="token punctuation">&lt;/</span>html</span><span class="token punctuation">&gt;</span></span>
</code></pre></div></div> <p>Press the button below and you will see the code running!</p> <div id="wasm-example"><!----> <button>Try Tutorial1_window!</button></div> <div class="auto-github-link"><a href="https://github.com/sotrh/learn-wgpu/tree/master/code/beginner/tutorial1-window/" target="_blank" rel="noopener noreferrer">Check out the code!</a> <span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></div></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:41:58 PM</span></div></footer> <div class="page-nav"><p class="inner"><span class="prev">
</code></pre></div></div> <p>Press the button below and you will see the code running!</p> <div id="wasm-example"><!----> <button>Try Tutorial1_window!</button></div> <div class="auto-github-link"><a href="https://github.com/sotrh/learn-wgpu/tree/master/code/beginner/tutorial1-window/" target="_blank" rel="noopener noreferrer">Check out the code!</a> <span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></div></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:44:34 PM</span></div></footer> <div class="page-nav"><p class="inner"><span class="prev">
<a href="/learn-wgpu/" class="prev router-link-active">
Introduction
@ -156,6 +156,6 @@ I won't get into the specifics of wasm-bindgen, so if you need a primer (or just
</a>
</span></p></div> </main></div></div><div class="global-ui"><!----></div></div>
<script src="/learn-wgpu/assets/js/app.cbd0d45d.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/41.d3e6544b.js" defer></script><script src="/learn-wgpu/assets/js/22.ae08829d.js" defer></script><script src="/learn-wgpu/assets/js/26.9b876ccf.js" defer></script>
<script src="/learn-wgpu/assets/js/app.c87751e7.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/41.c80ee966.js" defer></script><script src="/learn-wgpu/assets/js/22.ae08829d.js" defer></script><script src="/learn-wgpu/assets/js/26.9b876ccf.js" defer></script>
</body>
</html>

@ -7,7 +7,7 @@
<meta name="generator" content="VuePress 1.9.7">
<meta name="description" content="">
<meta property="article:modified_time" content="2022-05-14T18:41:58.000Z">
<meta property="article:modified_time" content="2022-05-14T18:44:34.000Z">
<meta property="og:site_name" content="Learn Wgpu">
<meta property="og:title" content="The Surface">
<meta property="og:type" content="website">
@ -19,11 +19,11 @@
<meta name="twitter:data1" content="Benjamin Hansen">
<meta name="twitter:creator" content="https://twitter.com/sotrh760">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.cbd0d45d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/28.9fe49a03.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/22.ae08829d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/26.9b876ccf.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.576af98e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.e74dc361.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.69a9ea7e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.e284d6f3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.9d07c0eb.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.6982a462.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.32361a22.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.c212a80a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.b61fde25.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.25585faa.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.acd07b95.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.a51ea3c5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.41e9897b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.2d445b19.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.9b4d649e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.e0f57de3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.155fa5a2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.d3e6544b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.2bed1539.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.9a3ae8d9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.4d927984.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.bcb97198.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.c87751e7.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/28.064a85a6.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/22.ae08829d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/26.9b876ccf.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.e0b6e526.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.c8349761.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.2469c630.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.3972e66e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.d649043b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.5b5cf4ba.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.a63ee8c2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.fb933617.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.2e0b419b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.09885c9c.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.19c7b8dd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.9a017b74.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.695eaab4.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.1ae459af.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.bb1a0e6a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.34968282.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.58137882.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.c80ee966.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.740cfe90.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.5be8c6e7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.2e48975d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.ec58808b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="stylesheet" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css">
</head>
<body>
<div id="app" data-server-rendered="true"><div class="theme-container"><header class="navbar"><div class="inner"><div class="sidebar-button"><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" viewBox="0 0 448 512" class="icon"><path fill="currentColor" d="M436 124H12c-6.627 0-12-5.373-12-12V80c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12z"></path></svg></div> <a href="/learn-wgpu/" class="home-link router-link-active"><!----> <span class="site-name">Learn Wgpu</span></a> <div class="links"><!----> <div class="search-box"><input aria-label="Search" autocomplete="off" spellcheck="false" value=""> <!----></div></div></div></header> <div class="sidebar-mask"></div> <div class="docs-layout"><aside class="sidebar"><!----> <ul class="sidebar-links"><li><a href="/learn-wgpu/" aria-current="page" class="sidebar-link">Introduction</a></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading open"><span>Beginner</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/beginner/tutorial1-window/" class="sidebar-link">Dependencies and the window</a></li><li><a href="/learn-wgpu/beginner/tutorial2-surface/" aria-current="page" class="active sidebar-link">The Surface</a><ul class="sidebar-sub-headers"><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#first-some-housekeeping-state" class="sidebar-link">First, some housekeeping: State</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#state-new" class="sidebar-link">State::new()</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#resize" class="sidebar-link">resize()</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#input" class="sidebar-link">input()</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#update" class="sidebar-link">update()</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#render" class="sidebar-link">render()</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#wait-what-s-going-on-with-renderpassdescriptor" class="sidebar-link">Wait, what's going on with RenderPassDescriptor?</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#validation-errors" class="sidebar-link">Validation Errors?</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#challenge" class="sidebar-link">Challenge</a></li></ul></li><li><a href="/learn-wgpu/beginner/tutorial3-pipeline/" class="sidebar-link">The Pipeline</a></li><li><a href="/learn-wgpu/beginner/tutorial4-buffer/" class="sidebar-link">Buffers and Indices</a></li><li><a href="/learn-wgpu/beginner/tutorial5-textures/" class="sidebar-link">Textures and bind groups</a></li><li><a href="/learn-wgpu/beginner/tutorial6-uniforms/" class="sidebar-link">Uniform buffers and a 3d camera</a></li><li><a href="/learn-wgpu/beginner/tutorial7-instancing/" class="sidebar-link">Instancing</a></li><li><a href="/learn-wgpu/beginner/tutorial8-depth/" class="sidebar-link">The Depth Buffer</a></li><li><a href="/learn-wgpu/beginner/tutorial9-models/" class="sidebar-link">Model Loading</a></li></ul></section></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Intermediate</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/intermediate/tutorial10-lighting/" class="sidebar-link">Working with Lights</a></li><li><a href="/learn-wgpu/intermediate/tutorial11-normals/" class="sidebar-link">Normal Mapping</a></li><li><a href="/learn-wgpu/intermediate/tutorial12-camera/" class="sidebar-link">A Better Camera</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>Showcase</span> <span class="arrow right"></span></p> <!----></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>News</span> <span class="arrow right"></span></p> <!----></section></li></ul> </aside> <main class="page"> <div class="theme-default-content content__default"><h1 id="the-surface"><a href="#the-surface" class="header-anchor">#</a> The Surface</h1> <h2 id="first-some-housekeeping-state"><a href="#first-some-housekeeping-state" class="header-anchor">#</a> First, some housekeeping: State</h2> <p>For convenience, we're going to pack all the fields into a struct and create some methods on that.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// lib.rs</span>
<div id="app" data-server-rendered="true"><div class="theme-container"><header class="navbar"><div class="inner"><div class="sidebar-button"><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" viewBox="0 0 448 512" class="icon"><path fill="currentColor" d="M436 124H12c-6.627 0-12-5.373-12-12V80c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12z"></path></svg></div> <a href="/learn-wgpu/" class="home-link router-link-active"><!----> <span class="site-name">Learn Wgpu</span></a> <div class="links"><!----> <div class="search-box"><input aria-label="Search" autocomplete="off" spellcheck="false" value=""> <!----></div></div></div></header> <div class="sidebar-mask"></div> <div class="docs-layout"><aside class="sidebar"><!----> <ul class="sidebar-links"><li><a href="/learn-wgpu/" aria-current="page" class="sidebar-link">Introduction</a></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading open"><span>Beginner</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/beginner/tutorial1-window/" class="sidebar-link">Dependencies and the window</a></li><li><a href="/learn-wgpu/beginner/tutorial2-surface/" aria-current="page" class="active sidebar-link">The Surface</a><ul class="sidebar-sub-headers"><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#first-some-housekeeping-state" class="sidebar-link">First, some housekeeping: State</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#state-new" class="sidebar-link">State::new()</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#resize" class="sidebar-link">resize()</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#input" class="sidebar-link">input()</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#update" class="sidebar-link">update()</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#render" class="sidebar-link">render()</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#wait-what-s-going-on-with-renderpassdescriptor" class="sidebar-link">Wait, what's going on with RenderPassDescriptor?</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#validation-errors" class="sidebar-link">Validation Errors?</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial2-surface/#challenge" class="sidebar-link">Challenge</a></li></ul></li><li><a href="/learn-wgpu/beginner/tutorial3-pipeline/" class="sidebar-link">The Pipeline</a></li><li><a href="/learn-wgpu/beginner/tutorial4-buffer/" class="sidebar-link">Buffers and Indices</a></li><li><a href="/learn-wgpu/beginner/tutorial5-textures/" class="sidebar-link">Textures and bind groups</a></li><li><a href="/learn-wgpu/beginner/tutorial6-uniforms/" class="sidebar-link">Uniform buffers and a 3d camera</a></li><li><a href="/learn-wgpu/beginner/tutorial7-instancing/" class="sidebar-link">Instancing</a></li><li><a href="/learn-wgpu/beginner/tutorial8-depth/" class="sidebar-link">The Depth Buffer</a></li><li><a href="/learn-wgpu/beginner/tutorial9-models/" class="sidebar-link">Model Loading</a></li></ul></section></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Intermediate</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/intermediate/tutorial10-lighting/" class="sidebar-link">Working with Lights</a></li><li><a href="/learn-wgpu/intermediate/tutorial11-normals/" class="sidebar-link">Normal Mapping</a></li><li><a href="/learn-wgpu/intermediate/tutorial12-camera/" class="sidebar-link">A Better Camera</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>Showcase</span> <span class="arrow right"></span></p> <!----></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>News</span> <span class="arrow right"></span></p> <!----></section></li></ul> </aside> <main class="page"> <div class="theme-default-content content__default"><h1 id="the-surface"><a href="#the-surface" class="header-anchor">#</a> The Surface</h1> <h2 id="first-some-housekeeping-state"><a href="#first-some-housekeeping-state" class="header-anchor">#</a> First, some housekeeping: State</h2> <p>For convenience, we're going to pack all the fields into a struct and create some methods for it.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// lib.rs</span>
<span class="token keyword">use</span> <span class="token namespace">winit<span class="token punctuation">::</span>window<span class="token punctuation">::</span></span><span class="token class-name">Window</span><span class="token punctuation">;</span>
<span class="token keyword">struct</span> <span class="token type-definition class-name">State</span> <span class="token punctuation">{</span>
@ -56,7 +56,7 @@
<span class="token macro property">todo!</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
<span class="token punctuation">}</span>
<span class="token punctuation">}</span>
</code></pre></div><p>I'm glossing over <code>State</code>s fields, but they'll make more sense as I explain the code behind the methods.</p> <h2 id="state-new"><a href="#state-new" class="header-anchor">#</a> State::new()</h2> <p>The code for this is pretty straight forward, but let's break this down a bit.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">impl</span> <span class="token class-name">State</span> <span class="token punctuation">{</span>
</code></pre></div><p>I'm glossing over <code>State</code>s fields, but they'll make more sense as I explain the code behind these methods.</p> <h2 id="state-new"><a href="#state-new" class="header-anchor">#</a> State::new()</h2> <p>The code for this is pretty straightforward, but let's break it down a bit.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">impl</span> <span class="token class-name">State</span> <span class="token punctuation">{</span>
<span class="token comment">// ...</span>
<span class="token keyword">async</span> <span class="token keyword">fn</span> <span class="token function-definition function">new</span><span class="token punctuation">(</span>window<span class="token punctuation">:</span> <span class="token operator">&amp;</span><span class="token class-name">Window</span><span class="token punctuation">)</span> <span class="token punctuation">-&gt;</span> <span class="token keyword">Self</span> <span class="token punctuation">{</span>
<span class="token keyword">let</span> size <span class="token operator">=</span> window<span class="token punctuation">.</span><span class="token function">inner_size</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
@ -73,7 +73,7 @@
<span class="token punctuation">}</span><span class="token punctuation">,</span>
<span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token keyword">await</span><span class="token punctuation">.</span><span class="token function">unwrap</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
</code></pre></div><h3 id="instance-and-adapter"><a href="#instance-and-adapter" class="header-anchor">#</a> Instance and Adapter</h3> <p>The <code>instance</code> is the first thing you create when using wgpu. Its main purpose
is to create <code>Adapter</code>s and <code>Surface</code>s.</p> <p>The <code>adapter</code> is a handle to our actual graphics card. You can use this to get information about the graphics card such as its name and what backend the adapter uses. We use this to create our <code>Device</code> and <code>Queue</code> later. Let's discuss the fields of <code>RequestAdapterOptions</code>.</p> <ul><li><code>power_preference</code> has two variants: <code>LowPower</code>, and <code>HighPerformance</code>. This means will pick an adapter that favors battery life such as a integrated GPU when using <code>LowPower</code>. <code>HighPerformance</code> as will pick an adapter for more power hungry yet more performant GPU's such as your dedicated graphics card. WGPU will favor <code>LowPower</code> if there is no adapter for the <code>HighPerformance</code> option.</li> <li>The <code>compatible_surface</code> field tells wgpu to find an adapter that can present to the supplied surface.</li> <li>The <code>force_fallback_adapter</code> forces wgpu to pick an adapter that will work on all hardware. This usually means that the rendering backend will use a &quot;software&quot; system, instead of hardware such as a GPU.</li></ul> <div class="note"><p>The options I've passed to <code>request_adapter</code> aren't guaranteed to work for all devices, but will work for most of them. If wgpu can't find an adapter with the required permissions, <code>request_adapter</code> will return <code>None</code>. If you want to get all adapters for a particular backend you can use <code>enumerate_adapters</code>. This will give you an iterator that you can loop over to check if one of the adapters works for your needs.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">let</span> adapter <span class="token operator">=</span> instance
is to create <code>Adapter</code>s and <code>Surface</code>s.</p> <p>The <code>adapter</code> is a handle to our actual graphics card. You can use this to get information about the graphics card such as its name and what backend the adapter uses. We use this to create our <code>Device</code> and <code>Queue</code> later. Let's discuss the fields of <code>RequestAdapterOptions</code>.</p> <ul><li><code>power_preference</code> has two variants: <code>LowPower</code>, and <code>HighPerformance</code>. <code>LowPower</code> will pick an adapter that favors battery life, such as an integrated GPU. <code>HighPerformance</code> will pick an adapter for more power-hungry yet more performant GPU's such as a dedicated graphics card. WGPU will favor <code>LowPower</code> if there is no adapter for the <code>HighPerformance</code> option.</li> <li>The <code>compatible_surface</code> field tells wgpu to find an adapter that can present to the supplied surface.</li> <li>The <code>force_fallback_adapter</code> forces wgpu to pick an adapter that will work on all hardware. This usually means that the rendering backend will use a &quot;software&quot; system, instead of hardware such as a GPU.</li></ul> <div class="note"><p>The options I've passed to <code>request_adapter</code> aren't guaranteed to work for all devices, but will work for most of them. If wgpu can't find an adapter with the required permissions, <code>request_adapter</code> will return <code>None</code>. If you want to get all adapters for a particular backend you can use <code>enumerate_adapters</code>. This will give you an iterator that you can loop over to check if one of the adapters works for your needs.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">let</span> adapter <span class="token operator">=</span> instance
<span class="token punctuation">.</span><span class="token function">enumerate_adapters</span><span class="token punctuation">(</span><span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">Backends</span><span class="token punctuation">::</span><span class="token function">all</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span>
<span class="token punctuation">.</span><span class="token function">filter</span><span class="token punctuation">(</span><span class="token closure-params"><span class="token closure-punctuation punctuation">|</span>adapter<span class="token closure-punctuation punctuation">|</span></span> <span class="token punctuation">{</span>
<span class="token comment">// Check if this adapter supports our surface</span>
@ -81,7 +81,7 @@ is to create <code>Adapter</code>s and <code>Surface</code>s.</p> <p>The <code>a
<span class="token punctuation">}</span><span class="token punctuation">)</span>
<span class="token punctuation">.</span><span class="token function">next</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
<span class="token punctuation">.</span><span class="token function">unwrap</span><span class="token punctuation">(</span><span class="token punctuation">)</span>
</code></pre></div><p>Another thing to note is that <code>Adapter</code>s are locked to a specific backend. If you are on Windows and have 2 graphics cards you'll have at least 4 adapters available to use, 2 Vulkan and 2 DirectX.</p> <p>For more fields you can use to refine your search <a href="https://docs.rs/wgpu/latest/wgpu/struct.Adapter.html" target="_blank" rel="noopener noreferrer">check out the docs<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a>.</p></div> <h3 id="the-surface-2"><a href="#the-surface-2" class="header-anchor">#</a> The Surface</h3> <p>The <code>surface</code> is the part of the window that we draw to. We need it to draw directly to the screen. Our <code>window</code> needs to implement <a href="https://crates.io/crates/raw-window-handle" target="_blank" rel="noopener noreferrer">raw-window-handle<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a>'s <code>HasRawWindowHandle</code> trait to create a surface. Fortunately, winit's <code>Window</code> fits the bill. We also need it to request our <code>adapter</code>.</p> <h3 id="device-and-queue"><a href="#device-and-queue" class="header-anchor">#</a> Device and Queue</h3> <p>Let's use the <code>adapter</code> to create the device and queue.</p> <div class="language-rust extra-class"><pre class="language-rust"><code> <span class="token keyword">let</span> <span class="token punctuation">(</span>device<span class="token punctuation">,</span> queue<span class="token punctuation">)</span> <span class="token operator">=</span> adapter<span class="token punctuation">.</span><span class="token function">request_device</span><span class="token punctuation">(</span>
</code></pre></div><p>Another thing to note is that <code>Adapter</code>s are locked to a specific backend. If you are on Windows and have 2 graphics cards you'll have at least 4 adapters available to use, 2 Vulkan and 2 DirectX.</p> <p>For more fields you can use to refine your search, <a href="https://docs.rs/wgpu/latest/wgpu/struct.Adapter.html" target="_blank" rel="noopener noreferrer">check out the docs<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a>.</p></div> <h3 id="the-surface-2"><a href="#the-surface-2" class="header-anchor">#</a> The Surface</h3> <p>The <code>surface</code> is the part of the window that we draw to. We need it to draw directly to the screen. Our <code>window</code> needs to implement <a href="https://crates.io/crates/raw-window-handle" target="_blank" rel="noopener noreferrer">raw-window-handle<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a>'s <code>HasRawWindowHandle</code> trait to create a surface. Fortunately, winit's <code>Window</code> fits the bill. We also need it to request our <code>adapter</code>.</p> <h3 id="device-and-queue"><a href="#device-and-queue" class="header-anchor">#</a> Device and Queue</h3> <p>Let's use the <code>adapter</code> to create the device and queue.</p> <div class="language-rust extra-class"><pre class="language-rust"><code> <span class="token keyword">let</span> <span class="token punctuation">(</span>device<span class="token punctuation">,</span> queue<span class="token punctuation">)</span> <span class="token operator">=</span> adapter<span class="token punctuation">.</span><span class="token function">request_device</span><span class="token punctuation">(</span>
<span class="token operator">&amp;</span><span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">DeviceDescriptor</span> <span class="token punctuation">{</span>
features<span class="token punctuation">:</span> <span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">Features</span><span class="token punctuation">::</span><span class="token function">empty</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
<span class="token comment">// WebGL doesn't support all of wgpu's features, so if</span>
@ -95,7 +95,7 @@ is to create <code>Adapter</code>s and <code>Surface</code>s.</p> <p>The <code>a
<span class="token punctuation">}</span><span class="token punctuation">,</span>
<span class="token class-name">None</span><span class="token punctuation">,</span> <span class="token comment">// Trace path</span>
<span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token keyword">await</span><span class="token punctuation">.</span><span class="token function">unwrap</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
</code></pre></div><p>The <code>features</code> field on <code>DeviceDescriptor</code>, allows us to specify what extra features we want. For this simple example, I've decided not to use any extra features.</p> <div class="note"><p>The graphics card you have limits the features you can use. If you want to use certain features you may need to limit what devices you support, or provide workarounds.</p> <p>You can get a list of features supported by your device using <code>adapter.features()</code>, or <code>device.features()</code>.</p> <p>You can view a full list of features <a href="https://docs.rs/wgpu/latest/wgpu/struct.Features.html" target="_blank" rel="noopener noreferrer">here<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a>.</p></div> <p>The <code>limits</code> field describes the limit of certain types of resources that we can create. We'll use the defaults for this tutorial, so we can support most devices. You can view a list of limits <a href="https://docs.rs/wgpu/latest/wgpu/struct.Limits.html" target="_blank" rel="noopener noreferrer">here<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a>.</p> <div class="language-rust extra-class"><pre class="language-rust"><code> <span class="token keyword">let</span> config <span class="token operator">=</span> <span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">SurfaceConfiguration</span> <span class="token punctuation">{</span>
</code></pre></div><p>The <code>features</code> field on <code>DeviceDescriptor</code>, allows us to specify what extra features we want. For this simple example, I've decided not to use any extra features.</p> <div class="note"><p>The graphics card you have limits the features you can use. If you want to use certain features you may need to limit what devices you support or provide workarounds.</p> <p>You can get a list of features supported by your device using <code>adapter.features()</code>, or <code>device.features()</code>.</p> <p>You can view a full list of features <a href="https://docs.rs/wgpu/latest/wgpu/struct.Features.html" target="_blank" rel="noopener noreferrer">here<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a>.</p></div> <p>The <code>limits</code> field describes the limit of certain types of resources that we can create. We'll use the defaults for this tutorial, so we can support most devices. You can view a list of limits <a href="https://docs.rs/wgpu/latest/wgpu/struct.Limits.html" target="_blank" rel="noopener noreferrer">here<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a>.</p> <div class="language-rust extra-class"><pre class="language-rust"><code> <span class="token keyword">let</span> config <span class="token operator">=</span> <span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">SurfaceConfiguration</span> <span class="token punctuation">{</span>
usage<span class="token punctuation">:</span> <span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">TextureUsages</span><span class="token punctuation">::</span><span class="token constant">RENDER_ATTACHMENT</span><span class="token punctuation">,</span>
format<span class="token punctuation">:</span> surface<span class="token punctuation">.</span><span class="token function">get_preferred_format</span><span class="token punctuation">(</span><span class="token operator">&amp;</span>adapter<span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">unwrap</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
width<span class="token punctuation">:</span> size<span class="token punctuation">.</span>width<span class="token punctuation">,</span>
@ -103,9 +103,9 @@ is to create <code>Adapter</code>s and <code>Surface</code>s.</p> <p>The <code>a
present_mode<span class="token punctuation">:</span> <span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">PresentMode</span><span class="token punctuation">::</span><span class="token class-name">Fifo</span><span class="token punctuation">,</span>
<span class="token punctuation">}</span><span class="token punctuation">;</span>
surface<span class="token punctuation">.</span><span class="token function">configure</span><span class="token punctuation">(</span><span class="token operator">&amp;</span>device<span class="token punctuation">,</span> <span class="token operator">&amp;</span>config<span class="token punctuation">)</span><span class="token punctuation">;</span>
</code></pre></div><p>Here we are defining a config for our surface. This will define how the surface creates its underlying <code>SurfaceTexture</code>s. We will talk about <code>SurfaceTexture</code> when we get to the <code>render</code> function. For now lets talk about the config's fields.</p> <p>The <code>usage</code> field describes how <code>SurfaceTexture</code>s will be used. <code>RENDER_ATTACHMENT</code> specifies that the textures will be used to write to the screen (we'll talk about more <code>TextureUsages</code>s later).</p> <p>The <code>format</code> defines how <code>SurfaceTexture</code>s will be stored on the gpu. Different displays prefer different formats. We use <code>surface.get_preferred_format(&amp;adapter)</code> to figure out the best format to use based on the display you're using.</p> <p><code>width</code> and <code>height</code> are the width and the height in pixels of a <code>SurfaceTexture</code>. This should usually be the width and the height of the window.</p> <div class="warning">
</code></pre></div><p>Here we are defining a config for our surface. This will define how the surface creates its underlying <code>SurfaceTexture</code>s. We will talk about <code>SurfaceTexture</code> when we get to the <code>render</code> function. For now, let's talk about the config's fields.</p> <p>The <code>usage</code> field describes how <code>SurfaceTexture</code>s will be used. <code>RENDER_ATTACHMENT</code> specifies that the textures will be used to write to the screen (we'll talk about more <code>TextureUsages</code>s later).</p> <p>The <code>format</code> defines how <code>SurfaceTexture</code>s will be stored on the gpu. Different displays prefer different formats. We use <code>surface.get_preferred_format(&amp;adapter)</code> to figure out the best format to use based on the display you're using.</p> <p><code>width</code> and <code>height</code> are the width and the height in pixels of a <code>SurfaceTexture</code>. This should usually be the width and the height of the window.</p> <div class="warning">
Make sure that the width and height of the `SurfaceTexture` are not 0, as that can cause your app to crash.
</div> <p><code>present_mode</code> uses <code>wgpu::PresentMode</code> enum which determines how to sync the surface with the display. The option we picked, <code>FIFO</code>, will cap the display rate at the displays framerate. This is essentially VSync. This is also the most optimal mode on mobile. There are other options and you can see all of them <a href="https://docs.rs/wgpu/latest/wgpu/enum.PresentMode.html" target="_blank" rel="noopener noreferrer">in the docs<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a></p> <p>Now that we've configured our surface properly we can add these new fields at the end of the method.</p> <div class="language-rust extra-class"><pre class="language-rust"><code> <span class="token keyword">Self</span> <span class="token punctuation">{</span>
</div> <p><code>present_mode</code> uses <code>wgpu::PresentMode</code> enum which determines how to sync the surface with the display. The option we picked, <code>FIFO</code>, will cap the display rate at the display's framerate. This is essentially VSync. This is also the most optimal mode on mobile. There are other options and you can see all of them <a href="https://docs.rs/wgpu/latest/wgpu/enum.PresentMode.html" target="_blank" rel="noopener noreferrer">in the docs<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a></p> <p>Now that we've configured our surface properly we can add these new fields at the end of the method.</p> <div class="language-rust extra-class"><pre class="language-rust"><code> <span class="token keyword">Self</span> <span class="token punctuation">{</span>
surface<span class="token punctuation">,</span>
device<span class="token punctuation">,</span>
queue<span class="token punctuation">,</span>
@ -115,7 +115,7 @@ Make sure that the width and height of the `SurfaceTexture` are not 0, as that c
<span class="token punctuation">}</span>
<span class="token comment">// ...</span>
<span class="token punctuation">}</span>
</code></pre></div><p>Since our <code>State::new()</code> method is async we need to change run to be async as well so that we can await it.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">pub</span> <span class="token keyword">async</span> <span class="token keyword">fn</span> <span class="token function-definition function">run</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
</code></pre></div><p>Since our <code>State::new()</code> method is async we need to change <code>run()</code> to be async as well so that we can await it.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">pub</span> <span class="token keyword">async</span> <span class="token keyword">fn</span> <span class="token function-definition function">run</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
<span class="token comment">// Window setup...</span>
<span class="token keyword">let</span> <span class="token keyword">mut</span> state <span class="token operator">=</span> <span class="token class-name">State</span><span class="token punctuation">::</span><span class="token function">new</span><span class="token punctuation">(</span><span class="token operator">&amp;</span>window<span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token keyword">await</span><span class="token punctuation">;</span>
@ -128,7 +128,7 @@ Make sure that the width and height of the `SurfaceTexture` are not 0, as that c
</code></pre></div><p>We then use the <code>block_on</code> function provided by pollster to await our future:</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">fn</span> <span class="token function-definition function">main</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
<span class="token namespace">pollster<span class="token punctuation">::</span></span><span class="token function">block_on</span><span class="token punctuation">(</span><span class="token function">run</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span>
</code></pre></div><div class="warning"><p>Don't use <code>block_on</code> inside of an async function if you plan to support WASM. Futures have to be run using the browsers executor. If you try to bring your own you code will crash when you encounter a future that doesn't execute immediately.</p></div> <p>If we try to build WASM now it will fail because <code>wasm-bindgen</code> doesn't support using async functions as <code>start</code> methods. You could switch to calling <code>run</code> manually in javascript, but for simplicity we'll add the <a href="https://docs.rs/wasm-bindgen-futures" target="_blank" rel="noopener noreferrer">wasm-bindgen-futures<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> crate to our WASM dependencies as that doesn't require us to change any code. Your dependecies should look something like this:</p> <div class="language-toml extra-class"><pre class="language-toml"><code><span class="token punctuation">[</span><span class="token table class-name">dependencies</span><span class="token punctuation">]</span>
</code></pre></div><div class="warning"><p>Don't use <code>block_on</code> inside of an async function if you plan to support WASM. Futures have to be run using the browser's executor. If you try to bring your own your code will crash when you encounter a future that doesn't execute immediately.</p></div> <p>If we try to build WASM now it will fail because <code>wasm-bindgen</code> doesn't support using async functions as <code>start</code> methods. You could switch to calling <code>run</code> manually in javascript, but for simplicity, we'll add the <a href="https://docs.rs/wasm-bindgen-futures" target="_blank" rel="noopener noreferrer">wasm-bindgen-futures<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> crate to our WASM dependencies as that doesn't require us to change any code. Your dependencies should look something like this:</p> <div class="language-toml extra-class"><pre class="language-toml"><code><span class="token punctuation">[</span><span class="token table class-name">dependencies</span><span class="token punctuation">]</span>
<span class="token key property">cfg-if</span> <span class="token punctuation">=</span> <span class="token string">&quot;1&quot;</span>
<span class="token key property">winit</span> <span class="token punctuation">=</span> <span class="token string">&quot;0.26&quot;</span>
<span class="token key property">env_logger</span> <span class="token punctuation">=</span> <span class="token string">&quot;0.9&quot;</span>
@ -147,7 +147,7 @@ Make sure that the width and height of the `SurfaceTexture` are not 0, as that c
<span class="token string">&quot;Window&quot;</span><span class="token punctuation">,</span>
<span class="token string">&quot;Element&quot;</span><span class="token punctuation">,</span>
<span class="token punctuation">]</span><span class="token punctuation">}</span>
</code></pre></div><h2 id="resize"><a href="#resize" class="header-anchor">#</a> resize()</h2> <p>If we want to support resizing in our application, we're going to need to reconfigure the <code>surface</code> everytime the window's size changes. That's the reason we stored the physical <code>size</code> and the <code>config</code> used to configure the <code>surface</code>. With all of these, the resize method is very simple.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// impl State</span>
</code></pre></div><h2 id="resize"><a href="#resize" class="header-anchor">#</a> resize()</h2> <p>If we want to support resizing in our application, we're going to need to reconfigure the <code>surface</code> every time the window's size changes. That's the reason we stored the physical <code>size</code> and the <code>config</code> used to configure the <code>surface</code>. With all of these, the resize method is very simple.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// impl State</span>
<span class="token keyword">pub</span> <span class="token keyword">fn</span> <span class="token function-definition function">resize</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token keyword">mut</span> <span class="token keyword">self</span><span class="token punctuation">,</span> new_size<span class="token punctuation">:</span> <span class="token namespace">winit<span class="token punctuation">::</span>dpi<span class="token punctuation">::</span></span><span class="token class-name">PhysicalSize</span><span class="token operator">&lt;</span><span class="token keyword">u32</span><span class="token operator">&gt;</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
<span class="token keyword">if</span> new_size<span class="token punctuation">.</span>width <span class="token operator">&gt;</span> <span class="token number">0</span> <span class="token operator">&amp;&amp;</span> new_size<span class="token punctuation">.</span>height <span class="token operator">&gt;</span> <span class="token number">0</span> <span class="token punctuation">{</span>
<span class="token keyword">self</span><span class="token punctuation">.</span>size <span class="token operator">=</span> new_size<span class="token punctuation">;</span>
@ -156,7 +156,7 @@ Make sure that the width and height of the `SurfaceTexture` are not 0, as that c
<span class="token keyword">self</span><span class="token punctuation">.</span>surface<span class="token punctuation">.</span><span class="token function">configure</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token keyword">self</span><span class="token punctuation">.</span>device<span class="token punctuation">,</span> <span class="token operator">&amp;</span><span class="token keyword">self</span><span class="token punctuation">.</span>config<span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span>
<span class="token punctuation">}</span>
</code></pre></div><p>There's nothing really different here from the initial <code>surface</code> configuration, so I won't get into it.</p> <p>We call this method in <code>main()</code> in the event loop for the following events.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">match</span> event <span class="token punctuation">{</span>
</code></pre></div><p>There's nothing different here from the initial <code>surface</code> configuration, so I won't get into it.</p> <p>We call this method in <code>run()</code> in the event loop for the following events.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">match</span> event <span class="token punctuation">{</span>
<span class="token comment">// ...</span>
<span class="token punctuation">}</span> <span class="token keyword">if</span> window_id <span class="token operator">==</span> window<span class="token punctuation">.</span><span class="token function">id</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">=&gt;</span> <span class="token keyword">if</span> <span class="token operator">!</span>state<span class="token punctuation">.</span><span class="token function">input</span><span class="token punctuation">(</span>event<span class="token punctuation">)</span> <span class="token punctuation">{</span>
@ -176,7 +176,7 @@ Make sure that the width and height of the `SurfaceTexture` are not 0, as that c
<span class="token keyword">fn</span> <span class="token function-definition function">input</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token keyword">mut</span> <span class="token keyword">self</span><span class="token punctuation">,</span> event<span class="token punctuation">:</span> <span class="token operator">&amp;</span><span class="token class-name">WindowEvent</span><span class="token punctuation">)</span> <span class="token punctuation">-&gt;</span> <span class="token keyword">bool</span> <span class="token punctuation">{</span>
<span class="token boolean">false</span>
<span class="token punctuation">}</span>
</code></pre></div><p>We need to do a little more work in the event loop. We want <code>State</code> to have priority over <code>main()</code>. Doing that (and previous changes) should have your loop looking like this.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// main()</span>
</code></pre></div><p>We need to do a little more work in the event loop. We want <code>State</code> to have priority over <code>run()</code>. Doing that (and previous changes) should have your loop looking like this.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// run()</span>
event_loop<span class="token punctuation">.</span><span class="token function">run</span><span class="token punctuation">(</span><span class="token keyword">move</span> <span class="token closure-params"><span class="token closure-punctuation punctuation">|</span>event<span class="token punctuation">,</span> _<span class="token punctuation">,</span> control_flow<span class="token closure-punctuation punctuation">|</span></span> <span class="token punctuation">{</span>
<span class="token keyword">match</span> event <span class="token punctuation">{</span>
<span class="token class-name">Event</span><span class="token punctuation">::</span><span class="token class-name">WindowEvent</span> <span class="token punctuation">{</span>
@ -209,7 +209,7 @@ event_loop<span class="token punctuation">.</span><span class="token function">r
</code></pre></div><h2 id="update"><a href="#update" class="header-anchor">#</a> update()</h2> <p>We don't have anything to update yet, so leave the method empty.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">fn</span> <span class="token function-definition function">update</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token keyword">mut</span> <span class="token keyword">self</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
<span class="token comment">// remove `todo!()`</span>
<span class="token punctuation">}</span>
</code></pre></div><p>We'll add some code here later on to move around objects.</p> <h2 id="render"><a href="#render" class="header-anchor">#</a> render()</h2> <p>Here's where the magic happens. First we need to get a frame to render to.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// impl State</span>
</code></pre></div><p>We'll add some code here later on to move around objects.</p> <h2 id="render"><a href="#render" class="header-anchor">#</a> render()</h2> <p>Here's where the magic happens. First, we need to get a frame to render to.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// impl State</span>
<span class="token keyword">fn</span> <span class="token function-definition function">render</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token keyword">mut</span> <span class="token keyword">self</span><span class="token punctuation">)</span> <span class="token punctuation">-&gt;</span> <span class="token class-name">Result</span><span class="token operator">&lt;</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">,</span> <span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">SurfaceError</span><span class="token operator">&gt;</span> <span class="token punctuation">{</span>
<span class="token keyword">let</span> output <span class="token operator">=</span> <span class="token keyword">self</span><span class="token punctuation">.</span>surface<span class="token punctuation">.</span><span class="token function">get_current_texture</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token operator">?</span><span class="token punctuation">;</span>
@ -217,7 +217,7 @@ event_loop<span class="token punctuation">.</span><span class="token function">r
</code></pre></div><p>This line creates a <code>TextureView</code> with default settings. We need to do this because we want to control how the render code interacts with the texture.</p> <p>We also need to create a <code>CommandEncoder</code> to create the actual commands to send to the gpu. Most modern graphics frameworks expect commands to be stored in a command buffer before being sent to the gpu. The <code>encoder</code> builds a command buffer that we can then send to the gpu.</p> <div class="language-rust extra-class"><pre class="language-rust"><code> <span class="token keyword">let</span> <span class="token keyword">mut</span> encoder <span class="token operator">=</span> <span class="token keyword">self</span><span class="token punctuation">.</span>device<span class="token punctuation">.</span><span class="token function">create_command_encoder</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">CommandEncoderDescriptor</span> <span class="token punctuation">{</span>
label<span class="token punctuation">:</span> <span class="token class-name">Some</span><span class="token punctuation">(</span><span class="token string">&quot;Render Encoder&quot;</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
<span class="token punctuation">}</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
</code></pre></div><p>Now we can actually get to clearing the screen (long time coming). We need to use the <code>encoder</code> to create a <code>RenderPass</code>. The <code>RenderPass</code> has all the methods for the actual drawing. The code for creating a <code>RenderPass</code> is a bit nested, so I'll copy it all here before talking about its pieces.</p> <div class="language-rust extra-class"><pre class="language-rust"><code> <span class="token punctuation">{</span>
</code></pre></div><p>Now we can get to clearing the screen (long time coming). We need to use the <code>encoder</code> to create a <code>RenderPass</code>. The <code>RenderPass</code> has all the methods for the actual drawing. The code for creating a <code>RenderPass</code> is a bit nested, so I'll copy it all here before talking about its pieces.</p> <div class="language-rust extra-class"><pre class="language-rust"><code> <span class="token punctuation">{</span>
<span class="token keyword">let</span> _render_pass <span class="token operator">=</span> encoder<span class="token punctuation">.</span><span class="token function">begin_render_pass</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">RenderPassDescriptor</span> <span class="token punctuation">{</span>
label<span class="token punctuation">:</span> <span class="token class-name">Some</span><span class="token punctuation">(</span><span class="token string">&quot;Render Pass&quot;</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
color_attachments<span class="token punctuation">:</span> <span class="token operator">&amp;</span><span class="token punctuation">[</span><span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">RenderPassColorAttachment</span> <span class="token punctuation">{</span>
@ -243,7 +243,7 @@ event_loop<span class="token punctuation">.</span><span class="token function">r
<span class="token class-name">Ok</span><span class="token punctuation">(</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span>
<span class="token punctuation">}</span>
</code></pre></div><p>First things first, let's talk about the extra block (<code>{}</code>) around <code>encoder.begin_render_pass(...)</code>. <code>begin_render_pass()</code> borrows <code>encoder</code> mutably (aka <code>&amp;mut self</code>). We can't call <code>encoder.finish()</code> until we release that mutable borrow. The block tells rust to drop any variables within it when the code leaves that scope thus releasing the mutable borrow on <code>encoder</code> and allowing us to <code>finish()</code> it. If you don't like the <code>{}</code>, you can also use <code>drop(render_pass)</code> to achieve the same effect.</p> <p>We can get the same results by removing the <code>{}</code>, and the <code>let _render_pass =</code> line, but we need access to the <code>_render_pass</code> in the next tutorial, so we'll leave it as is.</p> <p>The last lines of the code tell <code>wgpu</code> to finish the command buffer, and to submit it to the gpu's render queue.</p> <p>We need to update the event loop again to call this method. We'll also call update before it too.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// main()</span>
</code></pre></div><p>First things first, let's talk about the extra block (<code>{}</code>) around <code>encoder.begin_render_pass(...)</code>. <code>begin_render_pass()</code> borrows <code>encoder</code> mutably (aka <code>&amp;mut self</code>). We can't call <code>encoder.finish()</code> until we release that mutable borrow. The block tells rust to drop any variables within it when the code leaves that scope thus releasing the mutable borrow on <code>encoder</code> and allowing us to <code>finish()</code> it. If you don't like the <code>{}</code>, you can also use <code>drop(render_pass)</code> to achieve the same effect.</p> <p>We can get the same results by removing the <code>{}</code>, and the <code>let _render_pass =</code> line, but we need access to the <code>_render_pass</code> in the next tutorial, so we'll leave it as is.</p> <p>The last lines of the code tell <code>wgpu</code> to finish the command buffer, and to submit it to the gpu's render queue.</p> <p>We need to update the event loop again to call this method. We'll also call <code>update()</code> before it too.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// run()</span>
event_loop<span class="token punctuation">.</span><span class="token function">run</span><span class="token punctuation">(</span><span class="token keyword">move</span> <span class="token closure-params"><span class="token closure-punctuation punctuation">|</span>event<span class="token punctuation">,</span> _<span class="token punctuation">,</span> control_flow<span class="token closure-punctuation punctuation">|</span></span> <span class="token punctuation">{</span>
<span class="token keyword">match</span> event <span class="token punctuation">{</span>
<span class="token comment">// ...</span>
@ -287,7 +287,7 @@ event_loop<span class="token punctuation">.</span><span class="token function">r
store<span class="token punctuation">:</span> <span class="token boolean">true</span><span class="token punctuation">,</span>
<span class="token punctuation">}</span><span class="token punctuation">,</span>
<span class="token punctuation">}</span>
</code></pre></div><p>The <code>RenderPassColorAttachment</code> has the <code>view</code> field which informs <code>wgpu</code> what texture to save the colors to. In this case we specify <code>view</code> that we created using <code>surface.get_current_texture()</code>. This means that any colors we draw to this attachment will get drawn to the screen.</p> <p>The <code>resolve_target</code> is the texture that will receive the resolved output. This will be the same as <code>view</code> unless multisampling is enabled. We don't need to specify this, so we leave it as <code>None</code>.</p> <p>The <code>ops</code> field takes a <code>wpgu::Operations</code> object. This tells wgpu what to do with the colors on the screen (specified by <code>view</code>). The <code>load</code> field tells wgpu how to handle colors stored from the previous frame. Currently, we are clearing the screen with a bluish color. The <code>store</code> field tells wgpu whether we want to store the rendered results to the <code>Texture</code> behind our <code>TextureView</code> (in this case it's the <code>SurfaceTexture</code>). We use <code>true</code> as we do want to store our render results.</p> <div class="note"><p>It's not uncommon to not clear the screen if the screen is going to be completely covered up with objects. If your scene doesn't cover the entire screen however you can end up with something like this.</p> <p><img src="/learn-wgpu/assets/img/no-clear.753f913f.png" alt="./no-clear.png"></p></div> <h2 id="validation-errors"><a href="#validation-errors" class="header-anchor">#</a> Validation Errors?</h2> <p>If wgpu is using Vulkan on your machine, you may run into validation errors if you are running an older version of the Vulkan SDK. You should be using at least version <code>1.2.182</code> as older versions can give out some false positives. If errors persist, you may have encountered a bug in wgpu. You can post an issue at <a href="https://github.com/gfx-rs/wgpu" target="_blank" rel="noopener noreferrer">https://github.com/gfx-rs/wgpu<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a></p> <h2 id="challenge"><a href="#challenge" class="header-anchor">#</a> Challenge</h2> <p>Modify the <code>input()</code> method to capture mouse events, and update the clear color using that. <em>Hint: you'll probably need to use <code>WindowEvent::CursorMoved</code></em>.</p> <div id="wasm-example"><!----> <button>Try Tutorial2_surface!</button></div> <div class="auto-github-link"><a href="https://github.com/sotrh/learn-wgpu/tree/master/code/beginner/tutorial2-surface/" target="_blank" rel="noopener noreferrer">Check out the code!</a> <span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></div></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:41:58 PM</span></div></footer> <div class="page-nav"><p class="inner"><span class="prev">
</code></pre></div><p>The <code>RenderPassColorAttachment</code> has the <code>view</code> field which informs <code>wgpu</code> what texture to save the colors to. In this case we specify the <code>view</code> that we created using <code>surface.get_current_texture()</code>. This means that any colors we draw to this attachment will get drawn to the screen.</p> <p>The <code>resolve_target</code> is the texture that will receive the resolved output. This will be the same as <code>view</code> unless multisampling is enabled. We don't need to specify this, so we leave it as <code>None</code>.</p> <p>The <code>ops</code> field takes a <code>wpgu::Operations</code> object. This tells wgpu what to do with the colors on the screen (specified by <code>view</code>). The <code>load</code> field tells wgpu how to handle colors stored from the previous frame. Currently, we are clearing the screen with a bluish color. The <code>store</code> field tells wgpu whether we want to store the rendered results to the <code>Texture</code> behind our <code>TextureView</code> (in this case it's the <code>SurfaceTexture</code>). We use <code>true</code> as we do want to store our render results.</p> <div class="note"><p>It's not uncommon to not clear the screen if the screen is going to be completely covered up with objects. If your scene doesn't cover the entire screen however you can end up with something like this.</p> <p><img src="/learn-wgpu/assets/img/no-clear.753f913f.png" alt="./no-clear.png"></p></div> <h2 id="validation-errors"><a href="#validation-errors" class="header-anchor">#</a> Validation Errors?</h2> <p>If wgpu is using Vulkan on your machine, you may run into validation errors if you are running an older version of the Vulkan SDK. You should be using at least version <code>1.2.182</code> as older versions can give out some false positives. If errors persist, you may have encountered a bug in wgpu. You can post an issue at <a href="https://github.com/gfx-rs/wgpu" target="_blank" rel="noopener noreferrer">https://github.com/gfx-rs/wgpu<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a></p> <h2 id="challenge"><a href="#challenge" class="header-anchor">#</a> Challenge</h2> <p>Modify the <code>input()</code> method to capture mouse events, and update the clear color using that. <em>Hint: you'll probably need to use <code>WindowEvent::CursorMoved</code></em>.</p> <div id="wasm-example"><!----> <button>Try Tutorial2_surface!</button></div> <div class="auto-github-link"><a href="https://github.com/sotrh/learn-wgpu/tree/master/code/beginner/tutorial2-surface/" target="_blank" rel="noopener noreferrer">Check out the code!</a> <span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></div></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:44:34 PM</span></div></footer> <div class="page-nav"><p class="inner"><span class="prev">
<a href="/learn-wgpu/beginner/tutorial1-window/" class="prev">
Dependencies and the window
@ -296,6 +296,6 @@ event_loop<span class="token punctuation">.</span><span class="token function">r
</a>
</span></p></div> </main></div></div><div class="global-ui"><!----></div></div>
<script src="/learn-wgpu/assets/js/app.cbd0d45d.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/28.9fe49a03.js" defer></script><script src="/learn-wgpu/assets/js/22.ae08829d.js" defer></script><script src="/learn-wgpu/assets/js/26.9b876ccf.js" defer></script>
<script src="/learn-wgpu/assets/js/app.c87751e7.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/28.064a85a6.js" defer></script><script src="/learn-wgpu/assets/js/22.ae08829d.js" defer></script><script src="/learn-wgpu/assets/js/26.9b876ccf.js" defer></script>
</body>
</html>

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

@ -7,7 +7,7 @@
<meta name="generator" content="VuePress 1.9.7">
<meta name="description" content="">
<meta property="article:modified_time" content="2022-05-14T18:41:58.000Z">
<meta property="article:modified_time" content="2022-05-14T18:44:34.000Z">
<meta property="og:site_name" content="Learn Wgpu">
<meta property="og:title" content="Model Loading">
<meta property="og:type" content="website">
@ -19,11 +19,11 @@
<meta name="twitter:data1" content="Benjamin Hansen">
<meta name="twitter:creator" content="https://twitter.com/sotrh760">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.cbd0d45d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/25.32361a22.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/22.ae08829d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/26.9b876ccf.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.576af98e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.e74dc361.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.69a9ea7e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.e284d6f3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.9d07c0eb.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.6982a462.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.9fe49a03.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.c212a80a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.b61fde25.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.25585faa.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.acd07b95.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.a51ea3c5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.41e9897b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.2d445b19.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.9b4d649e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.e0f57de3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.155fa5a2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.d3e6544b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.2bed1539.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.9a3ae8d9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.4d927984.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.bcb97198.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.c87751e7.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/25.a63ee8c2.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/22.ae08829d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/26.9b876ccf.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.e0b6e526.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.c8349761.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.2469c630.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.3972e66e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.d649043b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.5b5cf4ba.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.064a85a6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.fb933617.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.2e0b419b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.09885c9c.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.19c7b8dd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.9a017b74.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.695eaab4.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.1ae459af.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.bb1a0e6a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.34968282.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.58137882.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.c80ee966.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.740cfe90.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.5be8c6e7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.2e48975d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.ec58808b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="stylesheet" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css">
</head>
<body>
<div id="app" data-server-rendered="true"><div class="theme-container"><header class="navbar"><div class="inner"><div class="sidebar-button"><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" viewBox="0 0 448 512" class="icon"><path fill="currentColor" d="M436 124H12c-6.627 0-12-5.373-12-12V80c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12z"></path></svg></div> <a href="/learn-wgpu/" class="home-link router-link-active"><!----> <span class="site-name">Learn Wgpu</span></a> <div class="links"><!----> <div class="search-box"><input aria-label="Search" autocomplete="off" spellcheck="false" value=""> <!----></div></div></div></header> <div class="sidebar-mask"></div> <div class="docs-layout"><aside class="sidebar"><!----> <ul class="sidebar-links"><li><a href="/learn-wgpu/" aria-current="page" class="sidebar-link">Introduction</a></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading open"><span>Beginner</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/beginner/tutorial1-window/" class="sidebar-link">Dependencies and the window</a></li><li><a href="/learn-wgpu/beginner/tutorial2-surface/" class="sidebar-link">The Surface</a></li><li><a href="/learn-wgpu/beginner/tutorial3-pipeline/" class="sidebar-link">The Pipeline</a></li><li><a href="/learn-wgpu/beginner/tutorial4-buffer/" class="sidebar-link">Buffers and Indices</a></li><li><a href="/learn-wgpu/beginner/tutorial5-textures/" class="sidebar-link">Textures and bind groups</a></li><li><a href="/learn-wgpu/beginner/tutorial6-uniforms/" class="sidebar-link">Uniform buffers and a 3d camera</a></li><li><a href="/learn-wgpu/beginner/tutorial7-instancing/" class="sidebar-link">Instancing</a></li><li><a href="/learn-wgpu/beginner/tutorial8-depth/" class="sidebar-link">The Depth Buffer</a></li><li><a href="/learn-wgpu/beginner/tutorial9-models/" aria-current="page" class="active sidebar-link">Model Loading</a><ul class="sidebar-sub-headers"><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial9-models/#accessing-files-in-the-res-folder" class="sidebar-link">Accessing files in the res folder</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial9-models/#accessing-files-from-wasm" class="sidebar-link">Accessing files from WASM</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial9-models/#loading-models-with-tobj" class="sidebar-link">Loading models with TOBJ</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial9-models/#rendering-a-mesh" class="sidebar-link">Rendering a mesh</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial9-models/#using-the-correct-textures" class="sidebar-link">Using the correct textures</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial9-models/#rendering-the-entire-model" class="sidebar-link">Rendering the entire model</a></li></ul></li></ul></section></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Intermediate</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/intermediate/tutorial10-lighting/" class="sidebar-link">Working with Lights</a></li><li><a href="/learn-wgpu/intermediate/tutorial11-normals/" class="sidebar-link">Normal Mapping</a></li><li><a href="/learn-wgpu/intermediate/tutorial12-camera/" class="sidebar-link">A Better Camera</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>Showcase</span> <span class="arrow right"></span></p> <!----></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>News</span> <span class="arrow right"></span></p> <!----></section></li></ul> </aside> <main class="page"> <div class="theme-default-content content__default"><h1 id="model-loading"><a href="#model-loading" class="header-anchor">#</a> Model Loading</h1> <p>Up to this point we've been creating our models manually. While this is an acceptable way to do this, but it's really slow if we want to include complex models with lots of polygons. Because of this, we're going modify our code to leverage the obj model format so that we can create a model in a software such as blender and display it in our code.</p> <p>Our <code>main.rs</code> file is getting pretty cluttered, let's create a <code>model.rs</code> file that we can put our model loading code into.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// model.rs</span>
<div id="app" data-server-rendered="true"><div class="theme-container"><header class="navbar"><div class="inner"><div class="sidebar-button"><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" viewBox="0 0 448 512" class="icon"><path fill="currentColor" d="M436 124H12c-6.627 0-12-5.373-12-12V80c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12z"></path></svg></div> <a href="/learn-wgpu/" class="home-link router-link-active"><!----> <span class="site-name">Learn Wgpu</span></a> <div class="links"><!----> <div class="search-box"><input aria-label="Search" autocomplete="off" spellcheck="false" value=""> <!----></div></div></div></header> <div class="sidebar-mask"></div> <div class="docs-layout"><aside class="sidebar"><!----> <ul class="sidebar-links"><li><a href="/learn-wgpu/" aria-current="page" class="sidebar-link">Introduction</a></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading open"><span>Beginner</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/beginner/tutorial1-window/" class="sidebar-link">Dependencies and the window</a></li><li><a href="/learn-wgpu/beginner/tutorial2-surface/" class="sidebar-link">The Surface</a></li><li><a href="/learn-wgpu/beginner/tutorial3-pipeline/" class="sidebar-link">The Pipeline</a></li><li><a href="/learn-wgpu/beginner/tutorial4-buffer/" class="sidebar-link">Buffers and Indices</a></li><li><a href="/learn-wgpu/beginner/tutorial5-textures/" class="sidebar-link">Textures and bind groups</a></li><li><a href="/learn-wgpu/beginner/tutorial6-uniforms/" class="sidebar-link">Uniform buffers and a 3d camera</a></li><li><a href="/learn-wgpu/beginner/tutorial7-instancing/" class="sidebar-link">Instancing</a></li><li><a href="/learn-wgpu/beginner/tutorial8-depth/" class="sidebar-link">The Depth Buffer</a></li><li><a href="/learn-wgpu/beginner/tutorial9-models/" aria-current="page" class="active sidebar-link">Model Loading</a><ul class="sidebar-sub-headers"><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial9-models/#accessing-files-in-the-res-folder" class="sidebar-link">Accessing files in the res folder</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial9-models/#accessing-files-from-wasm" class="sidebar-link">Accessing files from WASM</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial9-models/#loading-models-with-tobj" class="sidebar-link">Loading models with TOBJ</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial9-models/#rendering-a-mesh" class="sidebar-link">Rendering a mesh</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial9-models/#using-the-correct-textures" class="sidebar-link">Using the correct textures</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/beginner/tutorial9-models/#rendering-the-entire-model" class="sidebar-link">Rendering the entire model</a></li></ul></li></ul></section></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Intermediate</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/intermediate/tutorial10-lighting/" class="sidebar-link">Working with Lights</a></li><li><a href="/learn-wgpu/intermediate/tutorial11-normals/" class="sidebar-link">Normal Mapping</a></li><li><a href="/learn-wgpu/intermediate/tutorial12-camera/" class="sidebar-link">A Better Camera</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>Showcase</span> <span class="arrow right"></span></p> <!----></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>News</span> <span class="arrow right"></span></p> <!----></section></li></ul> </aside> <main class="page"> <div class="theme-default-content content__default"><h1 id="model-loading"><a href="#model-loading" class="header-anchor">#</a> Model Loading</h1> <p>Up to this point we've been creating our models manually. While this is an acceptable way to do this, it's really slow if we want to include complex models with lots of polygons. Because of this, we're going to modify our code to leverage the <code>.obj</code> model format so that we can create a model in software such as blender and display it in our code.</p> <p>Our <code>lib.rs</code> file is getting pretty cluttered, let's create a <code>model.rs</code> file that we can put our model loading code into.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// model.rs</span>
<span class="token keyword">pub</span> <span class="token keyword">trait</span> <span class="token type-definition class-name">Vertex</span> <span class="token punctuation">{</span>
<span class="token keyword">fn</span> <span class="token function-definition function">desc</span><span class="token operator">&lt;</span><span class="token lifetime-annotation symbol">'a</span><span class="token operator">&gt;</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">-&gt;</span> <span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">VertexBufferLayout</span><span class="token operator">&lt;</span><span class="token lifetime-annotation symbol">'a</span><span class="token operator">&gt;</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span>
@ -41,7 +41,7 @@
<span class="token macro property">todo!</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span>
<span class="token punctuation">}</span>
</code></pre></div><p>You'll notice a couple of things here. In <code>main.rs</code> we had <code>Vertex</code> as a struct, here we're using a trait. We could have multiple vertex types (model, UI, instance data, etc.). Making <code>Vertex</code> a trait will allow us to abstract our the <code>VertexBufferLayout</code> creation code to make creating <code>RenderPipeline</code>s simpler.</p> <p>Another thing to mention is the <code>normal</code> field in <code>ModelVertex</code>. We won't use this until we talk about lighting, but will add it to the struct for now.</p> <p>Let's define our <code>VertexBufferLayout</code>.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">impl</span> <span class="token class-name">Vertex</span> <span class="token keyword">for</span> <span class="token class-name">ModelVertex</span> <span class="token punctuation">{</span>
</code></pre></div><p>You'll notice a couple of things here. In <code>lib.rs</code> we had <code>Vertex</code> as a struct, here we're using a trait. We could have multiple vertex types (model, UI, instance data, etc.). Making <code>Vertex</code> a trait will allow us to abstract out the <code>VertexBufferLayout</code> creation code to make creating <code>RenderPipeline</code>s simpler.</p> <p>Another thing to mention is the <code>normal</code> field in <code>ModelVertex</code>. We won't use this until we talk about lighting, but will add it to the struct for now.</p> <p>Let's define our <code>VertexBufferLayout</code>.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">impl</span> <span class="token class-name">Vertex</span> <span class="token keyword">for</span> <span class="token class-name">ModelVertex</span> <span class="token punctuation">{</span>
<span class="token keyword">fn</span> <span class="token function-definition function">desc</span><span class="token operator">&lt;</span><span class="token lifetime-annotation symbol">'a</span><span class="token operator">&gt;</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">-&gt;</span> <span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">VertexBufferLayout</span><span class="token operator">&lt;</span><span class="token lifetime-annotation symbol">'a</span><span class="token operator">&gt;</span> <span class="token punctuation">{</span>
<span class="token keyword">use</span> <span class="token namespace">std<span class="token punctuation">::</span></span>mem<span class="token punctuation">;</span>
<span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">VertexBufferLayout</span> <span class="token punctuation">{</span>
@ -67,7 +67,7 @@
<span class="token punctuation">}</span>
<span class="token punctuation">}</span>
<span class="token punctuation">}</span>
</code></pre></div><p>This is basically the same as the original <code>VertexBufferLayout</code>, but we added a <code>VertexAttribute</code> for the <code>normal</code>. Remove the <code>Vertex</code> struct in <code>main.rs</code> as we won't need it anymore, and use our new <code>Vertex</code> from model for the <code>RenderPipeline</code>.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">let</span> render_pipeline <span class="token operator">=</span> device<span class="token punctuation">.</span><span class="token function">create_render_pipeline</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">RenderPipelineDescriptor</span> <span class="token punctuation">{</span>
</code></pre></div><p>This is basically the same as the original <code>VertexBufferLayout</code>, but we added a <code>VertexAttribute</code> for the <code>normal</code>. Remove the <code>Vertex</code> struct in <code>lib.rs</code> as we won't need it anymore, and use our new <code>Vertex</code> from <code>model</code> for the <code>RenderPipeline</code>.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">let</span> render_pipeline <span class="token operator">=</span> device<span class="token punctuation">.</span><span class="token function">create_render_pipeline</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">RenderPipelineDescriptor</span> <span class="token punctuation">{</span>
<span class="token comment">// ...</span>
vertex<span class="token punctuation">:</span> <span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">VertexState</span> <span class="token punctuation">{</span>
<span class="token comment">// ...</span>
@ -76,7 +76,7 @@
<span class="token comment">// ...</span>
<span class="token punctuation">}</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
</code></pre></div><p>Since the <code>desc</code> method is implemented on the <code>Vertex</code> trait, the trait needs to be imported before the method will be accessible. Put the import towards the top of the file with the others.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">use</span> <span class="token namespace">model<span class="token punctuation">::</span></span><span class="token class-name">Vertex</span><span class="token punctuation">;</span>
</code></pre></div><p>With all that in place we need a model to render. If you have one already that's great, but I've supplied a <a href="https://github.com/sotrh/learn-wgpu/blob/master/code/beginner/tutorial9-models/res/cube.zip" target="_blank" rel="noopener noreferrer">zip file<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> with the model and all of it's textures. We're going to put this model in a new <code>res</code> folder next to the existing <code>src</code> folder.</p> <h2 id="accessing-files-in-the-res-folder"><a href="#accessing-files-in-the-res-folder" class="header-anchor">#</a> Accessing files in the res folder</h2> <p>When cargo builds and runs our program it sets what's known as the current working directory. This directory is usually the folder containing your projects root <code>Cargo.toml</code>. The path to our res folder may differ depending on the structure of the project. In the <code>res</code> folder for the example code for this section tutorial is at <code>code/beginner/tutorial9-models/res/</code>. When loading our model we could use this path, and just append <code>cube.obj</code>. This is fine, but if we change our projects structure, our code will break.</p> <p>We're going to fix that by modifying our build script to copy our <code>res</code> folder to where cargo creates our executable, and we'll reference it from there. Create a file called <code>build.rs</code> and add the following:</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">use</span> <span class="token namespace">anyhow<span class="token punctuation">::</span></span><span class="token operator">*</span><span class="token punctuation">;</span>
</code></pre></div><p>With all that in place, we need a model to render. If you have one already that's great, but I've supplied a <a href="https://github.com/sotrh/learn-wgpu/blob/master/code/beginner/tutorial9-models/res/cube.zip" target="_blank" rel="noopener noreferrer">zip file<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> with the model and all of its textures. We're going to put this model in a new <code>res</code> folder next to the existing <code>src</code> folder.</p> <h2 id="accessing-files-in-the-res-folder"><a href="#accessing-files-in-the-res-folder" class="header-anchor">#</a> Accessing files in the res folder</h2> <p>When cargo builds and runs our program it sets what's known as the current working directory. This directory is usually the folder containing your project's root <code>Cargo.toml</code>. The path to our res folder may differ depending on the structure of the project. In the <code>res</code> folder for the example code for this section tutorial is at <code>code/beginner/tutorial9-models/res/</code>. When loading our model we could use this path, and just append <code>cube.obj</code>. This is fine, but if we change our project's structure, our code will break.</p> <p>We're going to fix that by modifying our build script to copy our <code>res</code> folder to where cargo creates our executable, and we'll reference it from there. Create a file called <code>build.rs</code> and add the following:</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">use</span> <span class="token namespace">anyhow<span class="token punctuation">::</span></span><span class="token operator">*</span><span class="token punctuation">;</span>
<span class="token keyword">use</span> <span class="token namespace">fs_extra<span class="token punctuation">::</span></span>copy_items<span class="token punctuation">;</span>
<span class="token keyword">use</span> <span class="token namespace">fs_extra<span class="token punctuation">::</span>dir<span class="token punctuation">::</span></span><span class="token class-name">CopyOptions</span><span class="token punctuation">;</span>
<span class="token keyword">use</span> <span class="token namespace">std<span class="token punctuation">::</span></span>env<span class="token punctuation">;</span>
@ -98,7 +98,7 @@
<span class="token key property">anyhow</span> <span class="token punctuation">=</span> <span class="token string">&quot;1.0&quot;</span>
<span class="token key property">fs_extra</span> <span class="token punctuation">=</span> <span class="token string">&quot;1.2&quot;</span>
<span class="token key property">glob</span> <span class="token punctuation">=</span> <span class="token string">&quot;0.3&quot;</span>
</code></pre></div><h2 id="accessing-files-from-wasm"><a href="#accessing-files-from-wasm" class="header-anchor">#</a> Accessing files from WASM</h2> <p>By design, you can't access files on a users filesystem in Web Assembly. Instead we'll serve those files up using a web serve, and then load those files into our code using an http request. In order to simplify this, let's create a file called <code>resources.rs</code> to handle this for us. We'll create two functions that will load text files and binary files respectively.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">use</span> <span class="token namespace">std<span class="token punctuation">::</span>io<span class="token punctuation">::</span></span><span class="token punctuation">{</span><span class="token class-name">BufReader</span><span class="token punctuation">,</span> <span class="token class-name">Cursor</span><span class="token punctuation">}</span><span class="token punctuation">;</span>
</code></pre></div><h2 id="accessing-files-from-wasm"><a href="#accessing-files-from-wasm" class="header-anchor">#</a> Accessing files from WASM</h2> <p>By design, you can't access files on a user's filesystem in Web Assembly. Instead, we'll serve those files up using a web serve, and then load those files into our code using an http request. In order to simplify this, let's create a file called <code>resources.rs</code> to handle this for us. We'll create two functions that will load text files and binary files respectively.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">use</span> <span class="token namespace">std<span class="token punctuation">::</span>io<span class="token punctuation">::</span></span><span class="token punctuation">{</span><span class="token class-name">BufReader</span><span class="token punctuation">,</span> <span class="token class-name">Cursor</span><span class="token punctuation">}</span><span class="token punctuation">;</span>
<span class="token keyword">use</span> <span class="token namespace">cfg_if<span class="token punctuation">::</span></span>cfg_if<span class="token punctuation">;</span>
<span class="token keyword">use</span> <span class="token namespace">wgpu<span class="token punctuation">::</span>util<span class="token punctuation">::</span></span><span class="token class-name">DeviceExt</span><span class="token punctuation">;</span>
@ -155,7 +155,7 @@
<span class="token class-name">Ok</span><span class="token punctuation">(</span>data<span class="token punctuation">)</span>
<span class="token punctuation">}</span>
</code></pre></div><div class="note"><p>We're using <code>OUT_DIR</code> on desktop to get at our <code>res</code> folder.</p></div> <p>I'm using <a href="https://docs.rs/reqwest" target="_blank" rel="noopener noreferrer">reqwest<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> to handle loading the requests when using WASM. Add the following to the Cargo.toml:</p> <div class="language-toml extra-class"><pre class="language-toml"><code><span class="token punctuation">[</span><span class="token table class-name">target.'cfg(target_arch = &quot;wasm32&quot;)'.dependencies</span><span class="token punctuation">]</span>
</code></pre></div><div class="note"><p>We're using <code>OUT_DIR</code> on desktop to get to our <code>res</code> folder.</p></div> <p>I'm using <a href="https://docs.rs/reqwest" target="_blank" rel="noopener noreferrer">reqwest<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> to handle loading the requests when using WASM. Add the following to the Cargo.toml:</p> <div class="language-toml extra-class"><pre class="language-toml"><code><span class="token punctuation">[</span><span class="token table class-name">target.'cfg(target_arch = &quot;wasm32&quot;)'.dependencies</span><span class="token punctuation">]</span>
<span class="token comment"># Other dependencies</span>
<span class="token key property">reqwest</span> <span class="token punctuation">=</span> <span class="token punctuation">{</span> <span class="token key property">version</span> <span class="token punctuation">=</span> <span class="token string">&quot;0.11&quot;</span> <span class="token punctuation">}</span>
</code></pre></div><p>We'll also need to add the <code>Location</code> feature to <code>web-sys</code>:</p> <div class="language-toml extra-class"><pre class="language-toml"><code><span class="token key property">web-sys</span> <span class="token punctuation">=</span> <span class="token punctuation">{</span> <span class="token key property">version</span> <span class="token punctuation">=</span> <span class="token string">&quot;0.3&quot;</span><span class="token punctuation">,</span> <span class="token key property">features</span> <span class="token punctuation">=</span> <span class="token punctuation">[</span>
@ -316,7 +316,7 @@
<span class="token keyword">self</span><span class="token punctuation">.</span><span class="token function">draw_indexed</span><span class="token punctuation">(</span><span class="token number">0</span><span class="token punctuation">..</span>mesh<span class="token punctuation">.</span>num_elements<span class="token punctuation">,</span> <span class="token number">0</span><span class="token punctuation">,</span> instances<span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span>
<span class="token punctuation">}</span>
</code></pre></div><p>We could have put this methods in an <code>impl Model</code>, but I felt it made more sense to have the <code>RenderPass</code> do all the rendering, as that's kind of it's job. This does mean we have to import <code>DrawModel</code> when we go to render though.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// lib.rs</span>
</code></pre></div><p>We could have put these methods in an <code>impl Model</code>, but I felt it made more sense to have the <code>RenderPass</code> do all the rendering, as that's kind of its job. This does mean we have to import <code>DrawModel</code> when we go to render though.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// lib.rs</span>
render_pass<span class="token punctuation">.</span><span class="token function">set_vertex_buffer</span><span class="token punctuation">(</span><span class="token number">1</span><span class="token punctuation">,</span> <span class="token keyword">self</span><span class="token punctuation">.</span>instance_buffer<span class="token punctuation">.</span><span class="token function">slice</span><span class="token punctuation">(</span><span class="token punctuation">..</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
render_pass<span class="token punctuation">.</span><span class="token function">set_pipeline</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token keyword">self</span><span class="token punctuation">.</span>render_pipeline<span class="token punctuation">)</span><span class="token punctuation">;</span>
render_pass<span class="token punctuation">.</span><span class="token function">set_bind_group</span><span class="token punctuation">(</span><span class="token number">0</span><span class="token punctuation">,</span> <span class="token operator">&amp;</span><span class="token keyword">self</span><span class="token punctuation">.</span>diffuse_bind_group<span class="token punctuation">,</span> <span class="token operator">&amp;</span><span class="token punctuation">[</span><span class="token punctuation">]</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
@ -390,7 +390,7 @@ render_pass<span class="token punctuation">.</span><span class="token function">
<span class="token keyword">let</span> mesh <span class="token operator">=</span> <span class="token operator">&amp;</span><span class="token keyword">self</span><span class="token punctuation">.</span>obj_model<span class="token punctuation">.</span>meshes<span class="token punctuation">[</span><span class="token number">0</span><span class="token punctuation">]</span><span class="token punctuation">;</span>
<span class="token keyword">let</span> material <span class="token operator">=</span> <span class="token operator">&amp;</span><span class="token keyword">self</span><span class="token punctuation">.</span>obj_model<span class="token punctuation">.</span>materials<span class="token punctuation">[</span>mesh<span class="token punctuation">.</span>material<span class="token punctuation">]</span><span class="token punctuation">;</span>
render_pass<span class="token punctuation">.</span><span class="token function">draw_mesh_instanced</span><span class="token punctuation">(</span>mesh<span class="token punctuation">,</span> material<span class="token punctuation">,</span> <span class="token number">0</span><span class="token punctuation">..</span><span class="token keyword">self</span><span class="token punctuation">.</span>instances<span class="token punctuation">.</span><span class="token function">len</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token keyword">as</span> <span class="token keyword">u32</span><span class="token punctuation">,</span> <span class="token operator">&amp;</span><span class="token keyword">self</span><span class="token punctuation">.</span>camera_bind_group<span class="token punctuation">)</span><span class="token punctuation">;</span>
</code></pre></div><p>With all that in place we should get the following.</p> <p><img src="/learn-wgpu/assets/img/cubes-correct.2db711eb.png" alt="cubes-correct.png"></p> <h2 id="rendering-the-entire-model"><a href="#rendering-the-entire-model" class="header-anchor">#</a> Rendering the entire model</h2> <p>Right now we are specifying the mesh and the material directly. This is useful if we want to draw a mesh with a different material. We're also not rendering other parts of the model (if we had some). Let's create a method for <code>DrawModel</code> that will draw all the parts of the model with their respective materials.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">pub</span> <span class="token keyword">trait</span> <span class="token type-definition class-name">DrawModel</span><span class="token operator">&lt;</span><span class="token lifetime-annotation symbol">'a</span><span class="token operator">&gt;</span> <span class="token punctuation">{</span>
</code></pre></div><p>With all that in place, we should get the following.</p> <p><img src="/learn-wgpu/assets/img/cubes-correct.2db711eb.png" alt="cubes-correct.png"></p> <h2 id="rendering-the-entire-model"><a href="#rendering-the-entire-model" class="header-anchor">#</a> Rendering the entire model</h2> <p>Right now we are specifying the mesh and the material directly. This is useful if we want to draw a mesh with a different material. We're also not rendering other parts of the model (if we had some). Let's create a method for <code>DrawModel</code> that will draw all the parts of the model with their respective materials.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">pub</span> <span class="token keyword">trait</span> <span class="token type-definition class-name">DrawModel</span><span class="token operator">&lt;</span><span class="token lifetime-annotation symbol">'a</span><span class="token operator">&gt;</span> <span class="token punctuation">{</span>
<span class="token comment">// ...</span>
<span class="token keyword">fn</span> <span class="token function-definition function">draw_model</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token keyword">mut</span> <span class="token keyword">self</span><span class="token punctuation">,</span> model<span class="token punctuation">:</span> <span class="token operator">&amp;</span><span class="token lifetime-annotation symbol">'a</span> <span class="token class-name">Model</span><span class="token punctuation">,</span> camera_bind_group<span class="token punctuation">:</span> <span class="token operator">&amp;</span><span class="token lifetime-annotation symbol">'a</span> <span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">BindGroup</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token keyword">fn</span> <span class="token function-definition function">draw_model_instanced</span><span class="token punctuation">(</span>
@ -421,10 +421,10 @@ render_pass<span class="token punctuation">.</span><span class="token function">
<span class="token punctuation">}</span>
<span class="token punctuation">}</span>
<span class="token punctuation">}</span>
</code></pre></div><p>The code in <code>main.rs</code> will change accordingly.</p> <div class="language-rust extra-class"><pre class="language-rust"><code>render_pass<span class="token punctuation">.</span><span class="token function">set_vertex_buffer</span><span class="token punctuation">(</span><span class="token number">1</span><span class="token punctuation">,</span> <span class="token keyword">self</span><span class="token punctuation">.</span>instance_buffer<span class="token punctuation">.</span><span class="token function">slice</span><span class="token punctuation">(</span><span class="token punctuation">..</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
</code></pre></div><p>The code in <code>lib.rs</code> will change accordingly.</p> <div class="language-rust extra-class"><pre class="language-rust"><code>render_pass<span class="token punctuation">.</span><span class="token function">set_vertex_buffer</span><span class="token punctuation">(</span><span class="token number">1</span><span class="token punctuation">,</span> <span class="token keyword">self</span><span class="token punctuation">.</span>instance_buffer<span class="token punctuation">.</span><span class="token function">slice</span><span class="token punctuation">(</span><span class="token punctuation">..</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
render_pass<span class="token punctuation">.</span><span class="token function">set_pipeline</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token keyword">self</span><span class="token punctuation">.</span>render_pipeline<span class="token punctuation">)</span><span class="token punctuation">;</span>
render_pass<span class="token punctuation">.</span><span class="token function">draw_model_instanced</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token keyword">self</span><span class="token punctuation">.</span>obj_model<span class="token punctuation">,</span> <span class="token number">0</span><span class="token punctuation">..</span><span class="token keyword">self</span><span class="token punctuation">.</span>instances<span class="token punctuation">.</span><span class="token function">len</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token keyword">as</span> <span class="token keyword">u32</span><span class="token punctuation">,</span> <span class="token operator">&amp;</span><span class="token keyword">self</span><span class="token punctuation">.</span>camera_bind_group<span class="token punctuation">)</span><span class="token punctuation">;</span>
</code></pre></div><div id="wasm-example"><!----> <button>Try Tutorial9_models!</button></div> <div class="auto-github-link"><a href="https://github.com/sotrh/learn-wgpu/tree/master/code/beginner/tutorial9-models/" target="_blank" rel="noopener noreferrer">Check out the code!</a> <span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></div></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:41:58 PM</span></div></footer> <div class="page-nav"><p class="inner"><span class="prev">
</code></pre></div><div id="wasm-example"><!----> <button>Try Tutorial9_models!</button></div> <div class="auto-github-link"><a href="https://github.com/sotrh/learn-wgpu/tree/master/code/beginner/tutorial9-models/" target="_blank" rel="noopener noreferrer">Check out the code!</a> <span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></div></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:44:34 PM</span></div></footer> <div class="page-nav"><p class="inner"><span class="prev">
<a href="/learn-wgpu/beginner/tutorial8-depth/" class="prev">
The Depth Buffer
@ -433,6 +433,6 @@ render_pass<span class="token punctuation">.</span><span class="token function">
</a>
</span></p></div> </main></div></div><div class="global-ui"><!----></div></div>
<script src="/learn-wgpu/assets/js/app.cbd0d45d.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/25.32361a22.js" defer></script><script src="/learn-wgpu/assets/js/22.ae08829d.js" defer></script><script src="/learn-wgpu/assets/js/26.9b876ccf.js" defer></script>
<script src="/learn-wgpu/assets/js/app.c87751e7.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/25.a63ee8c2.js" defer></script><script src="/learn-wgpu/assets/js/22.ae08829d.js" defer></script><script src="/learn-wgpu/assets/js/26.9b876ccf.js" defer></script>
</body>
</html>

File diff suppressed because one or more lines are too long

@ -7,7 +7,7 @@
<meta name="generator" content="VuePress 1.9.7">
<meta name="description" content="">
<meta property="article:modified_time" content="2022-05-14T18:41:58.000Z">
<meta property="article:modified_time" content="2022-05-14T18:44:34.000Z">
<meta property="og:site_name" content="Learn Wgpu">
<meta property="og:title" content="Sources">
<meta property="og:type" content="website">
@ -19,11 +19,11 @@
<meta name="twitter:data1" content="Benjamin Hansen">
<meta name="twitter:creator" content="https://twitter.com/sotrh760">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.cbd0d45d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/42.82e26587.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.576af98e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.e74dc361.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.69a9ea7e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.e284d6f3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/22.ae08829d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.9d07c0eb.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.6982a462.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.32361a22.js"><link rel="prefetch" href="/learn-wgpu/assets/js/26.9b876ccf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.9fe49a03.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.c212a80a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.b61fde25.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.25585faa.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.acd07b95.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.a51ea3c5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.41e9897b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.2d445b19.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.9b4d649e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.e0f57de3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.155fa5a2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.d3e6544b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.2bed1539.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.9a3ae8d9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.4d927984.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.bcb97198.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.c87751e7.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/42.82e26587.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.e0b6e526.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.c8349761.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.2469c630.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.3972e66e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/22.ae08829d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.d649043b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.5b5cf4ba.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.a63ee8c2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/26.9b876ccf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.064a85a6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.fb933617.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.2e0b419b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.09885c9c.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.19c7b8dd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.9a017b74.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.695eaab4.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.1ae459af.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.bb1a0e6a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.34968282.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.58137882.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.c80ee966.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.740cfe90.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.5be8c6e7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.2e48975d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.ec58808b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="stylesheet" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css">
</head>
<body>
<div id="app" data-server-rendered="true"><div class="theme-container"><header class="navbar"><div class="inner"><div class="sidebar-button"><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" viewBox="0 0 448 512" class="icon"><path fill="currentColor" d="M436 124H12c-6.627 0-12-5.373-12-12V80c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12z"></path></svg></div> <a href="/learn-wgpu/" class="home-link router-link-active"><!----> <span class="site-name">Learn Wgpu</span></a> <div class="links"><!----> <div class="search-box"><input aria-label="Search" autocomplete="off" spellcheck="false" value=""> <!----></div></div></div></header> <div class="sidebar-mask"></div> <div class="docs-layout"><aside class="sidebar"><!----> <ul class="sidebar-links"><li><a href="/learn-wgpu/" aria-current="page" class="sidebar-link">Introduction</a></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Beginner</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/beginner/tutorial1-window/" class="sidebar-link">Dependencies and the window</a></li><li><a href="/learn-wgpu/beginner/tutorial2-surface/" class="sidebar-link">The Surface</a></li><li><a href="/learn-wgpu/beginner/tutorial3-pipeline/" class="sidebar-link">The Pipeline</a></li><li><a href="/learn-wgpu/beginner/tutorial4-buffer/" class="sidebar-link">Buffers and Indices</a></li><li><a href="/learn-wgpu/beginner/tutorial5-textures/" class="sidebar-link">Textures and bind groups</a></li><li><a href="/learn-wgpu/beginner/tutorial6-uniforms/" class="sidebar-link">Uniform buffers and a 3d camera</a></li><li><a href="/learn-wgpu/beginner/tutorial7-instancing/" class="sidebar-link">Instancing</a></li><li><a href="/learn-wgpu/beginner/tutorial8-depth/" class="sidebar-link">The Depth Buffer</a></li><li><a href="/learn-wgpu/beginner/tutorial9-models/" class="sidebar-link">Model Loading</a></li></ul></section></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Intermediate</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/intermediate/tutorial10-lighting/" class="sidebar-link">Working with Lights</a></li><li><a href="/learn-wgpu/intermediate/tutorial11-normals/" class="sidebar-link">Normal Mapping</a></li><li><a href="/learn-wgpu/intermediate/tutorial12-camera/" class="sidebar-link">A Better Camera</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>Showcase</span> <span class="arrow right"></span></p> <!----></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>News</span> <span class="arrow right"></span></p> <!----></section></li></ul> </aside> <main class="page"> <div class="theme-default-content content__default"><h1 id="sources"><a href="#sources" class="header-anchor">#</a> Sources</h1> <ul><li>https://www.scratchapixel.com/lessons/3d-basic-rendering/introduction-to-shading/diffuse-lambertian-shading</li> <li>http://jimmiejohnsson84.me/pages/rendering_pbr.html</li></ul></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:41:58 PM</span></div></footer> <!----> </main></div></div><div class="global-ui"><!----></div></div>
<script src="/learn-wgpu/assets/js/app.cbd0d45d.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/42.82e26587.js" defer></script>
<div id="app" data-server-rendered="true"><div class="theme-container"><header class="navbar"><div class="inner"><div class="sidebar-button"><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" viewBox="0 0 448 512" class="icon"><path fill="currentColor" d="M436 124H12c-6.627 0-12-5.373-12-12V80c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12z"></path></svg></div> <a href="/learn-wgpu/" class="home-link router-link-active"><!----> <span class="site-name">Learn Wgpu</span></a> <div class="links"><!----> <div class="search-box"><input aria-label="Search" autocomplete="off" spellcheck="false" value=""> <!----></div></div></div></header> <div class="sidebar-mask"></div> <div class="docs-layout"><aside class="sidebar"><!----> <ul class="sidebar-links"><li><a href="/learn-wgpu/" aria-current="page" class="sidebar-link">Introduction</a></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Beginner</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/beginner/tutorial1-window/" class="sidebar-link">Dependencies and the window</a></li><li><a href="/learn-wgpu/beginner/tutorial2-surface/" class="sidebar-link">The Surface</a></li><li><a href="/learn-wgpu/beginner/tutorial3-pipeline/" class="sidebar-link">The Pipeline</a></li><li><a href="/learn-wgpu/beginner/tutorial4-buffer/" class="sidebar-link">Buffers and Indices</a></li><li><a href="/learn-wgpu/beginner/tutorial5-textures/" class="sidebar-link">Textures and bind groups</a></li><li><a href="/learn-wgpu/beginner/tutorial6-uniforms/" class="sidebar-link">Uniform buffers and a 3d camera</a></li><li><a href="/learn-wgpu/beginner/tutorial7-instancing/" class="sidebar-link">Instancing</a></li><li><a href="/learn-wgpu/beginner/tutorial8-depth/" class="sidebar-link">The Depth Buffer</a></li><li><a href="/learn-wgpu/beginner/tutorial9-models/" class="sidebar-link">Model Loading</a></li></ul></section></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Intermediate</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/intermediate/tutorial10-lighting/" class="sidebar-link">Working with Lights</a></li><li><a href="/learn-wgpu/intermediate/tutorial11-normals/" class="sidebar-link">Normal Mapping</a></li><li><a href="/learn-wgpu/intermediate/tutorial12-camera/" class="sidebar-link">A Better Camera</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>Showcase</span> <span class="arrow right"></span></p> <!----></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>News</span> <span class="arrow right"></span></p> <!----></section></li></ul> </aside> <main class="page"> <div class="theme-default-content content__default"><h1 id="sources"><a href="#sources" class="header-anchor">#</a> Sources</h1> <ul><li>https://www.scratchapixel.com/lessons/3d-basic-rendering/introduction-to-shading/diffuse-lambertian-shading</li> <li>http://jimmiejohnsson84.me/pages/rendering_pbr.html</li></ul></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:44:34 PM</span></div></footer> <!----> </main></div></div><div class="global-ui"><!----></div></div>
<script src="/learn-wgpu/assets/js/app.c87751e7.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/42.82e26587.js" defer></script>
</body>
</html>

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

@ -7,7 +7,7 @@
<meta name="generator" content="VuePress 1.9.7">
<meta name="description" content="">
<meta property="article:modified_time" content="2022-05-14T18:41:58.000Z">
<meta property="article:modified_time" content="2022-05-14T18:44:34.000Z">
<meta property="og:site_name" content="Learn Wgpu">
<meta property="og:title" content="A Better Camera">
<meta property="og:type" content="website">
@ -19,7 +19,7 @@
<meta name="twitter:data1" content="Benjamin Hansen">
<meta name="twitter:creator" content="https://twitter.com/sotrh760">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.cbd0d45d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/31.25585faa.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/22.ae08829d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/26.9b876ccf.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.576af98e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.e74dc361.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.69a9ea7e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.e284d6f3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.9d07c0eb.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.6982a462.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.32361a22.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.9fe49a03.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.c212a80a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.b61fde25.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.acd07b95.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.a51ea3c5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.41e9897b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.2d445b19.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.9b4d649e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.e0f57de3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.155fa5a2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.d3e6544b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.2bed1539.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.9a3ae8d9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.4d927984.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.bcb97198.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.c87751e7.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/31.09885c9c.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/22.ae08829d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/26.9b876ccf.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.e0b6e526.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.c8349761.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.2469c630.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.3972e66e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.d649043b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.5b5cf4ba.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.a63ee8c2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.064a85a6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.fb933617.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.2e0b419b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.19c7b8dd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.9a017b74.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.695eaab4.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.1ae459af.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.bb1a0e6a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.34968282.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.58137882.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.c80ee966.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.740cfe90.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.5be8c6e7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.2e48975d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.ec58808b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="stylesheet" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css">
</head>
<body>
@ -28,7 +28,7 @@
<span class="token punctuation">[</span><span class="token table class-name">target.'cfg(target_arch = &quot;wasm32&quot;)'.dependencies</span><span class="token punctuation">]</span>
<span class="token key property">wgpu</span> <span class="token punctuation">=</span> <span class="token punctuation">{</span> <span class="token key property">version</span> <span class="token punctuation">=</span> <span class="token string">&quot;0.12&quot;</span><span class="token punctuation">,</span> <span class="token key property">git</span><span class="token punctuation">=</span><span class="token string">&quot;https://github.com/gfx-rs/wgpu&quot;</span><span class="token punctuation">,</span> <span class="token key property">branch</span><span class="token punctuation">=</span><span class="token string">&quot;gecko&quot;</span><span class="token punctuation">,</span> <span class="token key property">features</span> <span class="token punctuation">=</span> <span class="token punctuation">[</span><span class="token string">&quot;webgl&quot;</span><span class="token punctuation">]</span><span class="token punctuation">}</span>
</code></pre></div><p>Once 0.13 comes out I'll revert to using the version published on crates.io.</p></div> <p>I've been putting this off for a while. Implementing a camera isn't specifically related to using WGPU properly, but it's been bugging me so let's do it.</p> <p><code>main.rs</code> is getting a little crowded, so let's create a <code>camera.rs</code> file to put our camera code. The first thing we're going to put in it in is some imports and our <code>OPENGL_TO_WGPU_MATRIX</code>.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">use</span> <span class="token namespace">cgmath<span class="token punctuation">::</span></span><span class="token operator">*</span><span class="token punctuation">;</span>
</code></pre></div><p>Once 0.13 comes out I'll revert to using the version published on crates.io.</p></div> <p>I've been putting this off for a while. Implementing a camera isn't specifically related to using WGPU properly, but it's been bugging me so let's do it.</p> <p><code>main.rs</code> is getting a little crowded, so let's create a <code>camera.rs</code> file to put our camera code. The first things we're going to put in it are some imports and our <code>OPENGL_TO_WGPU_MATRIX</code>.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">use</span> <span class="token namespace">cgmath<span class="token punctuation">::</span></span><span class="token operator">*</span><span class="token punctuation">;</span>
<span class="token keyword">use</span> <span class="token namespace">winit<span class="token punctuation">::</span>event<span class="token punctuation">::</span></span><span class="token operator">*</span><span class="token punctuation">;</span>
<span class="token keyword">use</span> <span class="token namespace">winit<span class="token punctuation">::</span>dpi<span class="token punctuation">::</span></span><span class="token class-name">PhysicalPosition</span><span class="token punctuation">;</span>
<span class="token keyword">use</span> <span class="token namespace">instant<span class="token punctuation">::</span></span><span class="token class-name">Duration</span><span class="token punctuation">;</span>
@ -44,7 +44,7 @@
<span class="token keyword">const</span> <span class="token constant">SAFE_FRAC_PI_2</span><span class="token punctuation">:</span> <span class="token keyword">f32</span> <span class="token operator">=</span> <span class="token constant">FRAC_PI_2</span> <span class="token operator">-</span> <span class="token number">0.0001</span><span class="token punctuation">;</span>
</code></pre></div><div class="note"><p><code>std::time::Instant</code> panics on WASM, so we'll use the <a href="https://docs.rs/instant" target="_blank" rel="noopener noreferrer">instant crate<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a>. You'll want to include it in your <code>Cargo.toml</code>:</p> <div class="language-toml extra-class"><pre class="language-toml"><code><span class="token key property">instant</span> <span class="token punctuation">=</span> <span class="token string">&quot;0.1&quot;</span>
</code></pre></div></div> <h2 id="the-camera"><a href="#the-camera" class="header-anchor">#</a> The Camera</h2> <p>Next we need create a new <code>Camera</code> struct. We're going to be using a FPS style camera, so we'll store the position and the yaw (horizontal rotation), and pitch (vertical rotation). We'll have a <code>calc_matrix</code> method to create our view matrix.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token attribute attr-name">#[derive(Debug)]</span>
</code></pre></div></div> <h2 id="the-camera"><a href="#the-camera" class="header-anchor">#</a> The Camera</h2> <p>Next, we need to create a new <code>Camera</code> struct. We're going to be using an FPS-style camera, so we'll store the position and the yaw (horizontal rotation), and pitch (vertical rotation). We'll have a <code>calc_matrix</code> method to create our view matrix.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token attribute attr-name">#[derive(Debug)]</span>
<span class="token keyword">pub</span> <span class="token keyword">struct</span> <span class="token type-definition class-name">Camera</span> <span class="token punctuation">{</span>
<span class="token keyword">pub</span> position<span class="token punctuation">:</span> <span class="token class-name">Point3</span><span class="token operator">&lt;</span><span class="token keyword">f32</span><span class="token operator">&gt;</span><span class="token punctuation">,</span>
yaw<span class="token punctuation">:</span> <span class="token class-name">Rad</span><span class="token operator">&lt;</span><span class="token keyword">f32</span><span class="token operator">&gt;</span><span class="token punctuation">,</span>
@ -111,7 +111,7 @@
<span class="token constant">OPENGL_TO_WGPU_MATRIX</span> <span class="token operator">*</span> <span class="token function">perspective</span><span class="token punctuation">(</span><span class="token keyword">self</span><span class="token punctuation">.</span>fovy<span class="token punctuation">,</span> <span class="token keyword">self</span><span class="token punctuation">.</span>aspect<span class="token punctuation">,</span> <span class="token keyword">self</span><span class="token punctuation">.</span>znear<span class="token punctuation">,</span> <span class="token keyword">self</span><span class="token punctuation">.</span>zfar<span class="token punctuation">)</span>
<span class="token punctuation">}</span>
<span class="token punctuation">}</span>
</code></pre></div><p>On thing to note: <code>cgmath</code> currently returns a right-handed projection matrix from the <code>perspective</code> function. This means that the z-axis points out of the screen. If you want the z-axis to be <em>into</em> the screen (aka. a left-handed projection matrix), you'll have to code your own.</p> <p>You can tell the difference between a right-handed coordinate system and a left-handed one by using your hands. Point your thumb to the right. This is the x-axis. Point your pointer finger up. This is the y-axis. Extend your middle finger. This is the z-axis. On your right hand your middle finger should be pointing towards you. On your left hand it should be pointing away.</p> <p><img src="/learn-wgpu/assets/img/left_right_hand.ccabf5d0.gif" alt="./left_right_hand.gif"></p> <h1 id="the-camera-controller"><a href="#the-camera-controller" class="header-anchor">#</a> The Camera Controller</h1> <p>Our camera is different, so we'll need a new camera controller. Add the following to <code>camera.rs</code>.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token attribute attr-name">#[derive(Debug)]</span>
</code></pre></div><p>One thing to note: <code>cgmath</code> currently returns a right-handed projection matrix from the <code>perspective</code> function. This means that the z-axis points out of the screen. If you want the z-axis to be <em>into</em> the screen (aka. a left-handed projection matrix), you'll have to code your own.</p> <p>You can tell the difference between a right-handed coordinate system and a left-handed one by using your hands. Point your thumb to the right. This is the x-axis. Point your pointer finger up. This is the y-axis. Extend your middle finger. This is the z-axis. On your right hand, your middle finger should be pointing towards you. On your left hand, it should be pointing away.</p> <p><img src="/learn-wgpu/assets/img/left_right_hand.ccabf5d0.gif" alt="./left_right_hand.gif"></p> <h1 id="the-camera-controller"><a href="#the-camera-controller" class="header-anchor">#</a> The Camera Controller</h1> <p>Our camera is different, so we'll need a new camera controller. Add the following to <code>camera.rs</code>.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token attribute attr-name">#[derive(Debug)]</span>
<span class="token keyword">pub</span> <span class="token keyword">struct</span> <span class="token type-definition class-name">CameraController</span> <span class="token punctuation">{</span>
amount_left<span class="token punctuation">:</span> <span class="token keyword">f32</span><span class="token punctuation">,</span>
amount_right<span class="token punctuation">:</span> <span class="token keyword">f32</span><span class="token punctuation">,</span>
@ -283,7 +283,7 @@
<span class="token keyword">self</span><span class="token punctuation">.</span>projection<span class="token punctuation">.</span><span class="token function">resize</span><span class="token punctuation">(</span>new_size<span class="token punctuation">.</span>width<span class="token punctuation">,</span> new_size<span class="token punctuation">.</span>height<span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token comment">// ...</span>
<span class="token punctuation">}</span>
</code></pre></div><p><code>input()</code> will need to be updated as well. Up to this point we have been using <code>WindowEvent</code>s for our camera controls. While this works, it's not the best solution. The <a href="https://docs.rs/winit/0.24.0/winit/event/enum.WindowEvent.html?search=#variant.CursorMoved" target="_blank" rel="noopener noreferrer">winit docs<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> inform us that OS will often transform the data for the <code>CursorMoved</code> event to allow effects such as cursor acceleration.</p> <p>Now to fix this we could change the <code>input()</code> function to process <code>DeviceEvent</code> instead of <code>WindowEvent</code>, but keyboard and button presses don't get emitted as <code>DeviceEvent</code>s on MacOS and WASM. Instead, we'll just remove the <code>CursorMoved</code> check in <code>input()</code>, and a manual call to <code>camera_controller.process_mouse()</code> in the <code>run()</code> function.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// UPDATED!</span>
</code></pre></div><p><code>input()</code> will need to be updated as well. Up to this point, we have been using <code>WindowEvent</code>s for our camera controls. While this works, it's not the best solution. The <a href="https://docs.rs/winit/0.24.0/winit/event/enum.WindowEvent.html?search=#variant.CursorMoved" target="_blank" rel="noopener noreferrer">winit docs<span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></a> inform us that OS will often transform the data for the <code>CursorMoved</code> event to allow effects such as cursor acceleration.</p> <p>Now to fix this we could change the <code>input()</code> function to process <code>DeviceEvent</code> instead of <code>WindowEvent</code>, but keyboard and button presses don't get emitted as <code>DeviceEvent</code>s on MacOS and WASM. Instead, we'll just remove the <code>CursorMoved</code> check in <code>input()</code>, and a manual call to <code>camera_controller.process_mouse()</code> in the <code>run()</code> function.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// UPDATED!</span>
<span class="token keyword">fn</span> <span class="token function-definition function">input</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token keyword">mut</span> <span class="token keyword">self</span><span class="token punctuation">,</span> event<span class="token punctuation">:</span> <span class="token operator">&amp;</span><span class="token class-name">WindowEvent</span><span class="token punctuation">)</span> <span class="token punctuation">-&gt;</span> <span class="token keyword">bool</span> <span class="token punctuation">{</span>
<span class="token keyword">match</span> event <span class="token punctuation">{</span>
<span class="token class-name">WindowEvent</span><span class="token punctuation">::</span><span class="token class-name">KeyboardInput</span> <span class="token punctuation">{</span>
@ -353,7 +353,7 @@
<span class="token punctuation">}</span>
<span class="token punctuation">}</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span>
</code></pre></div><p>The <code>update</code> function requires a bit more explanation. The <code>update_camera</code> function on the <code>CameraController</code> has a parameter <code>dt: Duration</code> which is the delta time or time between frames. This is to help smooth out the camera movement so that it's not locked be the framerate. Currently we aren't calculating <code>dt</code>, so I decided to pass it into <code>update</code> as a parameter.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">fn</span> <span class="token function-definition function">update</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token keyword">mut</span> <span class="token keyword">self</span><span class="token punctuation">,</span> dt<span class="token punctuation">:</span> <span class="token namespace">instant<span class="token punctuation">::</span></span><span class="token class-name">Duration</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
</code></pre></div><p>The <code>update</code> function requires a bit more explanation. The <code>update_camera</code> function on the <code>CameraController</code> has a parameter <code>dt: Duration</code> which is the delta time or time between frames. This is to help smooth out the camera movement so that it's not locked by the framerate. Currently, we aren't calculating <code>dt</code>, so I decided to pass it into <code>update</code> as a parameter.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">fn</span> <span class="token function-definition function">update</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token keyword">mut</span> <span class="token keyword">self</span><span class="token punctuation">,</span> dt<span class="token punctuation">:</span> <span class="token namespace">instant<span class="token punctuation">::</span></span><span class="token class-name">Duration</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
<span class="token comment">// UPDATED!</span>
<span class="token keyword">self</span><span class="token punctuation">.</span>camera_controller<span class="token punctuation">.</span><span class="token function">update_camera</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token keyword">mut</span> <span class="token keyword">self</span><span class="token punctuation">.</span>camera<span class="token punctuation">,</span> dt<span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token keyword">self</span><span class="token punctuation">.</span>camera_uniform<span class="token punctuation">.</span><span class="token function">update_view_proj</span><span class="token punctuation">(</span><span class="token operator">&amp;</span><span class="token keyword">self</span><span class="token punctuation">.</span>camera<span class="token punctuation">,</span> <span class="token operator">&amp;</span><span class="token keyword">self</span><span class="token punctuation">.</span>projection<span class="token punctuation">)</span><span class="token punctuation">;</span>
@ -383,7 +383,7 @@
<span class="token punctuation">}</span>
<span class="token punctuation">}</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span>
</code></pre></div><p>With that we should be able to move our camera wherever we want.</p> <p><img src="/learn-wgpu/assets/img/screenshot.4f5740bc.png" alt="./screenshot.png"></p> <div id="wasm-example"><!----> <button>Try Tutorial12_camera!</button></div> <div class="auto-github-link"><a href="https://github.com/sotrh/learn-wgpu/tree/master/code/intermediate/tutorial12-camera/" target="_blank" rel="noopener noreferrer">Check out the code!</a> <span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></div></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:41:58 PM</span></div></footer> <div class="page-nav"><p class="inner"><span class="prev">
</code></pre></div><p>With that, we should be able to move our camera wherever we want.</p> <p><img src="/learn-wgpu/assets/img/screenshot.4f5740bc.png" alt="./screenshot.png"></p> <div id="wasm-example"><!----> <button>Try Tutorial12_camera!</button></div> <div class="auto-github-link"><a href="https://github.com/sotrh/learn-wgpu/tree/master/code/intermediate/tutorial12-camera/" target="_blank" rel="noopener noreferrer">Check out the code!</a> <span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></div></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:44:34 PM</span></div></footer> <div class="page-nav"><p class="inner"><span class="prev">
<a href="/learn-wgpu/intermediate/tutorial11-normals/" class="prev">
Normal Mapping
@ -392,6 +392,6 @@
</a>
</span></p></div> </main></div></div><div class="global-ui"><!----></div></div>
<script src="/learn-wgpu/assets/js/app.cbd0d45d.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/31.25585faa.js" defer></script><script src="/learn-wgpu/assets/js/22.ae08829d.js" defer></script><script src="/learn-wgpu/assets/js/26.9b876ccf.js" defer></script>
<script src="/learn-wgpu/assets/js/app.c87751e7.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/31.09885c9c.js" defer></script><script src="/learn-wgpu/assets/js/22.ae08829d.js" defer></script><script src="/learn-wgpu/assets/js/26.9b876ccf.js" defer></script>
</body>
</html>

@ -7,7 +7,7 @@
<meta name="generator" content="VuePress 1.9.7">
<meta name="description" content="">
<meta property="article:modified_time" content="2022-05-14T18:41:58.000Z">
<meta property="article:modified_time" content="2022-05-14T18:44:34.000Z">
<meta property="og:site_name" content="Learn Wgpu">
<meta property="og:title" content="Multi-threading with Wgpu and Rayon">
<meta property="og:type" content="website">
@ -19,7 +19,7 @@
<meta name="twitter:data1" content="Benjamin Hansen">
<meta name="twitter:creator" content="https://twitter.com/sotrh760">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.cbd0d45d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/43.2bed1539.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/22.ae08829d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/26.9b876ccf.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.576af98e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.e74dc361.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.69a9ea7e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.e284d6f3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.9d07c0eb.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.6982a462.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.32361a22.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.9fe49a03.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.c212a80a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.b61fde25.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.25585faa.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.acd07b95.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.a51ea3c5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.41e9897b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.2d445b19.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.9b4d649e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.e0f57de3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.155fa5a2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.d3e6544b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.9a3ae8d9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.4d927984.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.bcb97198.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.c87751e7.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/43.740cfe90.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/22.ae08829d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/26.9b876ccf.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.e0b6e526.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.c8349761.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.2469c630.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.3972e66e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.d649043b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.5b5cf4ba.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.a63ee8c2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.064a85a6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.fb933617.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.2e0b419b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.09885c9c.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.19c7b8dd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.9a017b74.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.695eaab4.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.1ae459af.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.bb1a0e6a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.34968282.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.58137882.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.c80ee966.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.5be8c6e7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.2e48975d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.ec58808b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="stylesheet" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css">
</head>
<body>
@ -28,7 +28,7 @@
<span class="token punctuation">[</span><span class="token table class-name">target.'cfg(target_arch = &quot;wasm32&quot;)'.dependencies</span><span class="token punctuation">]</span>
<span class="token key property">wgpu</span> <span class="token punctuation">=</span> <span class="token punctuation">{</span> <span class="token key property">version</span> <span class="token punctuation">=</span> <span class="token string">&quot;0.12&quot;</span><span class="token punctuation">,</span> <span class="token key property">git</span><span class="token punctuation">=</span><span class="token string">&quot;https://github.com/gfx-rs/wgpu&quot;</span><span class="token punctuation">,</span> <span class="token key property">branch</span><span class="token punctuation">=</span><span class="token string">&quot;gecko&quot;</span><span class="token punctuation">,</span> <span class="token key property">features</span> <span class="token punctuation">=</span> <span class="token punctuation">[</span><span class="token string">&quot;webgl&quot;</span><span class="token punctuation">]</span><span class="token punctuation">}</span>
</code></pre></div><p>Once 0.13 comes out I'll revert to using the version published on crates.io.</p></div> <p>The main selling point of Vulkan, DirectX 12, Metal, and by extension Wgpu is that these APIs is that they designed from the ground up to be thread safe. Up to this point we have been doing everything on a single thread. That's about to change.</p> <div class="note"><p>I won't go into what threads are in this tutorial. That is a course in and of itself. All we'll be covering is using threading to make loading resources faster.</p> <p>We won't go over multithreading rendering as we don't have enough different types of objects to justify that yet. This will change in a coming tutorial</p></div> <h2 id="parallelizing-loading-models-and-textures"><a href="#parallelizing-loading-models-and-textures" class="header-anchor">#</a> Parallelizing loading models and textures</h2> <p>Currently we load the materials and meshes of our model one at a time. This is a perfect opportunity for multithreading! All our changes will be in <code>model.rs</code>. Let's first start with the materials. We'll convert the regular for loop into a <code>par_iter().map()</code>.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// resources.rs</span>
</code></pre></div><p>Once 0.13 comes out I'll revert to using the version published on crates.io.</p></div> <p>The main selling point of Vulkan, DirectX 12, Metal, and by extension Wgpu is that these APIs is that they designed from the ground up to be thread safe. Up to this point, we have been doing everything on a single thread. That's about to change.</p> <div class="note"><p>I won't go into what threads are in this tutorial. That is a course in and of itself. All we'll be covering is using threading to make loading resources faster.</p> <p>We won't go over multithreading rendering as we don't have enough different types of objects to justify that yet. This will change in a coming tutorial</p></div> <h2 id="parallelizing-loading-models-and-textures"><a href="#parallelizing-loading-models-and-textures" class="header-anchor">#</a> Parallelizing loading models and textures</h2> <p>Currently, we load the materials and meshes of our model one at a time. This is a perfect opportunity for multithreading! All our changes will be in <code>model.rs</code>. Let's first start with the materials. We'll convert the regular for loop into a <code>par_iter().map()</code>.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token comment">// resources.rs</span>
<span class="token attribute attr-name">#[cfg(not(target_arch=<span class="token string">&quot;wasm32&quot;</span>))]</span>
<span class="token keyword">use</span> <span class="token namespace">rayon<span class="token punctuation">::</span>iter<span class="token punctuation">::</span></span><span class="token class-name">IntoParallelIterator</span><span class="token punctuation">;</span>
@ -67,7 +67,7 @@
<span class="token punctuation">}</span>
<span class="token comment">// ...</span>
<span class="token punctuation">}</span>
</code></pre></div><p>Next we can update the meshes to be loaded in parallel.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">impl</span> <span class="token class-name">Model</span> <span class="token punctuation">{</span>
</code></pre></div><p>Next, we can update the meshes to be loaded in parallel.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">impl</span> <span class="token class-name">Model</span> <span class="token punctuation">{</span>
<span class="token keyword">pub</span> <span class="token keyword">fn</span> <span class="token function-definition function">load</span><span class="token operator">&lt;</span><span class="token class-name">P</span><span class="token punctuation">:</span> <span class="token class-name">AsRef</span><span class="token operator">&lt;</span><span class="token class-name">Path</span><span class="token operator">&gt;&gt;</span><span class="token punctuation">(</span>
device<span class="token punctuation">:</span> <span class="token operator">&amp;</span><span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">Device</span><span class="token punctuation">,</span>
queue<span class="token punctuation">:</span> <span class="token operator">&amp;</span><span class="token namespace">wgpu<span class="token punctuation">::</span></span><span class="token class-name">Queue</span><span class="token punctuation">,</span>
@ -118,7 +118,7 @@
<span class="token punctuation">}</span>
</code></pre></div><p>We've parallelized loading the meshes, and making the vertex array for them. Probably a bit overkill, but <code>rayon</code> should prevent us from using too many threads.</p> <div class="note"><p>You'll notice that we didn't use <code>rayon</code> for calculating the tangent, and bitangent. I tried to get it to work, but I was having trouble finding a way to do it without multiple mutable references to <code>vertices</code>. I don't feel like introducing a <code>std::sync::Mutex</code>, so I'll leave it for now.</p> <p>This is honestly a better job for a compute shader, as the model data is going to get loaded into a buffer anyway.</p></div> <h2 id="it-s-that-easy"><a href="#it-s-that-easy" class="header-anchor">#</a> It's that easy!</h2> <p>Most of the <code>wgpu</code> types are <code>Send + Sync</code>, so we can use them in threads without much trouble. It was so easy, that I feel like this tutorial is too short! I'll just leave off with a speed comparison between the previous model loading code and the current code.</p> <div class="language- extra-class"><pre class="language-text"><code>Elapsed (Original): 309.596382ms
Elapsed (Threaded): 199.645027ms
</code></pre></div><p>We're not loading that many resources, so the speed up is minimal. We'll be doing more stuff with threading, but this is a good introduction.</p> <div id="wasm-example"><!----> <button>Try Tutorial12_camera!</button></div> <div class="auto-github-link"><a href="https://github.com/sotrh/learn-wgpu/tree/master/code/intermediate/tutorial13-threading/" target="_blank" rel="noopener noreferrer">Check out the code!</a> <span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></div></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:41:58 PM</span></div></footer> <!----> </main></div></div><div class="global-ui"><!----></div></div>
<script src="/learn-wgpu/assets/js/app.cbd0d45d.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/43.2bed1539.js" defer></script><script src="/learn-wgpu/assets/js/22.ae08829d.js" defer></script><script src="/learn-wgpu/assets/js/26.9b876ccf.js" defer></script>
</code></pre></div><p>We're not loading that many resources, so the speedup is minimal. We'll be doing more stuff with threading, but this is a good introduction.</p> <div id="wasm-example"><!----> <button>Try Tutorial12_camera!</button></div> <div class="auto-github-link"><a href="https://github.com/sotrh/learn-wgpu/tree/master/code/intermediate/tutorial13-threading/" target="_blank" rel="noopener noreferrer">Check out the code!</a> <span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></div></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:44:34 PM</span></div></footer> <!----> </main></div></div><div class="global-ui"><!----></div></div>
<script src="/learn-wgpu/assets/js/app.c87751e7.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/43.740cfe90.js" defer></script><script src="/learn-wgpu/assets/js/22.ae08829d.js" defer></script><script src="/learn-wgpu/assets/js/26.9b876ccf.js" defer></script>
</body>
</html>

@ -7,7 +7,7 @@
<meta name="generator" content="VuePress 1.9.7">
<meta name="description" content="">
<meta property="article:modified_time" content="2022-05-14T18:41:58.000Z">
<meta property="article:modified_time" content="2022-05-14T18:44:34.000Z">
<meta property="og:site_name" content="Learn Wgpu">
<meta property="og:title" content="Update to 0.12!">
<meta property="og:type" content="website">
@ -19,15 +19,15 @@
<meta name="twitter:data1" content="Benjamin Hansen">
<meta name="twitter:creator" content="https://twitter.com/sotrh760">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.cbd0d45d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/44.9a3ae8d9.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.576af98e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.e74dc361.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.69a9ea7e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.e284d6f3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/22.ae08829d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.9d07c0eb.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.6982a462.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.32361a22.js"><link rel="prefetch" href="/learn-wgpu/assets/js/26.9b876ccf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.9fe49a03.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.c212a80a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.b61fde25.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.25585faa.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.acd07b95.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.a51ea3c5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.41e9897b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.2d445b19.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.9b4d649e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.e0f57de3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.155fa5a2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.d3e6544b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.2bed1539.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.4d927984.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.bcb97198.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.c87751e7.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/44.5be8c6e7.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.e0b6e526.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.c8349761.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.2469c630.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.3972e66e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/22.ae08829d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.d649043b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.5b5cf4ba.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.a63ee8c2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/26.9b876ccf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.064a85a6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.fb933617.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.2e0b419b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.09885c9c.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.19c7b8dd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.9a017b74.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.695eaab4.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.1ae459af.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.bb1a0e6a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.34968282.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.58137882.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.c80ee966.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.740cfe90.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.2e48975d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.ec58808b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="stylesheet" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css">
</head>
<body>
<div id="app" data-server-rendered="true"><div class="theme-container"><header class="navbar"><div class="inner"><div class="sidebar-button"><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" viewBox="0 0 448 512" class="icon"><path fill="currentColor" d="M436 124H12c-6.627 0-12-5.373-12-12V80c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12z"></path></svg></div> <a href="/learn-wgpu/" class="home-link router-link-active"><!----> <span class="site-name">Learn Wgpu</span></a> <div class="links"><!----> <div class="search-box"><input aria-label="Search" autocomplete="off" spellcheck="false" value=""> <!----></div></div></div></header> <div class="sidebar-mask"></div> <div class="docs-layout"><aside class="sidebar"><!----> <ul class="sidebar-links"><li><a href="/learn-wgpu/" aria-current="page" class="sidebar-link">Introduction</a></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Beginner</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/beginner/tutorial1-window/" class="sidebar-link">Dependencies and the window</a></li><li><a href="/learn-wgpu/beginner/tutorial2-surface/" class="sidebar-link">The Surface</a></li><li><a href="/learn-wgpu/beginner/tutorial3-pipeline/" class="sidebar-link">The Pipeline</a></li><li><a href="/learn-wgpu/beginner/tutorial4-buffer/" class="sidebar-link">Buffers and Indices</a></li><li><a href="/learn-wgpu/beginner/tutorial5-textures/" class="sidebar-link">Textures and bind groups</a></li><li><a href="/learn-wgpu/beginner/tutorial6-uniforms/" class="sidebar-link">Uniform buffers and a 3d camera</a></li><li><a href="/learn-wgpu/beginner/tutorial7-instancing/" class="sidebar-link">Instancing</a></li><li><a href="/learn-wgpu/beginner/tutorial8-depth/" class="sidebar-link">The Depth Buffer</a></li><li><a href="/learn-wgpu/beginner/tutorial9-models/" class="sidebar-link">Model Loading</a></li></ul></section></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Intermediate</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/intermediate/tutorial10-lighting/" class="sidebar-link">Working with Lights</a></li><li><a href="/learn-wgpu/intermediate/tutorial11-normals/" class="sidebar-link">Normal Mapping</a></li><li><a href="/learn-wgpu/intermediate/tutorial12-camera/" class="sidebar-link">A Better Camera</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>Showcase</span> <span class="arrow right"></span></p> <!----></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading open"><span>News</span> <span class="arrow down"></span></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/news/0.12/" aria-current="page" class="active sidebar-link">Update to 0.12!</a><ul class="sidebar-sub-headers"><li class="sidebar-sub-header"><a href="/learn-wgpu/news/0.12/#multi-view-added" class="sidebar-link">Multi view added</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/news/0.12/#no-more-block-attribute" class="sidebar-link">No more block attribute</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/news/0.12/#more-validation" class="sidebar-link">More validation</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/news/0.12/#misc" class="sidebar-link">Misc</a></li></ul></li><li><a href="/learn-wgpu/news/pre-0.12/" class="sidebar-link">News (Pre 0.12)</a></li></ul></section></li></ul> </aside> <main class="page"> <div class="theme-default-content content__default"><h1 id="update-to-0-12"><a href="#update-to-0-12" class="header-anchor">#</a> Update to 0.12!</h1> <p>There's not a ton of changes in this release, so the migration
<div id="app" data-server-rendered="true"><div class="theme-container"><header class="navbar"><div class="inner"><div class="sidebar-button"><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" viewBox="0 0 448 512" class="icon"><path fill="currentColor" d="M436 124H12c-6.627 0-12-5.373-12-12V80c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12z"></path></svg></div> <a href="/learn-wgpu/" class="home-link router-link-active"><!----> <span class="site-name">Learn Wgpu</span></a> <div class="links"><!----> <div class="search-box"><input aria-label="Search" autocomplete="off" spellcheck="false" value=""> <!----></div></div></div></header> <div class="sidebar-mask"></div> <div class="docs-layout"><aside class="sidebar"><!----> <ul class="sidebar-links"><li><a href="/learn-wgpu/" aria-current="page" class="sidebar-link">Introduction</a></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Beginner</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/beginner/tutorial1-window/" class="sidebar-link">Dependencies and the window</a></li><li><a href="/learn-wgpu/beginner/tutorial2-surface/" class="sidebar-link">The Surface</a></li><li><a href="/learn-wgpu/beginner/tutorial3-pipeline/" class="sidebar-link">The Pipeline</a></li><li><a href="/learn-wgpu/beginner/tutorial4-buffer/" class="sidebar-link">Buffers and Indices</a></li><li><a href="/learn-wgpu/beginner/tutorial5-textures/" class="sidebar-link">Textures and bind groups</a></li><li><a href="/learn-wgpu/beginner/tutorial6-uniforms/" class="sidebar-link">Uniform buffers and a 3d camera</a></li><li><a href="/learn-wgpu/beginner/tutorial7-instancing/" class="sidebar-link">Instancing</a></li><li><a href="/learn-wgpu/beginner/tutorial8-depth/" class="sidebar-link">The Depth Buffer</a></li><li><a href="/learn-wgpu/beginner/tutorial9-models/" class="sidebar-link">Model Loading</a></li></ul></section></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Intermediate</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/intermediate/tutorial10-lighting/" class="sidebar-link">Working with Lights</a></li><li><a href="/learn-wgpu/intermediate/tutorial11-normals/" class="sidebar-link">Normal Mapping</a></li><li><a href="/learn-wgpu/intermediate/tutorial12-camera/" class="sidebar-link">A Better Camera</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>Showcase</span> <span class="arrow right"></span></p> <!----></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading open"><span>News</span> <span class="arrow down"></span></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/news/0.12/" aria-current="page" class="active sidebar-link">Update to 0.12!</a><ul class="sidebar-sub-headers"><li class="sidebar-sub-header"><a href="/learn-wgpu/news/0.12/#multi-view-added" class="sidebar-link">Multi view added</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/news/0.12/#no-more-block-attribute" class="sidebar-link">No more block attribute</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/news/0.12/#more-validation" class="sidebar-link">More validation</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/news/0.12/#misc" class="sidebar-link">Misc</a></li></ul></li><li><a href="/learn-wgpu/news/pre-0.12/" class="sidebar-link">News (Pre 0.12)</a></li></ul></section></li></ul> </aside> <main class="page"> <div class="theme-default-content content__default"><h1 id="update-to-0-12"><a href="#update-to-0-12" class="header-anchor">#</a> Update to 0.12!</h1> <p>There are not a ton of changes in this release, so the migration
wasn't too painful.</p> <h2 id="multi-view-added"><a href="#multi-view-added" class="header-anchor">#</a> Multi view added</h2> <p>The <code>RenderPipelineDescriptor</code> now includes a <code>multiview</code> field that
indicates the number of array textures if array textures are used
as render attachments.</p> <h2 id="no-more-block-attribute"><a href="#no-more-block-attribute" class="header-anchor">#</a> No more block attribute</h2> <p>The WGSL spec has changed and the <code>block</code> attribute is no longer a thing.
This means that structs in WGSL no longer need to be anotated to be used
This means that structs in WGSL no longer need to be annotated to be used
as uniform input. For example:</p> <div class="language-wgsl extra-class"><pre class="language-text"><code>[[block]]
struct Camera {
view_pos: vec4&lt;f32&gt;;
@ -61,7 +61,7 @@ I needed to do was add a padding field:</p> <div class="language-rust extra-clas
breaks glob imports (aka. <code>use anyhow::*</code>), I had to switch qualified
imports and uses (ie. <code>anyhow::Result</code>). This was mostly an issue on my
build scripts for some of the showcase examples.</p> <p>The main tutorial examples weren't affected, and the changes are minor, so
if your curious feel free to look at the repo.</p></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:41:58 PM</span></div></footer> <div class="page-nav"><p class="inner"><span class="prev">
if you're curious feel free to look at the repo.</p></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:44:34 PM</span></div></footer> <div class="page-nav"><p class="inner"><span class="prev">
<a href="/learn-wgpu/showcase/alignment/" class="prev">
Memory Layout in WGSL
@ -70,6 +70,6 @@ if your curious feel free to look at the repo.</p></div> <footer class="page-edi
</a>
</span></p></div> </main></div></div><div class="global-ui"><!----></div></div>
<script src="/learn-wgpu/assets/js/app.cbd0d45d.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/44.9a3ae8d9.js" defer></script>
<script src="/learn-wgpu/assets/js/app.c87751e7.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/44.5be8c6e7.js" defer></script>
</body>
</html>

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

@ -7,7 +7,7 @@
<meta name="generator" content="VuePress 1.9.7">
<meta name="description" content="">
<meta property="article:modified_time" content="2022-05-14T18:41:58.000Z">
<meta property="article:modified_time" content="2022-05-14T18:44:34.000Z">
<meta property="og:site_name" content="Learn Wgpu">
<meta property="og:title" content="Compute Example: Tangents and Bitangents">
<meta property="og:type" content="website">
@ -19,11 +19,11 @@
<meta name="twitter:data1" content="Benjamin Hansen">
<meta name="twitter:creator" content="https://twitter.com/sotrh760">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.cbd0d45d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/24.6982a462.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/26.9b876ccf.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.576af98e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.e74dc361.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.69a9ea7e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.e284d6f3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/22.ae08829d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.9d07c0eb.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.32361a22.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.9fe49a03.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.c212a80a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.b61fde25.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.25585faa.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.acd07b95.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.a51ea3c5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.41e9897b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.2d445b19.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.9b4d649e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.e0f57de3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.155fa5a2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.d3e6544b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.2bed1539.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.9a3ae8d9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.4d927984.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.bcb97198.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.c87751e7.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/24.5b5cf4ba.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/26.9b876ccf.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.e0b6e526.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.c8349761.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.2469c630.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.3972e66e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/22.ae08829d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.d649043b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.a63ee8c2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.064a85a6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.fb933617.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.2e0b419b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.09885c9c.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.19c7b8dd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.9a017b74.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.695eaab4.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.1ae459af.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.bb1a0e6a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.34968282.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.58137882.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.c80ee966.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.740cfe90.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.5be8c6e7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.2e48975d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.ec58808b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="stylesheet" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css">
</head>
<body>
<div id="app" data-server-rendered="true"><div class="theme-container"><header class="navbar"><div class="inner"><div class="sidebar-button"><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" viewBox="0 0 448 512" class="icon"><path fill="currentColor" d="M436 124H12c-6.627 0-12-5.373-12-12V80c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12z"></path></svg></div> <a href="/learn-wgpu/" class="home-link router-link-active"><!----> <span class="site-name">Learn Wgpu</span></a> <div class="links"><!----> <div class="search-box"><input aria-label="Search" autocomplete="off" spellcheck="false" value=""> <!----></div></div></div></header> <div class="sidebar-mask"></div> <div class="docs-layout"><aside class="sidebar"><!----> <ul class="sidebar-links"><li><a href="/learn-wgpu/" aria-current="page" class="sidebar-link">Introduction</a></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Beginner</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/beginner/tutorial1-window/" class="sidebar-link">Dependencies and the window</a></li><li><a href="/learn-wgpu/beginner/tutorial2-surface/" class="sidebar-link">The Surface</a></li><li><a href="/learn-wgpu/beginner/tutorial3-pipeline/" class="sidebar-link">The Pipeline</a></li><li><a href="/learn-wgpu/beginner/tutorial4-buffer/" class="sidebar-link">Buffers and Indices</a></li><li><a href="/learn-wgpu/beginner/tutorial5-textures/" class="sidebar-link">Textures and bind groups</a></li><li><a href="/learn-wgpu/beginner/tutorial6-uniforms/" class="sidebar-link">Uniform buffers and a 3d camera</a></li><li><a href="/learn-wgpu/beginner/tutorial7-instancing/" class="sidebar-link">Instancing</a></li><li><a href="/learn-wgpu/beginner/tutorial8-depth/" class="sidebar-link">The Depth Buffer</a></li><li><a href="/learn-wgpu/beginner/tutorial9-models/" class="sidebar-link">Model Loading</a></li></ul></section></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Intermediate</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/intermediate/tutorial10-lighting/" class="sidebar-link">Working with Lights</a></li><li><a href="/learn-wgpu/intermediate/tutorial11-normals/" class="sidebar-link">Normal Mapping</a></li><li><a href="/learn-wgpu/intermediate/tutorial12-camera/" class="sidebar-link">A Better Camera</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading open"><span>Showcase</span> <span class="arrow down"></span></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/showcase/" aria-current="page" class="sidebar-link">Foreword</a></li><li><a href="/learn-wgpu/showcase/windowless/" class="sidebar-link">Wgpu without a window</a></li><li><a href="/learn-wgpu/showcase/gifs/" class="sidebar-link">Creating gifs</a></li><li><a href="/learn-wgpu/showcase/pong/" class="sidebar-link">Pong</a></li><li><a href="/learn-wgpu/showcase/compute/" aria-current="page" class="active sidebar-link">Compute Example: Tangents and Bitangents</a><ul class="sidebar-sub-headers"><li class="sidebar-sub-header"><a href="/learn-wgpu/showcase/compute/#possible-improvements" class="sidebar-link">Possible Improvements</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/showcase/compute/#results" class="sidebar-link">Results</a></li></ul></li><li><a href="/learn-wgpu/showcase/alignment/" class="sidebar-link">Memory Layout in WGSL</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>News</span> <span class="arrow right"></span></p> <!----></section></li></ul> </aside> <main class="page"> <div class="theme-default-content content__default"><h1 id="compute-example-tangents-and-bitangents"><a href="#compute-example-tangents-and-bitangents" class="header-anchor">#</a> Compute Example: Tangents and Bitangents</h1> <p>This proved more difficult than I anticipated. The first problem I encountered was some vertex data corruption due to the shader reading my vertex data incorrectly. I was using my <code>ModelVertex</code> struct I used in the <a href="/learn-wgpu/intermediate/tutorial11-normals/">normal mapping tutorial</a>.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token attribute attr-name">#[repr(C)]</span>
<div id="app" data-server-rendered="true"><div class="theme-container"><header class="navbar"><div class="inner"><div class="sidebar-button"><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" viewBox="0 0 448 512" class="icon"><path fill="currentColor" d="M436 124H12c-6.627 0-12-5.373-12-12V80c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12z"></path></svg></div> <a href="/learn-wgpu/" class="home-link router-link-active"><!----> <span class="site-name">Learn Wgpu</span></a> <div class="links"><!----> <div class="search-box"><input aria-label="Search" autocomplete="off" spellcheck="false" value=""> <!----></div></div></div></header> <div class="sidebar-mask"></div> <div class="docs-layout"><aside class="sidebar"><!----> <ul class="sidebar-links"><li><a href="/learn-wgpu/" aria-current="page" class="sidebar-link">Introduction</a></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Beginner</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/beginner/tutorial1-window/" class="sidebar-link">Dependencies and the window</a></li><li><a href="/learn-wgpu/beginner/tutorial2-surface/" class="sidebar-link">The Surface</a></li><li><a href="/learn-wgpu/beginner/tutorial3-pipeline/" class="sidebar-link">The Pipeline</a></li><li><a href="/learn-wgpu/beginner/tutorial4-buffer/" class="sidebar-link">Buffers and Indices</a></li><li><a href="/learn-wgpu/beginner/tutorial5-textures/" class="sidebar-link">Textures and bind groups</a></li><li><a href="/learn-wgpu/beginner/tutorial6-uniforms/" class="sidebar-link">Uniform buffers and a 3d camera</a></li><li><a href="/learn-wgpu/beginner/tutorial7-instancing/" class="sidebar-link">Instancing</a></li><li><a href="/learn-wgpu/beginner/tutorial8-depth/" class="sidebar-link">The Depth Buffer</a></li><li><a href="/learn-wgpu/beginner/tutorial9-models/" class="sidebar-link">Model Loading</a></li></ul></section></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Intermediate</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/intermediate/tutorial10-lighting/" class="sidebar-link">Working with Lights</a></li><li><a href="/learn-wgpu/intermediate/tutorial11-normals/" class="sidebar-link">Normal Mapping</a></li><li><a href="/learn-wgpu/intermediate/tutorial12-camera/" class="sidebar-link">A Better Camera</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading open"><span>Showcase</span> <span class="arrow down"></span></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/showcase/" aria-current="page" class="sidebar-link">Foreword</a></li><li><a href="/learn-wgpu/showcase/windowless/" class="sidebar-link">Wgpu without a window</a></li><li><a href="/learn-wgpu/showcase/gifs/" class="sidebar-link">Creating gifs</a></li><li><a href="/learn-wgpu/showcase/pong/" class="sidebar-link">Pong</a></li><li><a href="/learn-wgpu/showcase/compute/" aria-current="page" class="active sidebar-link">Compute Example: Tangents and Bitangents</a><ul class="sidebar-sub-headers"><li class="sidebar-sub-header"><a href="/learn-wgpu/showcase/compute/#possible-improvements" class="sidebar-link">Possible Improvements</a></li><li class="sidebar-sub-header"><a href="/learn-wgpu/showcase/compute/#results" class="sidebar-link">Results</a></li></ul></li><li><a href="/learn-wgpu/showcase/alignment/" class="sidebar-link">Memory Layout in WGSL</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>News</span> <span class="arrow right"></span></p> <!----></section></li></ul> </aside> <main class="page"> <div class="theme-default-content content__default"><h1 id="compute-example-tangents-and-bitangents"><a href="#compute-example-tangents-and-bitangents" class="header-anchor">#</a> Compute Example: Tangents and Bitangents</h1> <p>This proved more difficult than I anticipated. The first problem I encountered was some vertex data corruption due to the shader reading my vertex data incorrectly. I was using the <code>ModelVertex</code> struct I used in the <a href="/learn-wgpu/intermediate/tutorial11-normals/">normal mapping tutorial</a>.</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token attribute attr-name">#[repr(C)]</span>
<span class="token attribute attr-name">#[derive(Copy, Clone, Debug, bytemuck::Pod, bytemuck::Zeroable)]</span>
<span class="token keyword">pub</span> <span class="token keyword">struct</span> <span class="token type-definition class-name">ModelVertex</span> <span class="token punctuation">{</span>
position<span class="token punctuation">:</span> <span class="token punctuation">[</span><span class="token keyword">f32</span><span class="token punctuation">;</span> <span class="token number">3</span><span class="token punctuation">]</span><span class="token punctuation">,</span>
@ -39,19 +39,19 @@
<span class="token keyword">vec3</span> tangent<span class="token punctuation">;</span>
<span class="token keyword">vec3</span> bitangent<span class="token punctuation">;</span>
<span class="token punctuation">}</span><span class="token punctuation">;</span>
</code></pre></div><p>At first glance, this seems just fine, but OpenGL experts would likely see a problem with the structure. Our fields aren't aligned properly to support the <code>std430</code> alignment that storage buffers require.. I won't get into detail but you can check out the <a href="../alignment">alignment showcase</a> if you want to know more. To summarize, the <code>vec2</code> for the <code>tex_coords</code> was messing up the byte alignment, corrupting the vertex data resulting in the following:</p> <p><img src="/learn-wgpu/assets/img/corruption.675b1eca.png" alt="./corruption.png"></p> <p>I could have fixed this by adding a padding field after <code>tex_coords</code> on the Rust side, but that would require modifying the <code>VertexBufferLayout</code>. I ended up solving this problem by using the components of the vectors directly and resulted with a struct like this:</p> <div class="language-glsl extra-class"><pre class="language-glsl"><code><span class="token keyword">struct</span> <span class="token class-name">ModelVertex</span> <span class="token punctuation">{</span>
</code></pre></div><p>At first glance, this seems just fine, but OpenGL experts would likely see a problem with the structure. Our fields aren't aligned properly to support the <code>std430</code> alignment that storage buffers require. I won't get into detail but you can check out the <a href="../alignment">alignment showcase</a> if you want to know more. To summarize, the <code>vec2</code> for the <code>tex_coords</code> was messing up the byte alignment, corrupting the vertex data resulting in the following:</p> <p><img src="/learn-wgpu/assets/img/corruption.675b1eca.png" alt="./corruption.png"></p> <p>I could have fixed this by adding a padding field after <code>tex_coords</code> on the Rust side, but that would require modifying the <code>VertexBufferLayout</code>. I ended up solving this problem by using the components of the vectors directly which resulted in a struct like this:</p> <div class="language-glsl extra-class"><pre class="language-glsl"><code><span class="token keyword">struct</span> <span class="token class-name">ModelVertex</span> <span class="token punctuation">{</span>
<span class="token keyword">float</span> x<span class="token punctuation">;</span> <span class="token keyword">float</span> y<span class="token punctuation">;</span> <span class="token keyword">float</span> z<span class="token punctuation">;</span>
<span class="token keyword">float</span> uv<span class="token punctuation">;</span> <span class="token keyword">float</span> uw<span class="token punctuation">;</span>
<span class="token keyword">float</span> nx<span class="token punctuation">;</span> <span class="token keyword">float</span> ny<span class="token punctuation">;</span> <span class="token keyword">float</span> nz<span class="token punctuation">;</span>
<span class="token keyword">float</span> tx<span class="token punctuation">;</span> <span class="token keyword">float</span> ty<span class="token punctuation">;</span> <span class="token keyword">float</span> tz<span class="token punctuation">;</span>
<span class="token keyword">float</span> bx<span class="token punctuation">;</span> <span class="token keyword">float</span> by<span class="token punctuation">;</span> <span class="token keyword">float</span> bz<span class="token punctuation">;</span>
<span class="token punctuation">}</span><span class="token punctuation">;</span>
</code></pre></div><p>Since <code>std430</code> will use the alignment of the largest element of the struct, using all floats means the struct will be aligned to 4 bytes. This is alignment matches what <code>ModelVertex</code> uses in Rust. This was kind of a pain to work with, but it fixed the corruption issue.</p> <p>The second problem required me to rethink how I was computing the tangent and bitangent. The previous algorithm I was using only computed the tangent and bitangent for each triangle and set all the vertices in that triangle to use the same tangent and bitangent. While this is fine in a single threaded context, the code breaks down when trying to compute the triangles in parallel. The reason is that multiple triangles can share the same vertices. This means that when we go to save the resulting tangents, we inevitably end up trying to write to the same vertex from multiple different threads which is a big no no. You can see the issue with this method below:</p> <p><img src="/learn-wgpu/assets/img/black_triangles.df338b97.png" alt="./black_triangles.png"></p> <p>Those black triangles were the result of multiple GPU threads trying to modify the same vertices. Looking at the data in Render Doc I could see that the tangents and bitangents were garbage numbers such as <code>NaN</code>.</p> <p><img src="/learn-wgpu/assets/img/render_doc_output.e0c8b298.png" alt="./render_doc_output.png"></p> <p>While on the CPU we could introduce a synchronization primitive such as a <code>Mutex</code> to fix this issue, AFAIK there isn't really such a thing on the GPU. Instead I decided to swap my code to work with each vertex individually. There are some hurdles with that, but those will be easier to explain in code. Let's start with the <code>main</code> function.</p> <div class="language-glsl extra-class"><pre class="language-glsl"><code><span class="token keyword">void</span> <span class="token function">main</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
</code></pre></div><p>Since <code>std430</code> will use the alignment of the largest element of the struct, using all floats means the struct will be aligned to 4 bytes. This is alignment matches what <code>ModelVertex</code> uses in Rust. This was kind of a pain to work with, but it fixed the corruption issue.</p> <p>The second problem required me to rethink how I was computing the tangent and bitangent. The previous algorithm I was using only computed the tangent and bitangent for each triangle and set all the vertices in that triangle to use the same tangent and bitangent. While this is fine in a single-threaded context, the code breaks down when trying to compute the triangles in parallel. The reason is that multiple triangles can share the same vertices. This means that when we go to save the resulting tangents, we inevitably end up trying to write to the same vertex from multiple different threads which is a big no no. You can see the issue with this method below:</p> <p><img src="/learn-wgpu/assets/img/black_triangles.df338b97.png" alt="./black_triangles.png"></p> <p>Those black triangles were the result of multiple GPU threads trying to modify the same vertices. Looking at the data in Render Doc I could see that the tangents and bitangents were garbage numbers such as <code>NaN</code>.</p> <p><img src="/learn-wgpu/assets/img/render_doc_output.e0c8b298.png" alt="./render_doc_output.png"></p> <p>While on the CPU we could introduce a synchronization primitive such as a <code>Mutex</code> to fix this issue, AFAIK there isn't really such a thing on the GPU. Instead, I decided to swap my code to work with each vertex individually. There are some hurdles with that, but those will be easier to explain in code. Let's start with the <code>main</code> function.</p> <div class="language-glsl extra-class"><pre class="language-glsl"><code><span class="token keyword">void</span> <span class="token function">main</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token punctuation">{</span>
<span class="token keyword">uint</span> vertexIndex <span class="token operator">=</span> gl_GlobalInvocationID<span class="token punctuation">.</span>x<span class="token punctuation">;</span>
ModelVertex result <span class="token operator">=</span> <span class="token function">calcTangentBitangent</span><span class="token punctuation">(</span>vertexIndex<span class="token punctuation">)</span><span class="token punctuation">;</span>
dstVertices<span class="token punctuation">[</span>vertexIndex<span class="token punctuation">]</span> <span class="token operator">=</span> result<span class="token punctuation">;</span>
<span class="token punctuation">}</span>
</code></pre></div><p>We use the <code>gl_GlobalInvocationID.x</code> to get the index of the vertex we want to compute the tangents for. I opted to put the actual calculation into it's own method. Let's take a look at that.</p> <div class="language-glsl extra-class"><pre class="language-glsl"><code>ModelVertex <span class="token function">calcTangentBitangent</span><span class="token punctuation">(</span><span class="token keyword">uint</span> vertexIndex<span class="token punctuation">)</span> <span class="token punctuation">{</span>
</code></pre></div><p>We use the <code>gl_GlobalInvocationID.x</code> to get the index of the vertex we want to compute the tangents for. I opted to put the actual calculation into its own method. Let's take a look at that.</p> <div class="language-glsl extra-class"><pre class="language-glsl"><code>ModelVertex <span class="token function">calcTangentBitangent</span><span class="token punctuation">(</span><span class="token keyword">uint</span> vertexIndex<span class="token punctuation">)</span> <span class="token punctuation">{</span>
ModelVertex v <span class="token operator">=</span> srcVertices<span class="token punctuation">[</span>vertexIndex<span class="token punctuation">]</span><span class="token punctuation">;</span>
<span class="token keyword">vec3</span> tangent <span class="token operator">=</span> <span class="token keyword">vec3</span><span class="token punctuation">(</span><span class="token number">0</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
@ -112,7 +112,7 @@
<span class="token keyword">return</span> v<span class="token punctuation">;</span>
<span class="token punctuation">}</span>
</code></pre></div><h2 id="possible-improvements"><a href="#possible-improvements" class="header-anchor">#</a> Possible Improvements</h2> <p>Looping over every triangle for every vertex is likely raising some red flags for some of you. In a single threaded context, this algorithm would end up being O(N*M). As we are utilizing the high number of threads available to our GPU, this is less of an issue, but it still means our GPU is burning more cycles than it needs to.</p> <p>One way I came up with to possibly improve performance is to store the index of each triangle in a hash map like structure with the vertex index as keys. Here's some pseudo code:</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">for</span> t <span class="token keyword">in</span> <span class="token number">0</span><span class="token punctuation">..</span>indices<span class="token punctuation">.</span><span class="token function">len</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">/</span> <span class="token number">3</span> <span class="token punctuation">{</span>
</code></pre></div><h2 id="possible-improvements"><a href="#possible-improvements" class="header-anchor">#</a> Possible Improvements</h2> <p>Looping over every triangle for every vertex is likely raising some red flags for some of you. In a single-threaded context, this algorithm would end up being O(N*M). As we are utilizing the high number of threads available to our GPU, this is less of an issue, but it still means our GPU is burning more cycles than it needs to.</p> <p>One way I came up with to possibly improve performance is to store the index of each triangle in a hash map like structure with the vertex index as keys. Here's some pseudo code:</p> <div class="language-rust extra-class"><pre class="language-rust"><code><span class="token keyword">for</span> t <span class="token keyword">in</span> <span class="token number">0</span><span class="token punctuation">..</span>indices<span class="token punctuation">.</span><span class="token function">len</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">/</span> <span class="token number">3</span> <span class="token punctuation">{</span>
triangle_map<span class="token punctuation">[</span>indices<span class="token punctuation">[</span>t <span class="token operator">*</span> <span class="token number">3</span><span class="token punctuation">]</span><span class="token punctuation">]</span><span class="token punctuation">.</span><span class="token function">push</span><span class="token punctuation">(</span>t<span class="token punctuation">)</span><span class="token punctuation">;</span>
triangle_map<span class="token punctuation">.</span><span class="token function">push</span><span class="token punctuation">(</span><span class="token punctuation">(</span>indices<span class="token punctuation">[</span>t <span class="token operator">*</span> <span class="token number">3</span> <span class="token operator">+</span> <span class="token number">1</span><span class="token punctuation">]</span><span class="token punctuation">,</span> t<span class="token punctuation">)</span><span class="token punctuation">;</span>
triangle_map<span class="token punctuation">.</span><span class="token function">push</span><span class="token punctuation">(</span><span class="token punctuation">(</span>indices<span class="token punctuation">[</span>t <span class="token operator">*</span> <span class="token number">3</span> <span class="token operator">+</span> <span class="token number">2</span><span class="token punctuation">]</span><span class="token punctuation">,</span> t<span class="token punctuation">)</span><span class="token punctuation">;</span>
@ -124,7 +124,7 @@
<span class="token punctuation">}</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
flat_triangle_map<span class="token punctuation">.</span><span class="token function">extend</span><span class="token punctuation">(</span>t_list<span class="token punctuation">)</span><span class="token punctuation">;</span>
<span class="token punctuation">}</span>
</code></pre></div><p>I ultimately decided against this method as it was more complicated, and I haven't had time to benchmark it to see if it's faster that the simple method.</p> <h2 id="results"><a href="#results" class="header-anchor">#</a> Results</h2> <p>The tangents and bitangents are now getting calculated correctly and on the GPU!</p> <p><img src="/learn-wgpu/assets/img/results.7918efc1.png" alt="./results.png"></p> <div class="auto-github-link"><a href="https://github.com/sotrh/learn-wgpu/tree/master/code/showcase/compute/" target="_blank" rel="noopener noreferrer">Check out the code!</a> <span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></div></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:41:58 PM</span></div></footer> <div class="page-nav"><p class="inner"><span class="prev">
</code></pre></div><p>I ultimately decided against this method as it was more complicated, and I haven't had time to benchmark it to see if it's faster than the simple method.</p> <h2 id="results"><a href="#results" class="header-anchor">#</a> Results</h2> <p>The tangents and bitangents are now getting calculated correctly and on the GPU!</p> <p><img src="/learn-wgpu/assets/img/results.7918efc1.png" alt="./results.png"></p> <div class="auto-github-link"><a href="https://github.com/sotrh/learn-wgpu/tree/master/code/showcase/compute/" target="_blank" rel="noopener noreferrer">Check out the code!</a> <span><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" x="0px" y="0px" viewBox="0 0 100 100" width="15" height="15" class="icon outbound"><path fill="currentColor" d="M18.8,85.1h56l0,0c2.2,0,4-1.8,4-4v-32h-8v28h-48v-48h28v-8h-32l0,0c-2.2,0-4,1.8-4,4v56C14.8,83.3,16.6,85.1,18.8,85.1z"></path> <polygon fill="currentColor" points="45.7,48.7 51.3,54.3 77.2,28.5 77.2,37.2 85.2,37.2 85.2,14.9 62.8,14.9 62.8,22.9 71.5,22.9"></polygon></svg> <span class="sr-only">(opens new window)</span></span></div></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:44:34 PM</span></div></footer> <div class="page-nav"><p class="inner"><span class="prev">
<a href="/learn-wgpu/showcase/pong/" class="prev">
Pong
@ -133,6 +133,6 @@
</a>
</span></p></div> </main></div></div><div class="global-ui"><!----></div></div>
<script src="/learn-wgpu/assets/js/app.cbd0d45d.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/24.6982a462.js" defer></script><script src="/learn-wgpu/assets/js/26.9b876ccf.js" defer></script>
<script src="/learn-wgpu/assets/js/app.c87751e7.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/24.5b5cf4ba.js" defer></script><script src="/learn-wgpu/assets/js/26.9b876ccf.js" defer></script>
</body>
</html>

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

@ -7,7 +7,7 @@
<meta name="generator" content="VuePress 1.9.7">
<meta name="description" content="">
<meta property="article:modified_time" content="2022-05-14T18:41:58.000Z">
<meta property="article:modified_time" content="2022-05-14T18:44:34.000Z">
<meta property="og:site_name" content="Learn Wgpu">
<meta property="og:title" content="Foreword">
<meta property="og:type" content="website">
@ -19,11 +19,11 @@
<meta name="twitter:data1" content="Benjamin Hansen">
<meta name="twitter:creator" content="https://twitter.com/sotrh760">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.cbd0d45d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/45.4d927984.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.576af98e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.e74dc361.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.69a9ea7e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.e284d6f3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/22.ae08829d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.9d07c0eb.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.6982a462.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.32361a22.js"><link rel="prefetch" href="/learn-wgpu/assets/js/26.9b876ccf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.9fe49a03.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.c212a80a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.b61fde25.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.25585faa.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.acd07b95.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.a51ea3c5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.41e9897b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.2d445b19.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.9b4d649e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.e0f57de3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.155fa5a2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.d3e6544b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.2bed1539.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.9a3ae8d9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.bcb97198.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.c87751e7.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/45.2e48975d.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.e0b6e526.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.c8349761.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.2469c630.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.3972e66e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/22.ae08829d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.d649043b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.5b5cf4ba.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.a63ee8c2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/26.9b876ccf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.064a85a6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.fb933617.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.2e0b419b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.09885c9c.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.19c7b8dd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.9a017b74.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.695eaab4.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.1ae459af.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.bb1a0e6a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.34968282.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.58137882.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.c80ee966.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.740cfe90.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.5be8c6e7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.ec58808b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/47.04026fdc.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="stylesheet" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css">
</head>
<body>
<div id="app" data-server-rendered="true"><div class="theme-container"><header class="navbar"><div class="inner"><div class="sidebar-button"><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" viewBox="0 0 448 512" class="icon"><path fill="currentColor" d="M436 124H12c-6.627 0-12-5.373-12-12V80c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12z"></path></svg></div> <a href="/learn-wgpu/" class="home-link router-link-active"><!----> <span class="site-name">Learn Wgpu</span></a> <div class="links"><!----> <div class="search-box"><input aria-label="Search" autocomplete="off" spellcheck="false" value=""> <!----></div></div></div></header> <div class="sidebar-mask"></div> <div class="docs-layout"><aside class="sidebar"><!----> <ul class="sidebar-links"><li><a href="/learn-wgpu/" aria-current="page" class="sidebar-link">Introduction</a></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Beginner</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/beginner/tutorial1-window/" class="sidebar-link">Dependencies and the window</a></li><li><a href="/learn-wgpu/beginner/tutorial2-surface/" class="sidebar-link">The Surface</a></li><li><a href="/learn-wgpu/beginner/tutorial3-pipeline/" class="sidebar-link">The Pipeline</a></li><li><a href="/learn-wgpu/beginner/tutorial4-buffer/" class="sidebar-link">Buffers and Indices</a></li><li><a href="/learn-wgpu/beginner/tutorial5-textures/" class="sidebar-link">Textures and bind groups</a></li><li><a href="/learn-wgpu/beginner/tutorial6-uniforms/" class="sidebar-link">Uniform buffers and a 3d camera</a></li><li><a href="/learn-wgpu/beginner/tutorial7-instancing/" class="sidebar-link">Instancing</a></li><li><a href="/learn-wgpu/beginner/tutorial8-depth/" class="sidebar-link">The Depth Buffer</a></li><li><a href="/learn-wgpu/beginner/tutorial9-models/" class="sidebar-link">Model Loading</a></li></ul></section></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Intermediate</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/intermediate/tutorial10-lighting/" class="sidebar-link">Working with Lights</a></li><li><a href="/learn-wgpu/intermediate/tutorial11-normals/" class="sidebar-link">Normal Mapping</a></li><li><a href="/learn-wgpu/intermediate/tutorial12-camera/" class="sidebar-link">A Better Camera</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading open"><span>Showcase</span> <span class="arrow down"></span></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/showcase/" aria-current="page" class="active sidebar-link">Foreword</a></li><li><a href="/learn-wgpu/showcase/windowless/" class="sidebar-link">Wgpu without a window</a></li><li><a href="/learn-wgpu/showcase/gifs/" class="sidebar-link">Creating gifs</a></li><li><a href="/learn-wgpu/showcase/pong/" class="sidebar-link">Pong</a></li><li><a href="/learn-wgpu/showcase/compute/" class="sidebar-link">Compute Example: Tangents and Bitangents</a></li><li><a href="/learn-wgpu/showcase/alignment/" class="sidebar-link">Memory Layout in WGSL</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>News</span> <span class="arrow right"></span></p> <!----></section></li></ul> </aside> <main class="page"> <div class="theme-default-content content__default"><h1 id="foreword"><a href="#foreword" class="header-anchor">#</a> Foreword</h1> <p>The articles in this section are not meant to be tutorials. They are showcases of the various things you can do with <code>wgpu</code>. I won't go over specifics of creating <code>wgpu</code> resources, as those will be covered elsewhere. The code for these examples is still available however, and will be accessible on Github.</p></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:41:58 PM</span></div></footer> <div class="page-nav"><p class="inner"><span class="prev">
<div id="app" data-server-rendered="true"><div class="theme-container"><header class="navbar"><div class="inner"><div class="sidebar-button"><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" viewBox="0 0 448 512" class="icon"><path fill="currentColor" d="M436 124H12c-6.627 0-12-5.373-12-12V80c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12z"></path></svg></div> <a href="/learn-wgpu/" class="home-link router-link-active"><!----> <span class="site-name">Learn Wgpu</span></a> <div class="links"><!----> <div class="search-box"><input aria-label="Search" autocomplete="off" spellcheck="false" value=""> <!----></div></div></div></header> <div class="sidebar-mask"></div> <div class="docs-layout"><aside class="sidebar"><!----> <ul class="sidebar-links"><li><a href="/learn-wgpu/" aria-current="page" class="sidebar-link">Introduction</a></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Beginner</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/beginner/tutorial1-window/" class="sidebar-link">Dependencies and the window</a></li><li><a href="/learn-wgpu/beginner/tutorial2-surface/" class="sidebar-link">The Surface</a></li><li><a href="/learn-wgpu/beginner/tutorial3-pipeline/" class="sidebar-link">The Pipeline</a></li><li><a href="/learn-wgpu/beginner/tutorial4-buffer/" class="sidebar-link">Buffers and Indices</a></li><li><a href="/learn-wgpu/beginner/tutorial5-textures/" class="sidebar-link">Textures and bind groups</a></li><li><a href="/learn-wgpu/beginner/tutorial6-uniforms/" class="sidebar-link">Uniform buffers and a 3d camera</a></li><li><a href="/learn-wgpu/beginner/tutorial7-instancing/" class="sidebar-link">Instancing</a></li><li><a href="/learn-wgpu/beginner/tutorial8-depth/" class="sidebar-link">The Depth Buffer</a></li><li><a href="/learn-wgpu/beginner/tutorial9-models/" class="sidebar-link">Model Loading</a></li></ul></section></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Intermediate</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/intermediate/tutorial10-lighting/" class="sidebar-link">Working with Lights</a></li><li><a href="/learn-wgpu/intermediate/tutorial11-normals/" class="sidebar-link">Normal Mapping</a></li><li><a href="/learn-wgpu/intermediate/tutorial12-camera/" class="sidebar-link">A Better Camera</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading open"><span>Showcase</span> <span class="arrow down"></span></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/showcase/" aria-current="page" class="active sidebar-link">Foreword</a></li><li><a href="/learn-wgpu/showcase/windowless/" class="sidebar-link">Wgpu without a window</a></li><li><a href="/learn-wgpu/showcase/gifs/" class="sidebar-link">Creating gifs</a></li><li><a href="/learn-wgpu/showcase/pong/" class="sidebar-link">Pong</a></li><li><a href="/learn-wgpu/showcase/compute/" class="sidebar-link">Compute Example: Tangents and Bitangents</a></li><li><a href="/learn-wgpu/showcase/alignment/" class="sidebar-link">Memory Layout in WGSL</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>News</span> <span class="arrow right"></span></p> <!----></section></li></ul> </aside> <main class="page"> <div class="theme-default-content content__default"><h1 id="foreword"><a href="#foreword" class="header-anchor">#</a> Foreword</h1> <p>The articles in this section are not meant to be tutorials. They are showcases of the various things you can do with <code>wgpu</code>. I won't go over the specifics of creating <code>wgpu</code> resources, as those will be covered elsewhere. The code for these examples is still available however and will be accessible on Github.</p></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:44:34 PM</span></div></footer> <div class="page-nav"><p class="inner"><span class="prev">
<a href="/learn-wgpu/intermediate/tutorial12-camera/" class="prev">
A Better Camera
@ -32,6 +32,6 @@
</a>
</span></p></div> </main></div></div><div class="global-ui"><!----></div></div>
<script src="/learn-wgpu/assets/js/app.cbd0d45d.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/45.4d927984.js" defer></script>
<script src="/learn-wgpu/assets/js/app.c87751e7.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/45.2e48975d.js" defer></script>
</body>
</html>

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

@ -7,7 +7,7 @@
<meta name="generator" content="VuePress 1.9.7">
<meta name="description" content="">
<meta property="article:modified_time" content="2022-05-14T18:41:58.000Z">
<meta property="article:modified_time" content="2022-05-14T18:44:34.000Z">
<meta property="og:site_name" content="Learn Wgpu">
<meta property="og:title" content="Coming Soon!">
<meta property="og:type" content="website">
@ -19,11 +19,11 @@
<meta name="twitter:data1" content="Benjamin Hansen">
<meta name="twitter:creator" content="https://twitter.com/sotrh760">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.cbd0d45d.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/47.04026fdc.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.576af98e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.e74dc361.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.69a9ea7e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.e284d6f3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/22.ae08829d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.9d07c0eb.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.6982a462.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.32361a22.js"><link rel="prefetch" href="/learn-wgpu/assets/js/26.9b876ccf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.9fe49a03.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.c212a80a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.b61fde25.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.25585faa.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.acd07b95.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.a51ea3c5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.41e9897b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.2d445b19.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.9b4d649e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.e0f57de3.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.155fa5a2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.d3e6544b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.2bed1539.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.9a3ae8d9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.4d927984.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.bcb97198.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="preload" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css" as="style"><link rel="preload" href="/learn-wgpu/assets/js/app.c87751e7.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/17.7a15a658.js" as="script"><link rel="preload" href="/learn-wgpu/assets/js/47.04026fdc.js" as="script"><link rel="prefetch" href="/learn-wgpu/assets/js/0.538bc9f6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/10.734d87da.js"><link rel="prefetch" href="/learn-wgpu/assets/js/11.41034c46.js"><link rel="prefetch" href="/learn-wgpu/assets/js/12.a4ce281e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/13.50f08c5b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/14.65041ede.js"><link rel="prefetch" href="/learn-wgpu/assets/js/15.e0b6e526.js"><link rel="prefetch" href="/learn-wgpu/assets/js/18.c8349761.js"><link rel="prefetch" href="/learn-wgpu/assets/js/19.2469c630.js"><link rel="prefetch" href="/learn-wgpu/assets/js/2.63a2358e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/20.295c559d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/21.3972e66e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/22.ae08829d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/23.d649043b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/24.5b5cf4ba.js"><link rel="prefetch" href="/learn-wgpu/assets/js/25.a63ee8c2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/26.9b876ccf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/27.a15a4eb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/28.064a85a6.js"><link rel="prefetch" href="/learn-wgpu/assets/js/29.fb933617.js"><link rel="prefetch" href="/learn-wgpu/assets/js/3.e2a96752.js"><link rel="prefetch" href="/learn-wgpu/assets/js/30.2e0b419b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/31.09885c9c.js"><link rel="prefetch" href="/learn-wgpu/assets/js/32.19c7b8dd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/33.9a017b74.js"><link rel="prefetch" href="/learn-wgpu/assets/js/34.695eaab4.js"><link rel="prefetch" href="/learn-wgpu/assets/js/35.1ae459af.js"><link rel="prefetch" href="/learn-wgpu/assets/js/36.bb1a0e6a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/37.34968282.js"><link rel="prefetch" href="/learn-wgpu/assets/js/38.58137882.js"><link rel="prefetch" href="/learn-wgpu/assets/js/39.1c5b4548.js"><link rel="prefetch" href="/learn-wgpu/assets/js/4.3c71cf5e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/40.014bbfe2.js"><link rel="prefetch" href="/learn-wgpu/assets/js/41.c80ee966.js"><link rel="prefetch" href="/learn-wgpu/assets/js/42.82e26587.js"><link rel="prefetch" href="/learn-wgpu/assets/js/43.740cfe90.js"><link rel="prefetch" href="/learn-wgpu/assets/js/44.5be8c6e7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/45.2e48975d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/46.ec58808b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/48.b66ca2b7.js"><link rel="prefetch" href="/learn-wgpu/assets/js/49.a7984c3a.js"><link rel="prefetch" href="/learn-wgpu/assets/js/5.bb0f1b1e.js"><link rel="prefetch" href="/learn-wgpu/assets/js/50.3f70b0bd.js"><link rel="prefetch" href="/learn-wgpu/assets/js/51.3dddd985.js"><link rel="prefetch" href="/learn-wgpu/assets/js/52.d210553f.js"><link rel="prefetch" href="/learn-wgpu/assets/js/53.d8a00d81.js"><link rel="prefetch" href="/learn-wgpu/assets/js/54.dee38fb1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/55.1203faa5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/56.b08678b1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/57.8a33a691.js"><link rel="prefetch" href="/learn-wgpu/assets/js/58.0b75502d.js"><link rel="prefetch" href="/learn-wgpu/assets/js/59.9487d6c9.js"><link rel="prefetch" href="/learn-wgpu/assets/js/6.11fcad8b.js"><link rel="prefetch" href="/learn-wgpu/assets/js/60.8f0e7ac5.js"><link rel="prefetch" href="/learn-wgpu/assets/js/61.a146a907.js"><link rel="prefetch" href="/learn-wgpu/assets/js/7.769bb1c1.js"><link rel="prefetch" href="/learn-wgpu/assets/js/8.dac8f9bf.js"><link rel="prefetch" href="/learn-wgpu/assets/js/9.1150195e.js">
<link rel="stylesheet" href="/learn-wgpu/assets/css/1.styles.a99e0ea1.css">
</head>
<body>
<div id="app" data-server-rendered="true"><div class="theme-container"><header class="navbar"><div class="inner"><div class="sidebar-button"><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" viewBox="0 0 448 512" class="icon"><path fill="currentColor" d="M436 124H12c-6.627 0-12-5.373-12-12V80c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12z"></path></svg></div> <a href="/learn-wgpu/" class="home-link router-link-active"><!----> <span class="site-name">Learn Wgpu</span></a> <div class="links"><!----> <div class="search-box"><input aria-label="Search" autocomplete="off" spellcheck="false" value=""> <!----></div></div></div></header> <div class="sidebar-mask"></div> <div class="docs-layout"><aside class="sidebar"><!----> <ul class="sidebar-links"><li><a href="/learn-wgpu/" aria-current="page" class="sidebar-link">Introduction</a></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Beginner</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/beginner/tutorial1-window/" class="sidebar-link">Dependencies and the window</a></li><li><a href="/learn-wgpu/beginner/tutorial2-surface/" class="sidebar-link">The Surface</a></li><li><a href="/learn-wgpu/beginner/tutorial3-pipeline/" class="sidebar-link">The Pipeline</a></li><li><a href="/learn-wgpu/beginner/tutorial4-buffer/" class="sidebar-link">Buffers and Indices</a></li><li><a href="/learn-wgpu/beginner/tutorial5-textures/" class="sidebar-link">Textures and bind groups</a></li><li><a href="/learn-wgpu/beginner/tutorial6-uniforms/" class="sidebar-link">Uniform buffers and a 3d camera</a></li><li><a href="/learn-wgpu/beginner/tutorial7-instancing/" class="sidebar-link">Instancing</a></li><li><a href="/learn-wgpu/beginner/tutorial8-depth/" class="sidebar-link">The Depth Buffer</a></li><li><a href="/learn-wgpu/beginner/tutorial9-models/" class="sidebar-link">Model Loading</a></li></ul></section></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Intermediate</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/intermediate/tutorial10-lighting/" class="sidebar-link">Working with Lights</a></li><li><a href="/learn-wgpu/intermediate/tutorial11-normals/" class="sidebar-link">Normal Mapping</a></li><li><a href="/learn-wgpu/intermediate/tutorial12-camera/" class="sidebar-link">A Better Camera</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>Showcase</span> <span class="arrow right"></span></p> <!----></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>News</span> <span class="arrow right"></span></p> <!----></section></li></ul> </aside> <main class="page"> <div class="theme-default-content content__default"><h1 id="coming-soon"><a href="#coming-soon" class="header-anchor">#</a> Coming Soon!</h1> <p>This section has not yet been completed.</p></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:41:58 PM</span></div></footer> <!----> </main></div></div><div class="global-ui"><!----></div></div>
<script src="/learn-wgpu/assets/js/app.cbd0d45d.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/47.04026fdc.js" defer></script>
<div id="app" data-server-rendered="true"><div class="theme-container"><header class="navbar"><div class="inner"><div class="sidebar-button"><svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" viewBox="0 0 448 512" class="icon"><path fill="currentColor" d="M436 124H12c-6.627 0-12-5.373-12-12V80c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12zm0 160H12c-6.627 0-12-5.373-12-12v-32c0-6.627 5.373-12 12-12h424c6.627 0 12 5.373 12 12v32c0 6.627-5.373 12-12 12z"></path></svg></div> <a href="/learn-wgpu/" class="home-link router-link-active"><!----> <span class="site-name">Learn Wgpu</span></a> <div class="links"><!----> <div class="search-box"><input aria-label="Search" autocomplete="off" spellcheck="false" value=""> <!----></div></div></div></header> <div class="sidebar-mask"></div> <div class="docs-layout"><aside class="sidebar"><!----> <ul class="sidebar-links"><li><a href="/learn-wgpu/" aria-current="page" class="sidebar-link">Introduction</a></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Beginner</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/beginner/tutorial1-window/" class="sidebar-link">Dependencies and the window</a></li><li><a href="/learn-wgpu/beginner/tutorial2-surface/" class="sidebar-link">The Surface</a></li><li><a href="/learn-wgpu/beginner/tutorial3-pipeline/" class="sidebar-link">The Pipeline</a></li><li><a href="/learn-wgpu/beginner/tutorial4-buffer/" class="sidebar-link">Buffers and Indices</a></li><li><a href="/learn-wgpu/beginner/tutorial5-textures/" class="sidebar-link">Textures and bind groups</a></li><li><a href="/learn-wgpu/beginner/tutorial6-uniforms/" class="sidebar-link">Uniform buffers and a 3d camera</a></li><li><a href="/learn-wgpu/beginner/tutorial7-instancing/" class="sidebar-link">Instancing</a></li><li><a href="/learn-wgpu/beginner/tutorial8-depth/" class="sidebar-link">The Depth Buffer</a></li><li><a href="/learn-wgpu/beginner/tutorial9-models/" class="sidebar-link">Model Loading</a></li></ul></section></li><li><section class="sidebar-group depth-0"><p class="sidebar-heading"><span>Intermediate</span> <!----></p> <ul class="sidebar-links sidebar-group-items"><li><a href="/learn-wgpu/intermediate/tutorial10-lighting/" class="sidebar-link">Working with Lights</a></li><li><a href="/learn-wgpu/intermediate/tutorial11-normals/" class="sidebar-link">Normal Mapping</a></li><li><a href="/learn-wgpu/intermediate/tutorial12-camera/" class="sidebar-link">A Better Camera</a></li></ul></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>Showcase</span> <span class="arrow right"></span></p> <!----></section></li><li><section class="sidebar-group collapsable depth-0"><p class="sidebar-heading"><span>News</span> <span class="arrow right"></span></p> <!----></section></li></ul> </aside> <main class="page"> <div class="theme-default-content content__default"><h1 id="coming-soon"><a href="#coming-soon" class="header-anchor">#</a> Coming Soon!</h1> <p>This section has not yet been completed.</p></div> <footer class="page-edit"><!----> <div class="last-updated"><span class="prefix">Last Updated: </span> <span class="time">5/14/2022, 6:44:34 PM</span></div></footer> <!----> </main></div></div><div class="global-ui"><!----></div></div>
<script src="/learn-wgpu/assets/js/app.c87751e7.js" defer></script><script src="/learn-wgpu/assets/js/17.7a15a658.js" defer></script><script src="/learn-wgpu/assets/js/47.04026fdc.js" defer></script>
</body>
</html>

Loading…
Cancel
Save