2016-03-12 21:59:08 +08:00
|
|
|
/* Created by djazz
|
|
|
|
* http://djazz.mine.nu/
|
|
|
|
* @daniel_hede on Twitter
|
|
|
|
* daniel_hede in Minecraft
|
2016-01-03 14:08:23 +08:00
|
|
|
*
|
2016-03-12 21:59:08 +08:00
|
|
|
* https://github.com/daniel-j/chrome-minecraft-skin-preview/blob/master/skin.js
|
|
|
|
*
|
|
|
|
* Editted by printempw
|
|
|
|
* https://prinzeugen.net
|
|
|
|
* Added support for high-resolution skin & capes,
|
|
|
|
* also added external interfaces
|
2016-01-03 14:08:23 +08:00
|
|
|
*/
|
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
var MSP = (function (global, undefined) {
|
|
|
|
'use strict';
|
|
|
|
// shim layer with setTimeout fallback
|
|
|
|
window.requestAnimFrame = (function(){
|
|
|
|
return window.requestAnimationFrame ||
|
|
|
|
window.webkitRequestAnimationFrame ||
|
|
|
|
window.mozRequestAnimationFrame ||
|
|
|
|
window.oRequestAnimationFrame ||
|
|
|
|
window.msRequestAnimationFrame ||
|
|
|
|
function(/* function */ callback, /* DOMElement */ element) {
|
|
|
|
window.setTimeout(callback, 1000 / 60);
|
|
|
|
};
|
|
|
|
})();
|
|
|
|
|
|
|
|
var supportWebGL = !!global.WebGLRenderingContext && (!!global.document.createElement('canvas').getContext('experimental-webgl') || !!global.document.createElement('canvas').getContext('webgl'));
|
|
|
|
|
|
|
|
var container = global.document.querySelector('#skinpreview');
|
|
|
|
|
|
|
|
// Width and height
|
|
|
|
var cw = 300, ch = 300;
|
|
|
|
var tileUvWidth = 1/64;
|
|
|
|
var tileUvHeight = 1/32;
|
|
|
|
|
|
|
|
// Init <canvas>
|
|
|
|
var skinBig = global.document.createElement('canvas');
|
|
|
|
var sbc = skinBig.getContext('2d');
|
|
|
|
var sizeRatio = 8;
|
|
|
|
skinBig.width = 64*sizeRatio;
|
|
|
|
skinBig.height = 32*sizeRatio;
|
|
|
|
|
|
|
|
var skincanvas = global.document.createElement('canvas');
|
|
|
|
var skinc = skincanvas.getContext('2d');
|
|
|
|
skincanvas.width = 64;
|
|
|
|
skincanvas.height = 32;
|
|
|
|
var capecanvas = global.document.createElement('canvas');
|
|
|
|
var capec = capecanvas.getContext('2d');
|
|
|
|
capecanvas.width = 64;
|
|
|
|
capecanvas.height = 32;
|
|
|
|
|
|
|
|
// You can change these value via external interface
|
|
|
|
var isRotating = true;
|
|
|
|
var isPaused = false;
|
|
|
|
var isYfreezed = false;
|
|
|
|
var isFunnyRunning = false;
|
|
|
|
|
|
|
|
var getMaterial = function (img, trans) {
|
|
|
|
var material = new THREE.MeshBasicMaterial({
|
|
|
|
map: new THREE.Texture(
|
|
|
|
img,
|
|
|
|
new THREE.UVMapping(),
|
|
|
|
THREE.ClampToEdgeWrapping,
|
|
|
|
THREE.ClampToEdgeWrapping,
|
|
|
|
THREE.NearestFilter,
|
|
|
|
THREE.NearestFilter,
|
|
|
|
(trans? THREE.RGBAFormat : THREE.RGBFormat)
|
|
|
|
),
|
|
|
|
transparent: trans
|
|
|
|
});
|
|
|
|
material.map.needsUpdate = true;
|
|
|
|
return material;
|
|
|
|
};
|
|
|
|
var uvmap = function (mesh, face, x, y, w, h, rotateBy) {
|
|
|
|
if(!rotateBy) rotateBy = 0;
|
|
|
|
var uvs = mesh.geometry.faceVertexUvs[0][face];
|
|
|
|
var tileU = x;
|
|
|
|
var tileV = y;
|
|
|
|
|
|
|
|
uvs[ (0 + rotateBy) % 4 ].u = tileU * tileUvWidth;
|
|
|
|
uvs[ (0 + rotateBy) % 4 ].v = tileV * tileUvHeight;
|
|
|
|
uvs[ (1 + rotateBy) % 4 ].u = tileU * tileUvWidth;
|
|
|
|
uvs[ (1 + rotateBy) % 4 ].v = tileV * tileUvHeight + h * tileUvHeight;
|
|
|
|
uvs[ (2 + rotateBy) % 4 ].u = tileU * tileUvWidth + w * tileUvWidth;
|
|
|
|
uvs[ (2 + rotateBy) % 4 ].v = tileV * tileUvHeight + h * tileUvHeight;
|
|
|
|
uvs[ (3 + rotateBy) % 4 ].u = tileU * tileUvWidth + w * tileUvWidth;
|
|
|
|
uvs[ (3 + rotateBy) % 4 ].v = tileV * tileUvHeight;
|
|
|
|
};
|
|
|
|
var cubeFromPlanes = function (size, mat) {
|
|
|
|
var cube = new THREE.Object3D();
|
|
|
|
var meshes = [];
|
|
|
|
for(var i=0; i < 6; i++) {
|
|
|
|
var mesh = new THREE.Mesh(new THREE.PlaneGeometry(size, size), mat);
|
|
|
|
mesh.doubleSided = true;
|
|
|
|
cube.add(mesh);
|
|
|
|
meshes.push(mesh);
|
|
|
|
}
|
|
|
|
// Front
|
|
|
|
meshes[0].rotation.x = Math.PI/2;
|
|
|
|
meshes[0].rotation.z = -Math.PI/2;
|
|
|
|
meshes[0].position.x = size/2;
|
|
|
|
|
|
|
|
// Back
|
|
|
|
meshes[1].rotation.x = Math.PI/2;
|
|
|
|
meshes[1].rotation.z = Math.PI/2;
|
|
|
|
meshes[1].position.x = -size/2;
|
|
|
|
|
|
|
|
// Top
|
|
|
|
meshes[2].position.y = size/2;
|
|
|
|
|
|
|
|
// Bottom
|
|
|
|
meshes[3].rotation.y = Math.PI;
|
|
|
|
meshes[3].rotation.z = Math.PI;
|
|
|
|
meshes[3].position.y = -size/2;
|
|
|
|
|
|
|
|
// Left
|
|
|
|
meshes[4].rotation.x = Math.PI/2;
|
|
|
|
meshes[4].position.z = size/2;
|
|
|
|
|
|
|
|
// Right
|
|
|
|
meshes[5].rotation.x = -Math.PI/2;
|
|
|
|
meshes[5].rotation.y = Math.PI;
|
|
|
|
meshes[5].position.z = -size/2;
|
|
|
|
|
|
|
|
return cube;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
var charMaterial = getMaterial(skincanvas, false);
|
|
|
|
var charMaterialTrans = getMaterial(skincanvas, true);
|
|
|
|
var capeMaterial = getMaterial(capecanvas, false);
|
|
|
|
|
|
|
|
var camera = new THREE.PerspectiveCamera(35, cw / ch, 1, 1000);
|
|
|
|
camera.position.z = 50;
|
|
|
|
//camera.target.position.y = -2;
|
|
|
|
var scene = new THREE.Scene();
|
|
|
|
scene.add(camera);
|
|
|
|
|
|
|
|
var headgroup = new THREE.Object3D();
|
|
|
|
var upperbody = new THREE.Object3D();
|
|
|
|
|
|
|
|
//Light
|
|
|
|
var pointLightA = new THREE.DirectionalLight(0xFFFFFF, 1);
|
|
|
|
var pointLightB = new THREE.DirectionalLight(0x666666, 1);
|
|
|
|
var environmentalLight = new THREE.AmbientLight(0x555555);
|
|
|
|
pointLightA.position.set(100, 120, 80);
|
|
|
|
pointLightB.position.set(-100, -40, -80);
|
|
|
|
scene.add(pointLightA);
|
|
|
|
scene.add(pointLightB);
|
|
|
|
scene.add(environmentalLight);
|
|
|
|
|
|
|
|
// Left leg
|
|
|
|
var leftleggeo = new THREE.CubeGeometry(4, 12, 4);
|
|
|
|
for(var i=0; i < 8; i+=1) {
|
|
|
|
leftleggeo.vertices[i].y -= 6;
|
|
|
|
}
|
|
|
|
var leftleg = new THREE.Mesh(leftleggeo, charMaterial);
|
|
|
|
leftleg.position.z = -2;
|
|
|
|
leftleg.position.y = -6;
|
|
|
|
uvmap(leftleg, 0, 8, 20, -4, 12);
|
|
|
|
uvmap(leftleg, 1, 16, 20, -4, 12);
|
|
|
|
uvmap(leftleg, 2, 4, 16, 4, 4, 3);
|
|
|
|
uvmap(leftleg, 3, 8, 20, 4, -4, 1);
|
|
|
|
uvmap(leftleg, 4, 12, 20, -4, 12);
|
|
|
|
uvmap(leftleg, 5, 4, 20, -4, 12);
|
|
|
|
|
|
|
|
// Right leg
|
|
|
|
var rightleggeo = new THREE.CubeGeometry(4, 12, 4);
|
|
|
|
for(var i=0; i < 8; i+=1) {
|
|
|
|
rightleggeo.vertices[i].y -= 6;
|
2016-03-05 19:12:33 +08:00
|
|
|
}
|
2016-03-12 21:59:08 +08:00
|
|
|
var rightleg = new THREE.Mesh(rightleggeo, charMaterial);
|
|
|
|
rightleg.position.z = 2;
|
|
|
|
rightleg.position.y = -6;
|
|
|
|
uvmap(rightleg, 0, 4, 20, 4, 12);
|
|
|
|
uvmap(rightleg, 1, 12, 20, 4, 12);
|
|
|
|
uvmap(rightleg, 2, 8, 16, -4, 4, 3);
|
|
|
|
uvmap(rightleg, 3, 12, 20, -4, -4, 1);
|
|
|
|
uvmap(rightleg, 4, 0, 20, 4, 12);
|
|
|
|
uvmap(rightleg, 5, 8, 20, 4, 12);
|
|
|
|
|
|
|
|
// Body
|
|
|
|
var bodygeo = new THREE.CubeGeometry(4, 12, 8);
|
|
|
|
var bodymesh = new THREE.Mesh(bodygeo, charMaterial);
|
|
|
|
uvmap(bodymesh, 0, 20, 20, 8, 12);
|
|
|
|
uvmap(bodymesh, 1, 32, 20, 8, 12);
|
|
|
|
uvmap(bodymesh, 2, 20, 16, 8, 4, 1);
|
|
|
|
uvmap(bodymesh, 3, 28, 16, 8, 4, 3);
|
|
|
|
uvmap(bodymesh, 4, 16, 20, 4, 12);
|
|
|
|
uvmap(bodymesh, 5, 28, 20, 4, 12);
|
|
|
|
upperbody.add(bodymesh);
|
|
|
|
|
|
|
|
// Left arm
|
|
|
|
var leftarmgeo = new THREE.CubeGeometry(4, 12, 4);
|
|
|
|
for(var i=0; i < 8; i+=1) {
|
|
|
|
leftarmgeo.vertices[i].y -= 4;
|
|
|
|
}
|
|
|
|
var leftarm = new THREE.Mesh(leftarmgeo, charMaterial);
|
|
|
|
leftarm.position.z = -6;
|
|
|
|
leftarm.position.y = 4;
|
|
|
|
leftarm.rotation.x = Math.PI/32;
|
|
|
|
uvmap(leftarm, 0, 48, 20, -4, 12);
|
|
|
|
uvmap(leftarm, 1, 56, 20, -4, 12);
|
|
|
|
uvmap(leftarm, 2, 48, 16, -4, 4, 1);
|
|
|
|
uvmap(leftarm, 3, 52, 16, -4, 4, 3);
|
|
|
|
uvmap(leftarm, 4, 52, 20, -4, 12);
|
|
|
|
uvmap(leftarm, 5, 44, 20, -4, 12);
|
|
|
|
upperbody.add(leftarm);
|
|
|
|
|
|
|
|
// Right arm
|
|
|
|
var rightarmgeo = new THREE.CubeGeometry(4, 12, 4);
|
|
|
|
for(var i=0; i < 8; i+=1) {
|
|
|
|
rightarmgeo.vertices[i].y -= 4;
|
|
|
|
}
|
|
|
|
var rightarm = new THREE.Mesh(rightarmgeo, charMaterial);
|
|
|
|
rightarm.position.z = 6;
|
|
|
|
rightarm.position.y = 4;
|
|
|
|
rightarm.rotation.x = -Math.PI/32;
|
|
|
|
uvmap(rightarm, 0, 44, 20, 4, 12);
|
|
|
|
uvmap(rightarm, 1, 52, 20, 4, 12);
|
|
|
|
uvmap(rightarm, 2, 44, 16, 4, 4, 1);
|
|
|
|
uvmap(rightarm, 3, 48, 16, 4, 4, 3);
|
|
|
|
uvmap(rightarm, 4, 40, 20, 4, 12);
|
|
|
|
uvmap(rightarm, 5, 48, 20, 4, 12);
|
|
|
|
upperbody.add(rightarm);
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
//Head
|
|
|
|
var headgeo = new THREE.CubeGeometry(8, 8, 8);
|
|
|
|
var headmesh = new THREE.Mesh(headgeo, charMaterial);
|
|
|
|
headmesh.position.y = 2;
|
|
|
|
uvmap(headmesh, 0, 8, 8, 8, 8);
|
|
|
|
uvmap(headmesh, 1, 24, 8, 8, 8);
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
uvmap(headmesh, 2, 8, 0, 8, 8, 1);
|
|
|
|
uvmap(headmesh, 3, 16, 0, 8, 8, 1);
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
uvmap(headmesh, 4, 0, 8, 8, 8);
|
|
|
|
uvmap(headmesh, 5, 16, 8, 8, 8);
|
|
|
|
headgroup.add(headmesh);
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
// Helmet/hat
|
|
|
|
var helmetgeo = new THREE.CubeGeometry(9, 9, 9);
|
|
|
|
var helmetmesh = new THREE.Mesh(helmetgeo, charMaterialTrans);
|
|
|
|
helmetmesh.doubleSided = true;
|
|
|
|
helmetmesh.position.y = 2;
|
|
|
|
uvmap(helmetmesh, 0, 32+8, 8, 8, 8);
|
|
|
|
uvmap(helmetmesh, 1, 32+24, 8, 8, 8);
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
uvmap(helmetmesh, 2, 32+8, 0, 8, 8, 1);
|
|
|
|
uvmap(helmetmesh, 3, 32+16, 0, 8, 8, 3);
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
uvmap(helmetmesh, 4, 32+0, 8, 8, 8);
|
|
|
|
uvmap(helmetmesh, 5, 32+16, 8, 8, 8);
|
|
|
|
headgroup.add(helmetmesh);
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
var helmet = cubeFromPlanes(9, charMaterialTrans);
|
|
|
|
helmet.position.y = 2;
|
|
|
|
uvmap(helmet.children[0], 0, 32+8, 8, 8, 8);
|
|
|
|
uvmap(helmet.children[1], 0, 32+24, 8, 8, 8);
|
|
|
|
uvmap(helmet.children[2], 0, 32+8, 0, 8, 8, 1);
|
|
|
|
uvmap(helmet.children[3], 0, 32+16, 0, 8, 8, 3);
|
|
|
|
uvmap(helmet.children[4], 0, 32+0, 8, 8, 8);
|
|
|
|
uvmap(helmet.children[5], 0, 32+16, 8, 8, 8);
|
2016-03-05 19:12:33 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
headgroup.add(helmet);
|
2016-03-05 19:12:33 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
/* ===== Ears Start (作者的彩蛋,没啥卵用) ===== */
|
|
|
|
var ears = new THREE.Object3D();
|
2016-03-05 19:12:33 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
var eargeo = new THREE.CubeGeometry(1, (9/8)*6, (9/8)*6);
|
|
|
|
var leftear = new THREE.Mesh(eargeo, charMaterial);
|
|
|
|
var rightear = new THREE.Mesh(eargeo, charMaterial);
|
2016-03-05 19:12:33 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
leftear.position.y = 2+(9/8)*5;
|
|
|
|
rightear.position.y = 2+(9/8)*5;
|
|
|
|
leftear.position.z = -(9/8)*5;
|
|
|
|
rightear.position.z = (9/8)*5;
|
2016-03-05 19:12:33 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
// Right ear share same geometry, same uv-maps
|
|
|
|
uvmap(leftear, 0, 25, 1, 6, 6); // Front side
|
|
|
|
uvmap(leftear, 1, 32, 1, 6, 6); // Back side
|
2016-03-05 19:12:33 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
uvmap(leftear, 2, 25, 0, 6, 1, 1); // Top edge
|
|
|
|
uvmap(leftear, 3, 31, 0, 6, 1, 1); // Bottom edge
|
2016-03-05 19:12:33 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
uvmap(leftear, 4, 24, 1, 1, 6); // Left edge
|
|
|
|
uvmap(leftear, 5, 31, 1, 1, 6); // Right edge
|
2016-03-05 19:12:33 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
ears.add(leftear);
|
|
|
|
ears.add(rightear);
|
2016-03-05 19:12:33 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
leftear.visible = rightear.visible = false;
|
2016-03-05 19:12:33 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
headgroup.add(ears);
|
|
|
|
headgroup.position.y = 8;
|
|
|
|
/* ================ Ears End ================== */
|
|
|
|
|
|
|
|
// Init cape
|
|
|
|
var capeOrigo = new THREE.Object3D();
|
|
|
|
var capegeo = new THREE.CubeGeometry(1, 16, 10);
|
|
|
|
var capemesh = new THREE.Mesh(capegeo, capeMaterial);
|
|
|
|
capemesh.position.y = -8;
|
|
|
|
capemesh.visible = false;
|
|
|
|
|
|
|
|
uvmap(capemesh, 0, 1, 1, 10, 16); // Front side
|
|
|
|
uvmap(capemesh, 1, 12, 1, 10, 16); // Back side
|
|
|
|
|
|
|
|
uvmap(capemesh, 2, 1, 0, 10, 1); // Top edge
|
|
|
|
uvmap(capemesh, 3, 11, 0, 10, 1, 1); // Bottom edge
|
|
|
|
|
|
|
|
uvmap(capemesh, 4, 0, 1, 1, 16); // Left edge
|
|
|
|
uvmap(capemesh, 5, 11, 1, 1, 16); // Right edge
|
|
|
|
|
|
|
|
|
|
|
|
capeOrigo.rotation.y = Math.PI;
|
|
|
|
|
|
|
|
capeOrigo.position.x = -2;
|
|
|
|
capeOrigo.position.y = 6;
|
|
|
|
|
|
|
|
capeOrigo.add(capemesh);
|
|
|
|
|
|
|
|
|
|
|
|
var playerModel = new THREE.Object3D();
|
|
|
|
|
|
|
|
playerModel.add(leftleg);
|
|
|
|
playerModel.add(rightleg);
|
|
|
|
|
|
|
|
playerModel.add(upperbody);
|
|
|
|
playerModel.add(headgroup);
|
|
|
|
|
|
|
|
playerModel.add(capeOrigo);
|
|
|
|
|
|
|
|
playerModel.position.y = 6;
|
|
|
|
|
|
|
|
|
|
|
|
var playerGroup = new THREE.Object3D();
|
|
|
|
|
|
|
|
playerGroup.add(playerModel);
|
|
|
|
|
|
|
|
|
|
|
|
scene.add(playerGroup);
|
|
|
|
|
|
|
|
|
|
|
|
var mouseX = 0;
|
|
|
|
var mouseY = 0.1;
|
|
|
|
var originMouseX = 0;
|
|
|
|
var originMouseY = 0;
|
|
|
|
|
|
|
|
var rad = 0;
|
|
|
|
|
|
|
|
var isMouseOver = false;
|
|
|
|
var isMouseDown = false;
|
|
|
|
|
|
|
|
var counter = 0;
|
|
|
|
var firstRender = true;
|
|
|
|
|
|
|
|
var startTime = Date.now();
|
|
|
|
var pausedTime = 0;
|
|
|
|
|
|
|
|
var render = function () {
|
|
|
|
requestAnimFrame(render, renderer.domElement);
|
|
|
|
var oldRad = rad;
|
|
|
|
|
|
|
|
var time = (Date.now() - startTime)/1000;
|
|
|
|
|
|
|
|
if(!isMouseDown) {
|
|
|
|
//mouseX*=0.95;
|
|
|
|
if(!isYfreezed) {
|
|
|
|
mouseY*=0.97;
|
|
|
|
}
|
|
|
|
if(isRotating) {
|
|
|
|
rad += 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
rad = mouseX;
|
|
|
|
}
|
|
|
|
if(mouseY > 500) {
|
|
|
|
mouseY = 500;
|
|
|
|
}
|
|
|
|
else if(mouseY < -500) {
|
|
|
|
mouseY = -500;
|
|
|
|
}
|
|
|
|
camera.position.x = -Math.cos(rad/(cw/2)+(Math.PI/0.9));
|
|
|
|
camera.position.z = -Math.sin(rad/(cw/2)+(Math.PI/0.9));
|
|
|
|
camera.position.y = (mouseY/(ch/2))*1.5+0.2;
|
|
|
|
camera.position.setLength(70);
|
|
|
|
camera.lookAt(new THREE.Vector3(0, 1.5, 0));
|
|
|
|
|
|
|
|
|
|
|
|
if(!isPaused) {
|
|
|
|
counter+=0.01;
|
|
|
|
headgroup.rotation.y = Math.sin(time*1.5)/5;
|
|
|
|
headgroup.rotation.z = Math.sin(time)/6;
|
|
|
|
|
|
|
|
if(isFunnyRunning) {
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
rightarm.rotation.z = 2 * Math.cos(0.6662 * time*10 + Math.PI);
|
|
|
|
rightarm.rotation.x = 1 * (Math.cos(0.2812 * time*10) - 1);
|
|
|
|
leftarm.rotation.z = 2 * Math.cos(0.6662 * time*10);
|
|
|
|
leftarm.rotation.x = 1 * (Math.cos(0.2312 * time*10) + 1);
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
rightleg.rotation.z = 1.4 * Math.cos(0.6662 * time*10);
|
|
|
|
leftleg.rotation.z = 1.4 * Math.cos(0.6662 * time*10 + Math.PI);
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
playerGroup.position.y = -6+1 * Math.cos(0.6662 * time*10 * 2); // Jumping
|
|
|
|
playerGroup.position.z = 0.15 * Math.cos(0.6662 * time*10); // Dodging when running
|
|
|
|
playerGroup.rotation.x = 0.01 * Math.cos(0.6662 * time*10 + Math.PI); // Slightly tilting when running
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
capeOrigo.rotation.z = 0.1 * Math.sin(0.6662 * time*10 * 2)+Math.PI/2.5;
|
|
|
|
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
leftarm.rotation.z = -Math.sin(time*3)/2;
|
|
|
|
leftarm.rotation.x = (Math.cos(time*3)+Math.PI/2)/30;
|
|
|
|
rightarm.rotation.z = Math.sin(time*3)/2;
|
|
|
|
rightarm.rotation.x = -(Math.cos(time*3)+Math.PI/2)/30;
|
|
|
|
|
|
|
|
leftleg.rotation.z = Math.sin(time*3)/3;
|
|
|
|
rightleg.rotation.z = -Math.sin(time*3)/3;
|
|
|
|
capeOrigo.rotation.z = Math.sin(time*2)/15+Math.PI/15;
|
|
|
|
|
|
|
|
playerGroup.position.y = -6; // Not jumping
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
renderer.render(scene, camera);
|
|
|
|
};
|
|
|
|
if(supportWebGL) {
|
|
|
|
var renderer = new THREE.WebGLRenderer({antialias: true,precision:'highp'});
|
2016-02-28 14:20:41 +08:00
|
|
|
}
|
2016-03-12 21:59:08 +08:00
|
|
|
else {
|
|
|
|
var renderer = new THREE.CanvasRenderer({antialias: true});
|
|
|
|
}
|
|
|
|
var threecanvas = renderer.domElement;
|
|
|
|
renderer.setSize(cw, ch);
|
|
|
|
//renderer.setClearColorHex(0x000000, 0.25);
|
|
|
|
|
|
|
|
// container.appendChild(threecanvas);
|
|
|
|
|
|
|
|
var onMouseMove = function (e) {
|
|
|
|
if(isMouseDown) {
|
|
|
|
mouseX = (e.pageX - threecanvas.offsetLeft - originMouseX);
|
|
|
|
mouseY = (e.pageY - threecanvas.offsetTop - originMouseY);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
threecanvas.addEventListener('mousedown', function (e) {
|
2016-03-05 19:12:33 +08:00
|
|
|
e.preventDefault();
|
2016-03-12 21:59:08 +08:00
|
|
|
originMouseX = (e.pageX - threecanvas.offsetLeft) - rad;
|
|
|
|
originMouseY = (e.pageY - threecanvas.offsetTop) - mouseY;
|
2016-03-05 19:12:33 +08:00
|
|
|
isMouseDown = true;
|
2016-03-12 21:59:08 +08:00
|
|
|
isMouseOver = true;
|
|
|
|
onMouseMove(e);
|
2016-03-05 19:12:33 +08:00
|
|
|
}, false);
|
2016-03-12 21:59:08 +08:00
|
|
|
global.addEventListener('mouseup', function (e) {
|
2016-03-05 19:12:33 +08:00
|
|
|
isMouseDown = false;
|
|
|
|
}, false);
|
2016-03-12 21:59:08 +08:00
|
|
|
global.addEventListener('mousemove', onMouseMove, false);
|
|
|
|
threecanvas.addEventListener('mouseout', function (e) {
|
|
|
|
isMouseOver = false;
|
2016-03-05 19:12:33 +08:00
|
|
|
}, false);
|
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
// container.appendChild(skinBig);
|
|
|
|
|
|
|
|
/* ============================================================ */
|
|
|
|
|
|
|
|
// 默认开始渲染的地方
|
|
|
|
render();
|
|
|
|
|
|
|
|
var skin = new Image();
|
|
|
|
|
|
|
|
skin.onload = function () {
|
|
|
|
|
|
|
|
/* 高清皮肤支持 */
|
|
|
|
skincanvas.width = skin.width;
|
|
|
|
skincanvas.height = skin.height;
|
|
|
|
|
|
|
|
skinc.clearRect(0, 0, 64, 32);
|
|
|
|
|
|
|
|
skinc.drawImage(skin, 0, 0);
|
|
|
|
|
|
|
|
var imgdata = skinc.getImageData(0, 0, 64, 32);
|
|
|
|
var pixels = imgdata.data;
|
|
|
|
|
|
|
|
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
sbc.clearRect(0, 0, skinBig.width, skinBig.height);
|
|
|
|
sbc.save();
|
2016-03-05 21:03:22 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
var isOnecolor = true;
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
var colorCheckAgainst = [40, 0];
|
|
|
|
var colorIndex = (colorCheckAgainst[0]+colorCheckAgainst[1]*64)*4;
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
var isPixelDifferent = function (x, y) {
|
|
|
|
if(pixels[(x+y*64)*4+0] !== pixels[colorIndex+0] || pixels[(x+y*64)*4+1] !== pixels[colorIndex+1] || pixels[(x+y*64)*4+2] !== pixels[colorIndex+2] || pixels[(x+y*64)*4+3] !== pixels[colorIndex+3]) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
};
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
// Check if helmet/hat is a solid color
|
|
|
|
// Bottom row
|
|
|
|
for(var i=32; i < 64; i+=1) {
|
|
|
|
for(var j=8; j < 16; j+=1) {
|
|
|
|
if(isPixelDifferent(i, j)) {
|
|
|
|
isOnecolor = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(!isOnecolor) {
|
|
|
|
break;
|
|
|
|
}
|
2016-02-28 14:20:41 +08:00
|
|
|
}
|
2016-03-12 21:59:08 +08:00
|
|
|
if(!isOnecolor) {
|
|
|
|
// Top row
|
|
|
|
for(var i=40; i < 56; i+=1) {
|
|
|
|
for(var j=0; j < 8; j+=1) {
|
|
|
|
if(isPixelDifferent(i, j)) {
|
|
|
|
isOnecolor = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(!isOnecolor) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2016-03-05 19:12:33 +08:00
|
|
|
}
|
2016-03-12 21:59:08 +08:00
|
|
|
|
|
|
|
for(var i=0; i < 64; i+=1) {
|
|
|
|
for(var j=0; j < 32; j+=1) {
|
|
|
|
|
|
|
|
if(isOnecolor && ((i >= 32 && i < 64 && j >= 8 && j < 16) || (i >= 40 && i < 56 && j >= 0 && j < 8))) {
|
|
|
|
pixels[(i+j*64)*4+3] = 0;
|
|
|
|
}
|
|
|
|
sbc.fillStyle = 'rgba('+pixels[(i+j*64)*4+0]+', '+pixels[(i+j*64)*4+1]+', '+pixels[(i+j*64)*4+2]+', '+pixels[(i+j*64)*4+3]/255+')';
|
|
|
|
sbc.fillRect(i*sizeRatio, j*sizeRatio, sizeRatio, sizeRatio);
|
|
|
|
}
|
2016-02-28 14:20:41 +08:00
|
|
|
}
|
2016-03-12 21:59:08 +08:00
|
|
|
sbc.restore();
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
skinc.putImageData(imgdata, 0, 0);
|
|
|
|
|
|
|
|
charMaterial.map.needsUpdate = true;
|
|
|
|
charMaterialTrans.map.needsUpdate = true;
|
|
|
|
};
|
|
|
|
|
|
|
|
var cape = new Image();
|
2016-02-28 14:20:41 +08:00
|
|
|
|
2016-03-12 21:59:08 +08:00
|
|
|
cape.onload = function () {
|
|
|
|
/* 高清披风支持 */
|
|
|
|
capecanvas.width = cape.width;
|
|
|
|
capecanvas.height = cape.height;
|
|
|
|
|
|
|
|
capec.clearRect(0, 0, 64, 32);
|
|
|
|
|
|
|
|
capec.drawImage(cape, 0, 0);
|
|
|
|
|
|
|
|
capeMaterial.map.needsUpdate = true;
|
|
|
|
capemesh.visible = true;
|
|
|
|
};
|
|
|
|
cape.onerror = function () {
|
|
|
|
capemesh.visible = false;
|
|
|
|
};
|
|
|
|
|
|
|
|
threecanvas.addEventListener('dragenter', function (e) {
|
|
|
|
e.stopPropagation();
|
|
|
|
e.preventDefault();
|
|
|
|
threecanvas.className = "dragenter";
|
|
|
|
}, false);
|
|
|
|
threecanvas.addEventListener('dragleave', function (e) {
|
|
|
|
e.stopPropagation();
|
|
|
|
e.preventDefault();
|
|
|
|
threecanvas.className = "";
|
|
|
|
}, false);
|
|
|
|
threecanvas.addEventListener('dragover', function (e) {
|
|
|
|
e.stopPropagation();
|
|
|
|
e.preventDefault();
|
|
|
|
}, false);
|
|
|
|
threecanvas.addEventListener('drop', function (e) {
|
|
|
|
e.stopPropagation();
|
|
|
|
e.preventDefault();
|
|
|
|
threecanvas.className = "";
|
|
|
|
|
|
|
|
var dt = e.dataTransfer;
|
|
|
|
var files = dt.files;
|
|
|
|
handleFiles(files);
|
|
|
|
}, false);
|
|
|
|
|
|
|
|
// 定义外部接口
|
|
|
|
return {
|
|
|
|
getStatus: function(key) {
|
|
|
|
switch(key) {
|
|
|
|
case "rotation":
|
|
|
|
return isRotating;
|
|
|
|
break;
|
|
|
|
case "movements":
|
|
|
|
return !isPaused;
|
|
|
|
break;
|
|
|
|
case "camera":
|
|
|
|
return !isYfreezed;
|
|
|
|
break;
|
|
|
|
case "running":
|
|
|
|
return isFunnyRunning;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
},
|
|
|
|
setStatus: function(key, value) {
|
|
|
|
switch(key) {
|
|
|
|
case "rotation":
|
|
|
|
isRotating = value;
|
|
|
|
return isRotating;
|
|
|
|
break;
|
|
|
|
case "movements":
|
|
|
|
isPaused = !value;
|
|
|
|
// \o/
|
|
|
|
if(isPaused) {
|
|
|
|
pausedTime = Date.now() - startTime;
|
|
|
|
} else {
|
|
|
|
startTime = Date.now() - pausedTime;
|
|
|
|
}
|
|
|
|
return !isPaused;
|
|
|
|
break;
|
|
|
|
case "camera":
|
|
|
|
isYfreezed = !value;
|
|
|
|
return !isYfreezed;
|
|
|
|
break;
|
|
|
|
case "running":
|
|
|
|
isFunnyRunning = value;
|
|
|
|
return isFunnyRunning;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
},
|
|
|
|
showCape: function(val) {
|
|
|
|
if(val) {
|
|
|
|
capeOrigo.add(capemesh);
|
|
|
|
} else {
|
|
|
|
capeOrigo.remove(capemesh);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
changeSkin: function(url) {
|
|
|
|
skin.src = url;
|
|
|
|
},
|
|
|
|
changeCape: function(url) {
|
|
|
|
cape.src = url;
|
|
|
|
},
|
|
|
|
get3dSkinCanvas: function(width, height) {
|
|
|
|
renderer.setSize(width, height);
|
|
|
|
return threecanvas;
|
|
|
|
},
|
|
|
|
// 作者的彩蛋(笑
|
|
|
|
setEars: function (val) {
|
|
|
|
leftear.visible = rightear.visible = val;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}(this));
|