-
-
Notifications
You must be signed in to change notification settings - Fork 114
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* test(utils.resolve): add tests for camelCase, array indices * docs(attach): add playground demos for attach * docs(attach): add documentation for attach prop * Update docs/advanced/attach.md Co-authored-by: Tino Koch <[email protected]> --------- Co-authored-by: Alvaro Saburido <[email protected]> Co-authored-by: Tino Koch <[email protected]>
- Loading branch information
1 parent
4adc3bc
commit fbcbdc3
Showing
10 changed files
with
405 additions
and
5 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,190 @@ | ||
# `attach` 🖇 | ||
|
||
Using the `attach` prop, you can tell Tres exactly where you want to insert a child into its parent. | ||
|
||
:::info | ||
|
||
The `attach` prop is not required for many common cases. For instance: | ||
|
||
* adding a single `<Material>` to a `<Mesh>` | ||
* adding a `<Geometry>` to a `<Mesh>` | ||
* adding one or more `<Mesh>`s to a parent `<Mesh>` | ||
|
||
::: | ||
|
||
## Background | ||
|
||
Tres tries to automatically determine where to insert child tag into its parent. For example, in this code, Tres will: | ||
|
||
* automatically insert the geometry into `parent.geometry` | ||
* automatically insert the material into `parent.material` | ||
|
||
```vue | ||
<template> | ||
<TresMesh name="parent"> | ||
<TresBoxGeometry /> | ||
<TresMeshNormalMaterial /> | ||
</TresMesh> | ||
</template> | ||
``` | ||
|
||
## Problem | ||
|
||
Tres covers common cases, like above. But it doesn't cover every possible case. | ||
|
||
When Tres doesn't automatically choose the proper insertion location for a child, one solution is to fall back to procedural code in `<script>`. | ||
|
||
Here's how you might add multiple materials to a mesh using `<script>`: | ||
|
||
```vue | ||
<script setup lang="ts"> | ||
import { MeshBasicMaterial } from 'three' | ||
import { onMounted, shallowRef } from 'vue' | ||
const meshRef = shallowRef() | ||
onMounted(() => { | ||
meshRef.value.material = [ | ||
new MeshBasicMaterial({ color: 'red' }), | ||
new MeshBasicMaterial({ color: 'orange' }), | ||
new MeshBasicMaterial({ color: 'yellow' }), | ||
new MeshBasicMaterial({ color: 'green' }), | ||
new MeshBasicMaterial({ color: 'blue' }), | ||
new MeshBasicMaterial({ color: 'purple' }), | ||
] | ||
}) | ||
</script> | ||
<template> | ||
<TresMesh ref="meshRef"> | ||
<TresBoxGeometry /> | ||
</TresMesh> | ||
</template> | ||
``` | ||
|
||
But this workaround means: | ||
|
||
* your materials aren't managed by Tres | ||
* your code is imperative, not declarative | ||
* your code is non-reactive by default | ||
|
||
## Solution | ||
|
||
The `attach` prop lets you specify where an object will be added to the parent object using declarative code. | ||
|
||
## Usage | ||
|
||
Here's the example above, rewritten declaratively using `attach`: | ||
|
||
```vue | ||
<template> | ||
<TresMesh> | ||
<TresBoxGeometry /> | ||
<TresMeshBasicMaterial color="red" attach="material-0" /> | ||
<TresMeshBasicMaterial color="orange" attach="material-1" /> | ||
<TresMeshBasicMaterial color="yellow" attach="material-2" /> | ||
<TresMeshBasicMaterial color="green" attach="material-3" /> | ||
<TresMeshBasicMaterial color="blue" attach="material-4" /> | ||
<TresMeshBasicMaterial color="purple" attach="material-5" /> | ||
</TresMesh> | ||
</template> | ||
``` | ||
|
||
## "Pierced" `attach` | ||
|
||
You can deeply attach a child to a parent by "piercing" – i.e., using a kebab-case string. | ||
|
||
### Pseudocode | ||
|
||
First, here are a few simple pseudocode examples. This will attach `bar` at `foo.ab.cd`: | ||
|
||
```html | ||
<foo> | ||
<bar attach="ab-cd" /> | ||
</foo> | ||
``` | ||
|
||
This will attach `bar` at `foo.ab.cd.ef`: | ||
|
||
```html | ||
<foo> | ||
<bar attach="ab-cd-ef" /> | ||
</foo> | ||
``` | ||
|
||
### Usage | ||
|
||
As a concrete example, you can use "pierced" `attach` to add custom `BufferAttribute`s: | ||
|
||
```vue | ||
<script setup lang="ts"> | ||
import { TresCanvas } from '@tresjs/core' | ||
const positions = new Float32Array([-1.0, -1.0, 1.0, 1.0, -1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, -1.0, 1.0, 1.0, -1.0, -1.0, 1.0]) | ||
</script> | ||
<template> | ||
<TresCanvas clear-color="gray"> | ||
<TresMesh :scale="0.3333"> | ||
<TresBufferGeometry> | ||
<TresBufferAttribute | ||
attach="attributes-position" | ||
:count="positions.length / 3" | ||
:array="positions" | ||
:itemSize="3" | ||
/> | ||
</TresBufferGeometry> | ||
<TresMeshBasicMaterial color="red" /> | ||
</TresMesh> | ||
</TresCanvas> | ||
</template> | ||
``` | ||
|
||
## Arrays | ||
|
||
You can attach within arrays by using array indices in the `attach` string. | ||
|
||
### Usage | ||
|
||
For example, you can use array indices to attach `THREE` post-processing passes to the `THREE.EffectComposer.passes` array: | ||
|
||
```vue | ||
<script lang="ts" setup> | ||
import { EffectComposer } from 'three/examples/jsm/postprocessing/EffectComposer' | ||
import { RenderPass } from 'three/examples/jsm/postprocessing/RenderPass' | ||
import { OutputPass } from 'three/examples/jsm/postprocessing/OutputPass' | ||
import { UnrealBloomPass } from 'three-stdlib' | ||
import { extend, useLoop, useTres } from '@tresjs/core' | ||
import { shallowRef } from 'vue' | ||
extend({ EffectComposer, OutputPass, UnrealBloomPass, RenderPass }) | ||
const { renderer, scene, camera, sizes } = useTres() | ||
const composer = shallowRef<EffectComposer>() | ||
useLoop().render(() => { | ||
if (composer.value) { | ||
composer.value!.render() | ||
} | ||
}) | ||
</script> | ||
<template> | ||
<TresEffectComposer | ||
ref="composer" | ||
:args="[renderer]" | ||
:set-size="[sizes.width.value, sizes.height.value]" | ||
> | ||
<TresRenderPass | ||
:args="[scene, camera]" | ||
attach="passes-0" | ||
/> | ||
<TresUnrealBloomPass | ||
:args="[undefined, 0.5, 0.1, 0]" | ||
attach="passes-1" | ||
/> | ||
<TresOutputPass | ||
attach="passes-2" | ||
:set-size="[sizes.width.value, sizes.height.value]" | ||
/> | ||
</TresEffectComposer> | ||
</template> | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
47 changes: 47 additions & 0 deletions
47
playground/vue/src/pages/advanced/attachBufferGeometry/index.vue
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,47 @@ | ||
<script setup lang="ts"> | ||
import { TresCanvas } from '@tresjs/core' | ||
const previewDataUri = '' | ||
const positions = new Float32Array([ | ||
-1.0, | ||
-1.0, | ||
1.0, // v0 | ||
1.0, | ||
-1.0, | ||
1.0, // v1 | ||
1.0, | ||
1.0, | ||
1.0, // v2 | ||
1.0, | ||
1.0, | ||
1.0, // v3 | ||
-1.0, | ||
1.0, | ||
1.0, // v4 | ||
-1.0, | ||
-1.0, | ||
1.0, // v5 | ||
]) | ||
</script> | ||
|
||
<template> | ||
<TresCanvas clear-color="gray"> | ||
<TresMesh :scale="0.3333"> | ||
<TresBufferGeometry> | ||
<TresBufferAttribute attach="attributes-position" :count="positions.length / 3" :array="positions" :itemSize="3" /> | ||
</TresBufferGeometry> | ||
<TresMeshBasicMaterial color="red" /> | ||
</TresMesh> | ||
</TresCanvas> | ||
|
||
<OverlayInfo> | ||
<h1><code>attach</code>: BufferGeometry</h1> | ||
<h2>Setup</h2> | ||
<p> | ||
In this scene, there is a Mesh with a BufferGeometry, created with JSX. The BufferGeometry has a JSX BufferAttribute, attached to the BufferGeometry's <code>attributes.position</code> using <code>attach</code>. | ||
</p> | ||
<h2>Preview</h2> | ||
<img :src="previewDataUri" /> | ||
</OverlayInfo> | ||
</template> |
File renamed without changes.
Oops, something went wrong.