mirror of
https://github.com/Foltik/Shimapan
synced 2025-01-20 22:06:57 -05:00
599 lines
55 KiB
JavaScript
599 lines
55 KiB
JavaScript
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
|
|
};
|
|
}; |