no message

This commit is contained in:
hawk86104 2024-09-07 23:03:48 +08:00
parent 694c1944d7
commit dd72d70eff

View File

@ -1,125 +1,104 @@
<template>
<TresGroup :scale="[1, tgHeight, 1]">
<TresMesh :geometry="mesh.geometry" v-for="(mesh, index) in meshList" :key="index" :position="[0, mesh.y, 0]">
<TresMeshStandardMaterial :color="mesh.data.color" transparent />
</TresMesh>
<TresGroup v-for="(mesh, index) in meshList" :key="index" :rotation="[0, mesh.titleRotate, 0]">
<stackedBarText
:position="[0, mesh.y + mesh.height / 2, mesh.titleOffset]"
:rotation="[mesh.titleTilt, 0, 0]"
:textContent="`${mesh.data.value}${mesh.data.title}`"
:fontSize="mesh.titleSize"
/>
</TresGroup>
</TresGroup>
<TresGroup>
<TresMesh ref="tmRef" :rotation-x="-Math.PI / 2" :position-y="-0.1">
<TresPlaneGeometry ref="tpgRef" :args="[10, 10]" />
<TresMeshStandardMaterial ref="tmsmRef" v-bind="tmsMaterialConfig" />
</TresMesh>
<primitive :object="gridHelp" />
</TresGroup>
</template>
<script setup lang="ts">
import { ref, onMounted, watch } from 'vue'
import * as THREE from 'three'
import stackedBarText from './text.vue'
import { gsap } from 'gsap'
<script lang="ts" setup>
import { Vector2, RepeatWrapping, Color, GridHelper } from 'three'
import { useTexture } from '@tresjs/core'
import { Reflector } from '../lib/alienJS/all.three.js'
import { makeVertexShader, makeFragmentShader } from '../shaders/reflectorDiffuse.js'
import { watchEffect, ref, watch } from 'vue'
const props = withDefaults(
defineProps<{
data?: Array<{ value: number; title: string; color: string }>
bottomRadius?: number
height?: number
segments?: number
spacing?: number
heightDistribution?: string
showStyle?: string
delay?: number
}>(),
{
data: [
//#3aa274 #fe8256 #9962b2 #f8285b
{ value: 42, title: '%', color: '#5072c4' },
{ value: 20, title: '%', color: '#92cb79' },
{ value: 18, title: '%', color: '#fcc660' },
{ value: 12, title: '%', color: '#ef6467' },
{ value: 10, title: '%', color: '#70c1dd' },
], //
bottomRadius: 1.6, //
height: 6, //
segments: 4, //
spacing: 0.1, //
heightDistribution: 'uniform', // : uniform proportional
showStyle: 'Cone', // : Cone Cylinder
delay: 0.1, //
},
defineProps<{
mirror?: Number //
mixStrength?: Number //
showGridHelper?: boolean
color?: string
}>(),
{
mirror: 1,
mixStrength: 10,
showGridHelper: true,
color: '#ffffff',
},
)
const meshList: Array<any> = []
const calcHeightDistribution = () => {
let result: Array<number> = []
if (props.heightDistribution === 'uniform') {
const barheight = props.height / props.data.length
result = new Array(props.data.length).fill(barheight)
} else {
const total = props.data.map((x) => x.value).reduce((a, v) => a + v)
props.data.forEach((item) => {
result.push((item.value / total) * props.height)
})
}
return result
const gridHelp = new GridHelper(9.5, 10)
const tpgRef = ref()
const tmRef = ref()
const tmsmRef = ref()
const reflector = new Reflector()
const uniforms = {
mirror: { value: props.mirror },
mixStrength: { value: props.mixStrength },
}
const reCalcMeshList = () => {
meshList.splice(0, meshList.length)
const heightList = calcHeightDistribution()
let bottomRadius = props.bottomRadius
let curRadius = props.bottomRadius
let titleRotate = -THREE.MathUtils.degToRad(180 / props.segments)
let titleTilt = 0
let y = 0
if (props.showStyle === 'Cone') {
const facedistance = props.bottomRadius * Math.cos(Math.PI / props.segments)
titleTilt = Math.atan(props.height / facedistance) - Math.PI / 2
}
props.data.forEach((data, index) => {
const height = heightList[index]
if (props.showStyle === 'Cone') {
const radius = (height / props.height) * props.bottomRadius
const geometry = new THREE.CylinderGeometry(curRadius - radius, bottomRadius, height, props.segments)
geometry.translate(0, height / 2, 0)
const titleOffset = (curRadius - radius / 3) * Math.cos(Math.PI / props.segments)
const titleSize = THREE.MathUtils.mapLinear(index, 0, props.data.length, 0.05 * props.height, 0.005 * props.height)
meshList.push({ geometry, y, height, titleOffset, titleTilt, titleRotate, titleSize, data })
curRadius -= radius
bottomRadius = curRadius
} else {
const geometry = new THREE.CylinderGeometry(curRadius, bottomRadius, height, props.segments)
geometry.translate(0, height / 2, 0)
const titleOffset = curRadius * Math.cos(Math.PI / props.segments)
meshList.push({ geometry, y, height, titleOffset, titleTilt, titleRotate, titleSize: 0.05 * props.height, data })
}
y += height + props.spacing
})
const pTexture = await useTexture([
'./plugins/floor/image/polished_concrete_basecolor.jpg',
'./plugins/floor/image/polished_concrete_normal.jpg',
'./plugins/floor/image/polished_concrete_orm.jpg',
])
for (var i = 0; i < 3; i++) {
pTexture[i].wrapS = RepeatWrapping
pTexture[i].wrapT = RepeatWrapping
pTexture[i].repeat.set(16, 16)
}
reCalcMeshList()
const tmsMaterialConfig = {
color: new Color('#444'),
metalness: 1,
roughness: 1,
map: pTexture[0],
metalnessMap: pTexture[2],
roughnessMap: pTexture[2],
aoMap: pTexture[2],
aoMapIntensity: 1,
normalMap: pTexture[1],
normalScale: new Vector2(3, 3),
}
const makeOnBeforeCompile = (shader: any) => {
shader.uniforms.reflectMap = reflector.renderTargetUniform
shader.uniforms.textureMatrix = reflector.textureMatrixUniform
shader.uniforms = Object.assign(shader.uniforms, uniforms)
makeVertexShader(shader)
makeFragmentShader(shader)
}
watchEffect(() => {
if (tpgRef.value) {
tpgRef.value.attributes.uv1 = tpgRef.value.attributes.uv
}
if (tmsmRef.value) {
// Second channel for aoMap and lightMap
// https://threejs.org/docs/#api/en/materials/MeshStandardMaterial.aoMap
tmsmRef.value.aoMap.channel = 1
tmsmRef.value.onBeforeCompile = makeOnBeforeCompile
}
if (tmRef.value) {
tmRef.value.add(reflector)
tmRef.value.onBeforeRender = (rendererSelf: any, sceneSelf: any, cameraSelf: any) => {
reflector.update(rendererSelf, sceneSelf, cameraSelf)
}
}
if (props.color) {
if (tmsmRef.value) {
tmsmRef.value.color = new Color(props.color)
}
}
})
watch(
() => [props.height, props.bottomRadius, props.segments, props.spacing, props.heightDistribution, props.showStyle],
() => {
reCalcMeshList()
},
() => props.showGridHelper,
(newVal) => {
gridHelp.visible = newVal
},
)
const tgHeight = ref(0.1)
onMounted(() => {
playFunc()
})
const playFunc = () => {
tgHeight.value = 0.1
gsap.to(tgHeight, {
value: 1,
duration: 1,
delay: props.delay,
ease: 'circ.out',
})
}
defineExpose({
playFunc: playFunc,
})
</script>