mirror of
https://gitee.com/ice-gl/icegl-three-vue-tres.git
synced 2025-04-05 06:22:43 +08:00
修复 围栏 fence 在更变坐标点后,不同步问题。
This commit is contained in:
parent
918e54f5f6
commit
f878d13385
@ -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>
|
||||
|
Loading…
x
Reference in New Issue
Block a user