Skip to content

TresCustomShaderMaterial ^3.6.0

The cientos package provides a new <TresCustomShaderMaterial /> component which is a wrapper around the three-custom-shader-material class. As states in the repo, it "lets you extend Three.js' material library with your own Vertex and Fragment shaders".

Usage

vue
<script setup lang="ts">
import { TresCanvas, useRenderLoop } from '@tresjs/core'
import {
  CustomShaderMaterial,
  useTweakPane,
  OrbitControls,
} from '@tresjs/cientos'

import { MeshBasicMaterial } from 'three'
import { onMounted, nextTick } from 'vue'

const { onLoop } = useRenderLoop()

const gl = {
  clearColor: '#82DBC5',
}

const materialProps = {
  baseMaterial: MeshBasicMaterial,
  fragmentShader: `
    varying float vWobble;

    uniform float u_Time;

    void main() {
      float wobble = vWobble * 0.5 + 0.5;
      csm_FragColor = mix(vec4(0.0, 0.4, 1.5, 1.0), vec4(1.2, 0.6, 0.8, 1.0), wobble);
    }
  `,
  vertexShader: `
    uniform float u_Time;
    uniform float u_WobbleSpeed;
    uniform float u_WobbleAmplitude;
    uniform float u_WobbleFrequency;

    varying float vWobble;

    void main() {
      float wobble = sin(csm_Position.z * u_WobbleFrequency + u_Time * u_WobbleSpeed);
      csm_Position += normal * wobble * u_WobbleAmplitude;

      vWobble = wobble;
    }
  `,
  uniforms: {
    u_Time: { value: 0 },
    u_WobbleSpeed: { value: 3 },
    u_WobbleAmplitude: { value: 0.07 },
    u_WobbleFrequency: { value: 3 },
  },
}

onMounted(async () => {
  await nextTick()

  createDebugPanel()

  onLoop(() => {
    materialProps.uniforms.u_Time.value
			+= 0.01 * materialProps.uniforms.u_WobbleSpeed.value
  })
})

function createDebugPanel() {
  const { pane } = useTweakPane('.debug-container')

  const folder = pane.addFolder({
    title: 'Settings',
  })

  folder.addInput(materialProps.uniforms.u_WobbleSpeed, 'value', {
    label: 'Wobble Speed',
    min: 0,
    max: 10,
  })

  folder.addInput(materialProps.uniforms.u_WobbleAmplitude, 'value', {
    label: 'Wobble Amplitude',
    min: 0,
    max: 0.2,
  })

  folder.addInput(materialProps.uniforms.u_WobbleFrequency, 'value', {
    label: 'Wobble Frequency',
    min: 1,
    max: 30,
  })
}
</script>

<template>
  <TresCanvas v-bind="gl">
    <TresPerspectiveCamera
      :position="[0, 2, 4]"
      :look-at="[0, 0, 0]"
    />

    <OrbitControls />

    <TresMesh>
      <TresTorusKnotGeometry :args="[1, 0.3, 512, 32]" />
      <CustomShaderMaterial v-bind="materialProps" />
    </TresMesh>
  </TresCanvas>

  <div class="debug-container" />
</template>

<style scoped>
.debug-container {
	position: absolute;
	top: 0px;
	right: 0px;
}
</style>
<script setup lang="ts">
import { TresCanvas, useRenderLoop } from '@tresjs/core'
import {
  CustomShaderMaterial,
  useTweakPane,
  OrbitControls,
} from '@tresjs/cientos'

import { MeshBasicMaterial } from 'three'
import { onMounted, nextTick } from 'vue'

const { onLoop } = useRenderLoop()

const gl = {
  clearColor: '#82DBC5',
}

const materialProps = {
  baseMaterial: MeshBasicMaterial,
  fragmentShader: `
    varying float vWobble;

    uniform float u_Time;

    void main() {
      float wobble = vWobble * 0.5 + 0.5;
      csm_FragColor = mix(vec4(0.0, 0.4, 1.5, 1.0), vec4(1.2, 0.6, 0.8, 1.0), wobble);
    }
  `,
  vertexShader: `
    uniform float u_Time;
    uniform float u_WobbleSpeed;
    uniform float u_WobbleAmplitude;
    uniform float u_WobbleFrequency;

    varying float vWobble;

    void main() {
      float wobble = sin(csm_Position.z * u_WobbleFrequency + u_Time * u_WobbleSpeed);
      csm_Position += normal * wobble * u_WobbleAmplitude;

      vWobble = wobble;
    }
  `,
  uniforms: {
    u_Time: { value: 0 },
    u_WobbleSpeed: { value: 3 },
    u_WobbleAmplitude: { value: 0.07 },
    u_WobbleFrequency: { value: 3 },
  },
}

onMounted(async () => {
  await nextTick()

  createDebugPanel()

  onLoop(() => {
    materialProps.uniforms.u_Time.value
			+= 0.01 * materialProps.uniforms.u_WobbleSpeed.value
  })
})

function createDebugPanel() {
  const { pane } = useTweakPane('.debug-container')

  const folder = pane.addFolder({
    title: 'Settings',
  })

  folder.addInput(materialProps.uniforms.u_WobbleSpeed, 'value', {
    label: 'Wobble Speed',
    min: 0,
    max: 10,
  })

  folder.addInput(materialProps.uniforms.u_WobbleAmplitude, 'value', {
    label: 'Wobble Amplitude',
    min: 0,
    max: 0.2,
  })

  folder.addInput(materialProps.uniforms.u_WobbleFrequency, 'value', {
    label: 'Wobble Frequency',
    min: 1,
    max: 30,
  })
}
</script>

<template>
  <TresCanvas v-bind="gl">
    <TresPerspectiveCamera
      :position="[0, 2, 4]"
      :look-at="[0, 0, 0]"
    />

    <OrbitControls />

    <TresMesh>
      <TresTorusKnotGeometry :args="[1, 0.3, 512, 32]" />
      <CustomShaderMaterial v-bind="materialProps" />
    </TresMesh>
  </TresCanvas>

  <div class="debug-container" />
</template>

<style scoped>
.debug-container {
	position: absolute;
	top: 0px;
	right: 0px;
}
</style>

Props

Being a wrapper around the CustomShaderMaterial class, the <TresCustomShaderMaterial /> component accepts all the props that the class does. You can find the full list of props in the official repo.