Vue 에서 3D 태그 클 라 우 드 를 실현 하 는 상세 한 코드
코드:
페이지 부분:
<template>
<div class="tagcloud-all"
ref="tagcloudall">
<a v-for="item in tagList" :href="item.url" rel="external nofollow" :style="'color:' + item.color + ';top: 0;left: 0;filter:none;'">{{item.name}}</a>
</div>
</template>
CSS 부분:
//
.tagcloud-all {
position: relative;
a {
position: absolute;
top: 0px;
left: 0px;
color: #fff;
font-weight: bold;
text-decoration: none;
padding: 3px 6px;
&:hover {
color: #FF0000;
letter-spacing: 2px;
}
}
}
JS 부분:
export default {
name: "tagcloud",
data() {
return {
tagList: [],
radius: 120,
dtr: Math.PI / 180,
d: 300,
mcList: [],
active: false,
lasta: 1,
lastb: 1,
distr: true,
tspeed: 10,
size: 250,
mouseX: 0,
mouseY: 0,
howElliptical: 1,
oList: null,
oA: null,
sa: 0,
ca: 0,
sb: 0,
cb: 0,
sc: 0,
cc: 0
}
},
methods: {
//
getRandomNum() {
return Math.floor(Math.random() * (255 + 1));
},
//
sineCosine(a, b, c) {
this.sa = Math.sin(a * this.dtr);
this.ca = Math.cos(a * this.dtr);
this.sb = Math.sin(b * this.dtr);
this.cb = Math.cos(b * this.dtr);
this.sc = Math.sin(c * this.dtr);
this.cc = Math.cos(c * this.dtr);
},
//
positionAll() {
this.$nextTick(() => { // : onReady $nextTick
var phi = 0;
var theta = 0;
var max = this.mcList.length;
var aTmp = [];
var oFragment = document.createDocumentFragment();
//
for (let i = 0; i < this.tagList.length; i++) {
aTmp.push(this.oA[i]);
}
aTmp.sort(() => {
return Math.random() < 0.5 ? 1 : -1;
});
for (let i = 0; i < aTmp.length; i++) {
oFragment.appendChild(aTmp[i]);
}
this.oList.appendChild(oFragment);
for (let i = 1; i < max + 1; i++) {
if (this.distr) {
phi = Math.acos(-1 + (2 * i - 1) / max);
theta = Math.sqrt(max * Math.PI) * phi;
} else {
phi = Math.random() * (Math.PI);
theta = Math.random() * (2 * Math.PI);
}
//
this.mcList[i - 1].cx = this.radius * Math.cos(theta) * Math.sin(phi);
this.mcList[i - 1].cy = this.radius * Math.sin(theta) * Math.sin(phi);
this.mcList[i - 1].cz = this.radius * Math.cos(phi);
this.oA[i - 1].style.left = this.mcList[i - 1].cx + this.oList.offsetWidth / 2 - this.mcList[i - 1].offsetWidth / 2 + 'px';
this.oA[i - 1].style.top = this.mcList[i - 1].cy + this.oList.offsetHeight / 2 - this.mcList[i - 1].offsetHeight / 2 + 'px';
}
})
},
//
update() {
this.$nextTick(() => { // : onReady $nextTick
var a;
var b;
if (this.active) {
a = (-Math.min(Math.max(-this.mouseY, -this.size), this.size) / this.radius) * this.tspeed;
b = (Math.min(Math.max(-this.mouseX, -this.size), this.size) / this.radius) * this.tspeed;
} else {
a = this.lasta * 0.98;
b = this.lastb * 0.98;
}
this.lasta = a;
this.lastb = b;
if (Math.abs(a) <= 0.01 && Math.abs(b) <= 0.01) {
return
}
var c = 0;
this.sineCosine(a, b, c);
for (var j = 0; j < this.mcList.length; j++) {
var rx1 = this.mcList[j].cx;
var ry1 = this.mcList[j].cy * this.ca + this.mcList[j].cz * (-this.sa);
var rz1 = this.mcList[j].cy * this.sa + this.mcList[j].cz * this.ca;
var rx2 = rx1 * this.cb + rz1 * this.sb;
var ry2 = ry1;
var rz2 = rx1 * (-this.sb) + rz1 * this.cb;
var rx3 = rx2 * this.cc + ry2 * (-this.sc);
var ry3 = rx2 * this.sc + ry2 * this.cc;
var rz3 = rz2;
this.mcList[j].cx = rx3;
this.mcList[j].cy = ry3;
this.mcList[j].cz = rz3;
var per = this.d / (this.d + rz3);
this.mcList[j].x = (this.howElliptical * rx3 * per) - (this.howElliptical * 2);
this.mcList[j].y = ry3 * per;
this.mcList[j].scale = per;
this.mcList[j].alpha = per;
this.mcList[j].alpha = (this.mcList[j].alpha - 0.6) * (10 / 6);
}
this.doPosition();
this.depthSort();
})
},
//
doPosition() {
this.$nextTick(() => { // : onReady $nextTick
var l = this.oList.offsetWidth / 2;
var t = this.oList.offsetHeight / 2;
for (var i = 0; i < this.mcList.length; i++) {
this.oA[i].style.left = this.mcList[i].cx + l - this.mcList[i].offsetWidth / 2 + 'px';
this.oA[i].style.top = this.mcList[i].cy + t - this.mcList[i].offsetHeight / 2 + 'px';
this.oA[i].style.fontSize = Math.ceil(12 * this.mcList[i].scale / 2) + 8 + 'px';
// this.oA[i].style.filter = "alpha(opacity=" + 100 * this.mcList[i].alpha + ")";
this.oA[i].style.opacity = this.mcList[i].alpha;
}
})
},
//
depthSort() {
this.$nextTick(() => { // : onReady $nextTick
var aTmp = [];
for (let i = 0; i < this.oA.length; i++) {
aTmp.push(this.oA[i]);
}
aTmp.sort(function (vItem1, vItem2) {
if (vItem1.cz > vItem2.cz) {
return -1;
} else if (vItem1.cz < vItem2.cz) {
return 1;
} else {
return 0;
}
});
for (let i = 0; i < aTmp.length; i++) {
aTmp[i].style.zIndex = i;
}
})
},
// tagList
query() {
//
let tagListOrg = [
{ name: ' 1', url: 'www.baidu.com' },
{ name: ' 2', url: 'www.baidu.com' },
{ name: ' 3', url: 'www.baidu.com' },
{ name: ' 4', url: 'www.baidu.com' },
{ name: ' 5', url: 'www.baidu.com' },
{ name: ' 6', url: 'www.baidu.com' },
{ name: ' 7', url: 'www.baidu.com' },
{ name: ' 8', url: 'www.baidu.com' },
{ name: ' 9', url: 'www.baidu.com' },
{ name: ' 10', url: 'www.baidu.com' },
{ name: ' 11', url: 'www.baidu.com' },
{ name: ' 12', url: 'www.baidu.com' },
{ name: ' 13', url: 'www.baidu.com' },
{ name: ' 14', url: 'www.baidu.com' },
{ name: ' 15', url: 'www.baidu.com' },
{ name: ' 16', url: 'www.baidu.com' },
{ name: ' 16', url: 'www.baidu.com' },
{ name: ' 16', url: 'www.baidu.com' },
{ name: ' 16', url: 'www.baidu.com' },
{ name: ' 16', url: 'www.baidu.com' },
{ name: ' 16', url: 'www.baidu.com' },
{ name: ' 16', url: 'www.baidu.com' },
{ name: ' 16', url: 'www.baidu.com' },
{ name: ' 16', url: 'www.baidu.com' },
{ name: ' 16', url: 'www.baidu.com' },
{ name: ' 16', url: 'www.baidu.com' },
{ name: ' 16', url: 'www.baidu.com' },
{ name: ' 16', url: 'www.baidu.com' },
{ name: ' 16', url: 'www.baidu.com' },
{ name: ' 17', url: 'www.baidu.com' }
];
// tagList
tagListOrg.forEach(item => {
item.color = "rgb(" + this.getRandomNum() + "," + this.getRandomNum() + "," + this.getRandomNum() + ")";
})
this.tagList = tagListOrg;
this.onReady();
},
//
onReady() {
this.$nextTick(() => {
this.oList = this.$refs.tagcloudall;
this.oA = this.oList.getElementsByTagName('a')
var oTag = null;
for (var i = 0; i < this.oA.length; i++) {
oTag = {};
oTag.offsetWidth = this.oA[i].offsetWidth;
oTag.offsetHeight = this.oA[i].offsetHeight;
this.mcList.push(oTag);
}
this.sineCosine(0, 0, 0);
this.positionAll();
this.oList.onmouseover = () => {
this.active = true;
}
this.oList.onmouseout = () => {
this.active = false;
}
this.oList.onmousemove = (event) => {
var oEvent = window.event || event;
this.mouseX = oEvent.clientX - (this.oList.offsetLeft + this.oList.offsetWidth / 2);
this.mouseY = oEvent.clientY - (this.oList.offsetTop + this.oList.offsetHeight / 2);
this.mouseX /= 5;
this.mouseY /= 5;
}
setInterval(() => {
this.update()
}, 30); // setInterval(this.update(), 30)
})
}
},
created() {
this.$nextTick(() => {
this.query();
})
}
}
Vue 에서 3D 라벨 클 라 우 드 를 실현 하 는 것 에 관 한 이 글 은 여기까지 소개 되 었 습 니 다.Vue 3D 라벨 클 라 우 드 에 관 한 더 많은 내용 은 저희 의 이전 글 을 검색 하거나 아래 의 관련 글 을 계속 조회 하 시기 바 랍 니 다.앞으로 저 희 를 많이 사랑 해 주세요!
이 내용에 흥미가 있습니까?
현재 기사가 여러분의 문제를 해결하지 못하는 경우 AI 엔진은 머신러닝 분석(스마트 모델이 방금 만들어져 부정확한 경우가 있을 수 있음)을 통해 가장 유사한 기사를 추천합니다:
Vue Render 함수로 DOM 노드 코드 인스턴스 만들기render에서createElement 함수를 사용하여 DOM 노드를 만드는 것은 직관적이지 않지만 일부 독립 구성 요소의 디자인에서 특수한 수요를 충족시킬 수 있습니다.간단한 렌더링 예는 다음과 같습니다. 또한 v...
텍스트를 자유롭게 공유하거나 복사할 수 있습니다.하지만 이 문서의 URL은 참조 URL로 남겨 두십시오.
CC BY-SA 2.5, CC BY-SA 3.0 및 CC BY-SA 4.0에 따라 라이센스가 부여됩니다.