1
0
mirror of https://github.com/Foltik/Shimapan synced 2024-11-11 07:54:55 -05:00
shimapan/public/js/typegraph.js

599 lines
55 KiB
JavaScript
Raw Normal View History

2017-10-14 15:16:58 -04:00
var camera = {
focus: 600,
self: {
x: 0,
y: 0,
z: 0
},
rotate: {
x: 0,
y: 0,
z: 0
},
up: {
x: 0,
y: 1,
z: 0
},
zoom: 1,
display: {
x: window.innerWidth / 2,
y: window.innerHeight * (13/16),
z: 0
}
};
var affine = {
world: {
size: function (p, size) {
return {
x: p.x * size.x,
y: p.y * size.y,
z: p.z * size.z
}
},
rotate: {
x: function (p, rotate) {
return {
x: p.x,
y: p.y * Math.cos(dtr(rotate.x)) - p.z * Math.sin(dtr(rotate.x)),
z: p.y * Math.sin(dtr(rotate.x)) + p.z * Math.cos(dtr(rotate.x))
}
},
y: function (p, rotate) {
return {
x: p.x * Math.cos(dtr(rotate.y)) + p.z * Math.sin(dtr(rotate.y)),
y: p.y,
z: -p.x * Math.sin(dtr(rotate.y)) + p.z * Math.cos(dtr(rotate.y))
}
},
z: function (p, rotate) {
return {
x: p.x * Math.cos(dtr(rotate.z)) - p.y * Math.sin(dtr(rotate.z)),
y: p.x * Math.sin(dtr(rotate.z)) + p.y * Math.cos(dtr(rotate.z)),
z: p.z
}
}
},
position: function (p, position) {
return {
x: p.x + position.x,
y: p.y + position.y,
z: p.z + position.z
}
}
},
view: {
point: function (p) {
return {
x: p.x - camera.self.x,
y: p.y - camera.self.y,
z: p.z - camera.self.z
}
},
x: function (p) {
return {
x: p.x,
y: p.y * Math.cos(dtr(camera.rotate.x)) - p.z * Math.sin(dtr(camera.rotate.x)),
z: p.y * Math.sin(dtr(camera.rotate.x)) + p.z * Math.cos(dtr(camera.rotate.x))
}
},
y: function (p) {
return {
x: p.x * Math.cos(dtr(camera.rotate.y)) + p.z * Math.sin(dtr(camera.rotate.y)),
y: p.y,
z: p.x * -Math.sin(dtr(camera.rotate.y)) + p.z * Math.cos(dtr(camera.rotate.y))
}
},
viewReset: function (p) {
return {
x: p.x - camera.self.x,
y: p.y - camera.self.y,
z: p.z - camera.self.z
}
},
righthandedReversal: function (p) {
return {
x: p.x,
y: -p.y,
z: p.z
}
}
},
perspective: function (p) {
return {
x: p.x * ((camera.focus - camera.self.z) / ((camera.focus - camera.self.z) - p.z)) * camera.zoom,
y: p.y * ((camera.focus - camera.self.z) / ((camera.focus - camera.self.z) - p.z)) * camera.zoom,
z: p.z * ((camera.focus - camera.self.z) / ((camera.focus - camera.self.z) - p.z)) * camera.zoom,
p: ((camera.focus - camera.self.z) / ((camera.focus - camera.self.z) - p.z)) * camera.zoom
}
},
display: function (p, display) {
return {
x: p.x + display.x,
y: p.y + display.y,
z: p.z + display.z,
p: p.p
}
},
process: function (model, size, rotate, position, display) {
var ret = affine.world.size(model, size);
ret = affine.world.rotate.x(ret, rotate);
ret = affine.world.rotate.y(ret, rotate);
ret = affine.world.rotate.z(ret, rotate);
ret = affine.world.position(ret, position);
ret = affine.view.point(ret);
ret = affine.view.x(ret);
ret = affine.view.y(ret);
ret = affine.view.viewReset(ret);
ret = affine.view.righthandedReversal(ret);
ret = affine.perspective(ret);
ret = affine.display(ret, display);
return ret;
}
};
var vertex3d = function (param) {
this.affineIn = {};
this.affineOut = {};
if (param.vertex !== undefined) {
this.affineIn.vertex = param.vertex;
} else {
this.affineIn.vertex = {x: 0, y: 0, z: 0};
}
if (param.size !== undefined) {
this.affineIn.size = param.size;
} else {
this.affineIn.size = {x: 1, y: 1, z: 1};
}
if (param.rotate !== undefined) {
this.affineIn.rotate = param.rotate;
} else {
this.affineIn.rotate = {x: 0, y: 0, z: 0};
}
if (param.position !== undefined) {
this.affineIn.position = param.position;
} else {
this.affineIn.position = {x: 0, y: 0, z: 0};
}
};
vertex3d.prototype = {
vertexUpdate: function () {
this.affineOut = affine.process(
this.affineIn.vertex,
this.affineIn.size,
this.affineIn.rotate,
this.affineIn.position,
camera.display
);
}
};
var dtr = function (v) {
return v * Math.PI / 180;
};
//cordinate system transformation.
//polar to rectangle.
var polarToRectangle = function (dX, dY, radius) {
var x = Math.sin(dtr(dX)) * Math.cos(dtr(dY)) * radius;
var y = Math.sin(dtr(dX)) * Math.sin(dtr(dY)) * radius;
var z = Math.cos(dtr(dX)) * radius;
return {x: y, y: z, z: x};
};
//rectangle to polar.
var rectangleToPolar = function (x, y, z) {
var xD;
var yD;
var zD;
if (x === 0) xD = 0.001;
else xD = x;
if (y === 0) yD = 0.001;
else yD = y;
if (z === 0) zD = 0.001;
else zD = z;
var radius = Math.sqrt(xD * xD + yD * yD + zD * zD);
var theta = Math.atan(zD / Math.sqrt(xD * xD + yD * yD));
var phi = Math.atan(yD / xD);
return {x: theta * (180 / Math.PI), y: phi * (180 / Math.PI), r: radius};
};
var closeValue = function (minTime, maxTime) {
this.flag = 0;
this.progress = 0;
this.startTime = 0;
this.durationTime = 0;
this.fromValue = 0;
this.toValue = 0;
this.minValue = 0;
this.maxValue = 1;
this.minDuration = minTime;
this.maxDuration = maxTime;
};
closeValue.prototype = {
init: function () {
this.durationTime = this.minDuration + (this.maxDuration - this.minDuration) * Math.random();
this.startTime = Date.now();
this.progress = Math.min(1, ((Date.now() - this.startTime) / this.durationTime));
this.fromValue = this.toValue;
this.toValue = this.minValue + this.maxValue * Math.random();
this.flag = 1;
return this.fromValue + (this.toValue - this.fromValue) * this.progress;
},
update: function () {
this.progress = Math.min(1, ((Date.now() - this.startTime) / this.durationTime));
if (this.progress === 1) this.flag = 0;
return this.fromValue + (this.toValue - this.fromValue) * this.progress;
},
execution: function () {
if (this.flag === 0) {
return this.init()
}
else if (this.flag === 1) {
return this.update()
}
}
};
var strokeColor = "rgba(255,255,255,0.1)";
var backgroundColor = "rgba(0,0,0,0)";
var vibrateFlag = false;
var canvas = document.getElementById("canvas");
var canvasWidth = window.innerWidth;
var canvasHeight = window.innerHeight;
canvas.width = canvasWidth;
canvas.height = canvasHeight;
var ctx = canvas.getContext("2d");
ctx.strokeStyle = strokeColor;
window.onresize = function () {
canvas.width = window.innerWidth;
canvas.height = window.innerHeight;
camera.display.x = window.innerWidth / 2;
camera.display.y = window.innerHeight * (13/16);
};
/* class */
var sphere = function (arg) {
this.flag = true;
this.type = "_";
this.particleNum = arg.particleNum;
this.center = {x: 0, y: 0, z: 0};
this.targetCenter = arg.center;
this.radius = 0;
this.targetRadius = arg.radius;
this.degree = [];
this.freeDegreeSpeed = [];
for (var j = 0; j < this.particleNum; j++) {
this.degree[j] = {theta: 0, phi: 0};
this.freeDegreeSpeed[j] = {theta: Math.random() - 0.5, phi: Math.random() - 0.5};
}
this.charsMap = {};
for (var i in chars) {
var buffer = document.getElementById(i).getContext("2d").getImageData(0, 0, 100, 100).data;
this.charsMap[i] = [];
var self = this;
for (var j = 0; j < this.particleNum; j++) {
var redo = function () {
var theta = Math.floor(Math.random() * 100);
var phi = Math.floor(Math.random() * 100);
if (buffer[(theta * 400 + (phi * 4))] === 0) {
self.charsMap[i].push(
{
theta: theta - 50 + 360 * Math.round(Math.random() * 2) - 1,
phi: phi - 50 + 360 * Math.round(Math.random() * 2) - 1
}
);
} else {
redo();
}
};
redo();
}
}
this.charsMap["@"] = [];
for (var i = 0; i < this.particleNum; i++) {
this.charsMap["@"][i] = {theta: 360 * Math.random(), phi: 360 * Math.random()};
}
this.charsMap["_"] = [];
for (var i = 0; i < this.particleNum; i++) {
this.charsMap["_"][i] = {theta: 0, phi: 0};
}
this.veticies = [];
for (var i = 0; i < this.particleNum; i++) {
this.veticies[i] = new vertex3d({});
}
};
sphere.prototype = {
update: function () {
for (var i = 0; i < this.charsMap[this.type].length; i++) {
if (this.degree[i].theta >= 30 && this.degree[i].phi >= 30) {
this.flag = true;
break;
} else {
this.flag = false;
}
}
this.radius = this.radius + (this.targetRadius - this.radius) / 8;
this.center.x = this.center.x + (this.targetCenter.x - this.center.x) / 8;
this.center.y = this.center.y + (this.targetCenter.y - this.center.y) / 8;
this.center.z = this.center.z + (this.targetCenter.z - this.center.z) / 8;
for (var i = 0; i < this.charsMap[this.type].length; i++) {
if (this.type === "@") {
this.charsMap[this.type][i].theta += this.freeDegreeSpeed[i].theta;
this.charsMap[this.type][i].phi += this.freeDegreeSpeed[i].phi;
}
this.degree[i].theta = this.degree[i].theta + (this.charsMap[this.type][i].theta - this.degree[i].theta) / (4 + 20 * Math.random());
this.degree[i].phi = this.degree[i].phi + (this.charsMap[this.type][i].phi - this.degree[i].phi) / (4 + 20 * Math.random());
var getPosition;
if (vibrateFlag === true) {
getPosition = polarToRectangle(this.degree[i].theta + 90, this.degree[i].phi, this.radius + Math.random() * 10);
} else {
getPosition = polarToRectangle(this.degree[i].theta + 90, this.degree[i].phi, this.radius);
}
this.veticies[i].affineIn.vertex = {
x: getPosition.x,
y: getPosition.y,
z: getPosition.z
};
this.veticies[i].affineIn.position = {
x: this.center.x,
y: this.center.y,
z: this.center.z
};
this.veticies[i].vertexUpdate();
}
},
draw: function () {
if (this.flag === true) {
ctx.beginPath();
for (var i = 0; i < this.veticies.length; i++) {
for (var j = i; j < this.veticies.length; j++) {
var distance =
(this.veticies[i].affineOut.x - this.veticies[j].affineOut.x) * (this.veticies[i].affineOut.x - this.veticies[j].affineOut.x) +
(this.veticies[i].affineOut.y - this.veticies[j].affineOut.y) * (this.veticies[i].affineOut.y - this.veticies[j].affineOut.y);
if (distance <= this.radius * 3) {
ctx.moveTo(
this.veticies[i].affineOut.x,
this.veticies[i].affineOut.y
);
ctx.lineTo(
this.veticies[j].affineOut.x,
this.veticies[j].affineOut.y
);
}
}
}
ctx.closePath();
ctx.stroke();
}
}
};
/* class */
var sphereNum = 20;
var s = [];
/*-----------------------------------------------------*/
var setup = function () {
for (var i = 0; i < sphereNum; i++) {
s[i] = new sphere({radius: 100, particleNum: 250, center: {x: 70 * i - (sphereNum - 1) * 70 / 2, y: 0, z: 0}});
}
};
/*-----------------------------------------------------*/
var update = function () {
for (var i = 0; i < sphereNum; i++) {
s[i].update();
}
};
/*-----------------------------------------------------*/
var draw = function () {
for (var i = 0; i < sphereNum; i++) {
s[i].draw();
}
};
var chars = {
A: "",
B: "",
C: "",
D: "",
E: "",
F: "",
G: "",
H: "",
I: "",
J: "",
K: "",
L: "",
M: "",
N: "",
O: "",
P: "",
Q: "",
R: "",
S: "",
T: "",
U: "",
V: "",
W: "",
X: "",
Y: "",
Z: "",
"!": "",
"?": "",
"0": "",
"1": "",
"2": "",
"3": "",
"4": "",
"5": "",
"6": "",
"7": "",
"8": "",
"9": "",
".": "",
"ガ": "",
"ラ": "",
"ス": "",
};
//ガラス
var charsLength = 0;
var charCounter = 0;
var bufferImages = {};
var bufferCanvases = {};
for (var i in chars) {
charsLength++;
bufferImages[i] = new Image();
bufferImages[i].src = chars[i];
bufferImages[i].onload = function () {
charCounter++;
if (charCounter === charsLength) {
bufferDraw();
}
};
}
var bufferDraw = function () {
for (var i in chars) {
var canvas = document.createElement("canvas");
canvas.id = i;
document.getElementById("buffer").appendChild(canvas);
document.getElementById(i).getContext("2d").drawImage(
bufferImages[i],
0,
0,
100,
100
);
}
start();
};
var textChanger = function (text, sphereRadius, sphereSpace, unitTime) {
var changeIncrement = 0;
var charNum = text.length;
var center = [];
for (var i = 0; i < charNum; i++) {
center[i] = {x: sphereSpace * i - sphereSpace * (charNum - 1) / 2, y: 0, z: 0};
}
var changer = function () {
setTimeout(function () {
s[changeIncrement].type = text[changeIncrement];
s[changeIncrement].targetCenter = center[changeIncrement];
s[changeIncrement].targetRadius = sphereRadius;
changeIncrement++;
if (changeIncrement < charNum) {
changer();
}
}, unitTime);
};
for (var i = charNum; i < s.length; i++) {
s[i].type = "_";
}
changer();
};
var fullSet = function () {
var alpha = "ABCDEFGHIJKLMNOPQRSTUVWXYZ__?!1234567890";
var col = 10;
var colUnit = 80;
var rowUnit = 120;
for (var i = 0; i < alpha.length; i++) {
s[i].targetCenter = {
x: (i % 10) * colUnit - (col - 1) * colUnit / 2,
y: Math.floor(i / 10) * -rowUnit + 180,
z: 0
};
s[i].type = alpha[i];
}
};
var textSet = [
{text: "SHIMAPAN.ROCKS", sphereRadius: 140, sphereSpace: 80, unitTime: 100, time: 3000},
{text: "@@@@@@", sphereRadius: 100, sphereSpace: 130, unitTime: 50, time: 5000},
{text: "FOLTIK", sphereRadius: 130, sphereSpace: 100, unitTime: 100, time: 750},
{text: "@@@@@@@@", sphereRadius: 100, sphereSpace: 130, unitTime: 50, time: 3000},
{text: "WXYZZYRD", sphereRadius: 130, sphereSpace: 100, unitTime: 100, time: 750},
{text: "@@@", sphereRadius: 100, sphereSpace: 130, unitTime: 100, time: 3000},
{text: "ガラス", sphereRadius: 100, sphereSpace: 160, unitTime: 50, time: 750},
{text: "@@@@", sphereRadius: 100, sphereSpace: 130, unitTime: 100, time: 3000},
{text: "ADQT", sphereRadius: 140, sphereSpace: 130, unitTime: 100, time: 750},
{text: "@@@@@@@", sphereRadius: 100, sphereSpace: 130, unitTime: 50, time: 3000},
{text: "YOWGURT", sphereRadius: 130, sphereSpace: 120, unitTime: 100, time: 750}
];
var textSetChangerIncrement = 0;
var textSetChanger = function () {
setTimeout(function () {
textChanger(
textSet[textSetChangerIncrement].text,
textSet[textSetChangerIncrement].sphereRadius,
textSet[textSetChangerIncrement].sphereSpace,
textSet[textSetChangerIncrement].unitTime
);
textSetChangerIncrement++;
if (textSetChangerIncrement === textSet.length) {
textSetChangerIncrement = 0;
}
textSetChanger();
}, textSet[textSetChangerIncrement].time);
};
var vibrateCV = new closeValue(200, 500);
var invertCV = new closeValue(1000, 1200);
var start = function () {
setup();
setInterval(function () {
vibrateFlag = vibrateCV.execution() > 0.8;
strokeColor = "rgba(255, 255, 255, 0.1)";
backgroundColor = "rgba(0, 0, 0, 0)";
ctx.clearRect(0, 0, canvasWidth, canvasHeight);
ctx.fillStyle = backgroundColor;
ctx.fillRect(0, 0, window.innerWidth, window.innerHeight);
ctx.strokeStyle = strokeColor;
update();
draw();
}, 1000 / 60);
textSetChanger();
};
document.body.onmousemove = function (e) {
camera.rotate.x = (e.pageY / window.innerHeight * 180 - 90) * 0.2;
camera.rotate.y = (e.pageX / window.innerWidth * 180 - 90) * 0.2;
document.onmousedown = function () {
camera.zoom = Math.random() + 1
};
document.onmouseup = function () {
camera.zoom = 1
};
};