修复 围栏 fence 在更变坐标点后,不同步问题。

This commit is contained in:
hawk86104 2025-02-19 17:51:16 +08:00
parent 918e54f5f6
commit f878d13385

View File

@ -4,37 +4,40 @@
* @Autor: 地虎降天龙
* @Date: 2024-02-02 10:15:51
* @LastEditors: 地虎降天龙
* @LastEditTime: 2024-03-18 17:57:24
* @LastEditTime: 2025-02-19 17:50:20
-->
<template>
<TresMesh :renderOrder="2200" ref="tresMeshRef">
<TresBufferGeometry :position="[positionArray, 3]" :uv="[uvArray, 2]" />
<TresShaderMaterial v-bind="rippleShader" />
<TresBufferGeometry :position="[positionArray, 3]" :uv="[uvArray, 2]" />
<TresShaderMaterial v-bind="rippleShader" />
</TresMesh>
</template>
<script setup lang="ts">
import * as THREE from 'three'
import { watchEffect, ref } from 'vue'
import { watchEffect, ref, watch } from 'vue'
import { useRenderLoop } from '@tresjs/core'
import { getcenterPoint } from '../../common/utils'
const props = withDefaults(
defineProps<{
positionSrc?: Array<Object>
color?: string
opacity?: number
height?: number
num?: number
speed?: number
positionSrc?: Array<Object>
color?: string
opacity?: number
height?: number
num?: number
speed?: number
}>(),
{
positionSrc: [{ x: 0, y: 0 }, { x: 10, y: 10 }],
color: '#ffff00',
opacity: 0.8,
height: 100,
num: 8,
speed: 0.15
positionSrc: [
{ x: 0, y: 0 },
{ x: 10, y: 10 },
],
color: '#ffff00',
opacity: 0.8,
height: 100,
num: 8,
speed: 0.15,
},
)
@ -51,60 +54,60 @@ precision lowp int;
${THREE.ShaderChunk.fog_pars_vertex}
varying vec2 vUv;
void main() {
vec4 mvPosition = modelViewMatrix * vec4( position, 1.0 );
vUv = uv;
gl_Position = projectionMatrix * modelViewMatrix * vec4(position, 1.0);
${THREE.ShaderChunk.fog_vertex}
vec4 mvPosition = modelViewMatrix * vec4( position, 1.0 );
vUv = uv;
gl_Position = projectionMatrix * modelViewMatrix * vec4(position, 1.0);
${THREE.ShaderChunk.fog_vertex}
}
`,
fragmentShader: `
precision lowp float;
precision lowp int;
uniform float time;
uniform float opacity;
uniform vec3 color;
uniform float num;
uniform float speed;
varying vec2 vUv;
void main() {
vec4 fragColor = vec4(0.);
float sin = sin((vUv.y - time * speed) * 10. * num);
float high = 0.92;
float medium = 0.4;
if (sin > high) {
fragColor = vec4(mix(vec3(.8, 1., 1.), color, (1. - sin) / (1. - high)), 1.);
} else if(sin > medium) {
fragColor = vec4(color, mix(1., 0., 1.-(sin - medium) / (high - medium)));
} else {
fragColor = vec4(color, 0.);
}
vec3 fade = mix(color, vec3(0., 0., 0.), vUv.y);
fragColor = mix(fragColor, vec4(fade, 1.), 0.85);
gl_FragColor = vec4(fragColor.rgb, fragColor.a * opacity * (1. - vUv.y));
}
`,
uniforms: {
time: {
type: "pv2",
value: 0
},
color: {
type: "uvs",
value: new THREE.Color(props.color)
},
opacity: {
type: "pv2",
value: props.opacity
},
num: {
type: "pv2",
value: props.num
},
speed: {
type: "pv2",
value: props.speed
}
precision lowp float;
precision lowp int;
uniform float time;
uniform float opacity;
uniform vec3 color;
uniform float num;
uniform float speed;
varying vec2 vUv;
void main() {
vec4 fragColor = vec4(0.);
float sin = sin((vUv.y - time * speed) * 10. * num);
float high = 0.92;
float medium = 0.4;
if (sin > high) {
fragColor = vec4(mix(vec3(.8, 1., 1.), color, (1. - sin) / (1. - high)), 1.);
} else if(sin > medium) {
fragColor = vec4(color, mix(1., 0., 1.-(sin - medium) / (high - medium)));
} else {
fragColor = vec4(color, 0.);
}
vec3 fade = mix(color, vec3(0., 0., 0.), vUv.y);
fragColor = mix(fragColor, vec4(fade, 1.), 0.85);
gl_FragColor = vec4(fragColor.rgb, fragColor.a * opacity * (1. - vUv.y));
}
`,
uniforms: {
time: {
type: 'pv2',
value: 0,
},
color: {
type: 'uvs',
value: new THREE.Color(props.color),
},
opacity: {
type: 'pv2',
value: props.opacity,
},
num: {
type: 'pv2',
value: props.num,
},
speed: {
type: 'pv2',
value: props.speed,
},
},
}
let positionArray = null as any
@ -113,44 +116,44 @@ function getRippleGeometry(points = [] as Array<any>, height: number) {
const positions = [] as Array<any>
const uvs = [] as Array<any>
for (let i = 0, j = positions.length, t = uvs.length; i < points.length - 1; i++) {
let vUvyMax = 1
let left = points[i]
let right = points[i + 1]
positions[j++] = left.x
positions[j++] = 0
positions[j++] = left.y
uvs[t++] = 0
uvs[t++] = 0
let vUvyMax = 1
let left = points[i]
let right = points[i + 1]
positions[j++] = left.x
positions[j++] = 0
positions[j++] = left.y
uvs[t++] = 0
uvs[t++] = 0
positions[j++] = right.x
positions[j++] = 0
positions[j++] = right.y
uvs[t++] = 1
uvs[t++] = 0
positions[j++] = right.x
positions[j++] = 0
positions[j++] = right.y
uvs[t++] = 1
uvs[t++] = 0
positions[j++] = left.x
positions[j++] = height
positions[j++] = left.y
uvs[t++] = 0
uvs[t++] = vUvyMax
positions[j++] = left.x
positions[j++] = height
positions[j++] = left.y
uvs[t++] = 0
uvs[t++] = vUvyMax
positions[j++] = left.x
positions[j++] = height
positions[j++] = left.y
uvs[t++] = 0
uvs[t++] = vUvyMax
positions[j++] = left.x
positions[j++] = height
positions[j++] = left.y
uvs[t++] = 0
uvs[t++] = vUvyMax
positions[j++] = right.x
positions[j++] = 0
positions[j++] = right.y
uvs[t++] = 1
uvs[t++] = 0
positions[j++] = right.x
positions[j++] = 0
positions[j++] = right.y
uvs[t++] = 1
uvs[t++] = 0
positions[j++] = right.x
positions[j++] = height
positions[j++] = right.y
uvs[t++] = 1
uvs[t++] = vUvyMax
positions[j++] = right.x
positions[j++] = height
positions[j++] = right.y
uvs[t++] = 1
uvs[t++] = vUvyMax
}
positionArray = new Float32Array(positions)
uvArray = new Float32Array(uvs)
@ -158,7 +161,7 @@ function getRippleGeometry(points = [] as Array<any>, height: number) {
// //
//
const { centerPoint, points } = getcenterPoint(props.positionSrc)
let { centerPoint, points } = getcenterPoint(props.positionSrc)
getRippleGeometry(points, props.height)
@ -168,20 +171,32 @@ onLoop(({ delta }) => {
})
watchEffect(() => {
if (props.color) {
rippleShader.uniforms.color.value = new THREE.Color(props.color)
rippleShader.uniforms.color.value = new THREE.Color(props.color)
}
if (props.opacity) {
rippleShader.uniforms.opacity.value = props.opacity
rippleShader.uniforms.opacity.value = props.opacity
}
if (props.num) {
rippleShader.uniforms.num.value = props.num
rippleShader.uniforms.num.value = props.num
}
if (props.speed) {
rippleShader.uniforms.speed.value = props.speed
rippleShader.uniforms.speed.value = props.speed
}
if (tresMeshRef.value) {
tresMeshRef.value.position.set(centerPoint.x, tresMeshRef.value.position.y, centerPoint.y)
tresMeshRef.value.position.set(centerPoint.x, tresMeshRef.value.position.y, centerPoint.y)
}
})
</script>
watch(
() => props.positionSrc,
(val) => {
const { centerPoint: tcenterPoint, points: tpoints } = getcenterPoint(val)
centerPoint = tcenterPoint
points = tpoints
getRippleGeometry(points, props.height)
if (tresMeshRef.value) {
tresMeshRef.value.position.set(centerPoint.x, tresMeshRef.value.position.y, centerPoint.y)
}
},
)
</script>