From ee427ccca4e0f326987002b9538623554f2897fc Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Wed, 2 Oct 2019 06:25:07 -0400 Subject: [PATCH 001/562] Add initial RiggedModel hack, not accurate or good as a basis for anything --- app.html | 184 +++++++++++++------------------------------------------ 1 file changed, 42 insertions(+), 142 deletions(-) diff --git a/app.html b/app.html index 4241cb1..cb11323 100644 --- a/app.html +++ b/app.html @@ -352,6 +352,22 @@ subtree: true, }); +class RiggedModel { + constructor(avatarMesh) { + this.avatarMesh = avatarMesh; + } + setState(hmd, gamepads) { + this.avatarMesh.position.fromArray(hmd.position).sub(localVector.set(0, 1.5, 0)); + localQuaternion.fromArray(hmd.quaternion); + localEuler.setFromQuaternion(localQuaternion, localEuler.order); + localEuler.y += Math.PI; + localEuler.x *= -1; + this.avatarMesh.quaternion.setFromEuler(localEuler); + } + update() { + // console.log('per-frame update called'); + } +} const _makePeerConnection = peerConnectionId => { const peerConnectionConfig = { iceServers: [ @@ -365,7 +381,7 @@ console.log('got track', e); }; - let skinMesh = null; + let avatarMesh = null; const sendChannel = peerConnection.createDataChannel('sendChannel'); peerConnection.sendChannel = sendChannel; let pingInterval = 0; @@ -373,9 +389,19 @@ sendChannel.onopen = () => { // console.log('data channel local open'); - skinMesh = _makeSkinMesh(); - skinMesh.setSkinUrl(DEFAULT_SKIN_URL); - skinMeshes.push(skinMesh); + avatarMesh = new THREE.Object3D(); + avatarMesh.riggedModel = null; + const loader = new THREE.GLTFLoader(); + loader.load('https://modulesio.github.io/models/miku.glb', object => { + if (avatarMesh) { + avatarMesh.riggedModel = new RiggedModel(object.scene); + avatarMesh.add(object.scene); + } + }, xhr => {}, err => { + console.warn(err); + }); + scene.add(avatarMesh); + avatarMeshes.push(avatarMesh); peerConnection.open = true; @@ -459,9 +485,9 @@ const data = JSON.parse(e.data); const {method} = data; if (method === 'pose') { - if (skinMesh) { + if (avatarMesh && avatarMesh.riggedModel) { const {hmd, gamepads} = data; - skinMesh.setState(hmd, gamepads); + avatarMesh.riggedModel.setState(hmd, gamepads); } } else if (landState && method === 'initState') { const {state} = data; @@ -507,10 +533,10 @@ if (index !== -1) { peerConnections.splice(index, 1); } - if (skinMesh) { - skinMesh.destroy(); - skinMeshes.splice(skinMeshes.indexOf(skinMesh), 1); - skinMesh = null; + if (avatarMesh) { + scene.remove(avatarMesh); + avatarMeshes.splice(avatarMeshes.indexOf(avatarMesh), 1); + avatarMesh = null; } if (landState) { // remove owned xr-iframes @@ -6095,12 +6121,13 @@ }; _updateSceneMeshes(); */ - const _updateSkinMeshes = () => { - for (let i = 0; i < skinMeshes.length; i++) { - skinMeshes[i].update(); + const _updateAvatarMeshes = () => { + for (let i = 0; i < avatarMeshes.length; i++) { + const avatarMesh = avatarMeshes[i]; + avatarMesh.riggedModel && avatarMesh.riggedModel.update(); } }; - _updateSkinMeshes(); + _updateAvatarMeshes(); const _handleTrigger = (gamepad, i, pressed, lastPressed) => { const start = pressed && !lastPressed; @@ -6870,139 +6897,12 @@ console.warn(err.stack); }); -const skinMeshes = []; +const avatarMeshes = []; const _mod = (v, d) => { const n = v % d; return n < 0 ? (d + n) : n; }; const _angleDiff = (a, b) => _mod((b - a) + Math.PI, Math.PI * 2) - Math.PI; -const _makeSkinMesh = () => { - const object = {}; - let onsetstate = null; - const uniforms = THREE.UniformsUtils.clone(skin.SKIN_SHADER.uniforms); - object.setSkinUrl = skinUrl => { - if (skinUrl) { - const mesh = skin({ - limbs: true, - }); - mesh.frustumCulled = false; - - mesh.onBeforeRender = (onBeforeRender => function() { - mesh.material.uniforms.headRotation.value.copy(uniforms.headRotation.value); - mesh.material.uniforms.leftArmRotation.value.copy(uniforms.leftArmRotation.value); - mesh.material.uniforms.rightArmRotation.value.copy(uniforms.rightArmRotation.value); - mesh.material.uniforms.theta.value = uniforms.theta.value; - mesh.material.uniforms.headVisible.value = uniforms.headVisible.value; - mesh.material.uniforms.hit.value = uniforms.hit.value; - - onBeforeRender.apply(this, arguments); - })(mesh.onBeforeRender); - - return new Promise((accept, reject) => { - const skinImg = new Image(); - skinImg.crossOrigin = 'Anonymous'; - skinImg.src = skinUrl; - skinImg.onload = () => { - accept(skinImg); - }; - skinImg.onerror = err => { - reject(err); - }; - }) - .then(skinImg => { - mesh.setImage(skinImg); - - onsetstate = (hmd, gamepads) => { - const hmdPosition = localVector.fromArray(hmd.position); - const hmdQuaternion = localQuaternion.fromArray(hmd.quaternion); - - const hmdEuler = localEuler.setFromQuaternion(hmdQuaternion, localEuler.order); - const playerEuler = localEuler2.setFromQuaternion(mesh.quaternion, localEuler2.order); - const angleDiff = _angleDiff(hmdEuler.y, playerEuler.y); - const angleDiffAbs = Math.abs(angleDiff); - if (angleDiffAbs > Math.PI / 2) { - playerEuler.y += (angleDiffAbs - (Math.PI / 2)) * (angleDiff < 0 ? 1 : -1); - mesh.quaternion.setFromEuler(playerEuler); - } - - const oldWorldPosition = mesh.getWorldPosition(localVector2); - mesh.position.copy(hmdPosition) - .sub(mesh.eye.getWorldPosition(localVector3)) - .add(oldWorldPosition); - - const playerQuaternionInverse = localQuaternion2.copy(mesh.quaternion).inverse(); - mesh.head.quaternion.copy(playerQuaternionInverse).multiply(hmdQuaternion); - mesh.updateMatrixWorld(); - - const headQuaternionInverse = localQuaternion3.copy(mesh.head.quaternion).inverse(); - uniforms.headRotation.value.set(headQuaternionInverse.x, headQuaternionInverse.y, headQuaternionInverse.z, headQuaternionInverse.w); - - for (let i = 0; i < 2; i++) { - const armRotation = uniforms[i === 0 ? 'leftArmRotation' : 'rightArmRotation']; - - const gamepad = gamepads[i]; - if (gamepad.visible) { - const gamepadPosition = localVector.fromArray(gamepad.position); - const gamepadQuaternion = localQuaternion.fromArray(gamepad.quaternion); - - const armQuaternionInverse = localQuaternion3.setFromRotationMatrix( - localMatrix.lookAt( - mesh.arms[i === 0 ? 'left' : 'right'] - .getWorldPosition(localVector2), - gamepadPosition, - localVector3 - .set(0, 1, 0) - .applyQuaternion(gamepadQuaternion) - ) - ) - .multiply(armQuaternionOffset) - .premultiply(playerQuaternionInverse) - .inverse(); - armRotation.value.set(armQuaternionInverse.x, armQuaternionInverse.y, armQuaternionInverse.z, armQuaternionInverse.w); - // console.log('arm rotation', i, armRotation.value.toArray().join(',')); - } else { - armRotation.value.set(0, 0, 0, 1); - } - } - }; - - if (object.skinMesh) { - object.skinMesh.parent.remove(object.skinMesh); - object.skinMesh = null; - } - - scene.add(mesh); - object.skinMesh = mesh; - }); - } else { - onsetstate = null; - - if (object.skinMesh) { - object.skinMesh.parent.remove(object.skinMesh); - object.skinMesh = null; - } - } - }; - object.setState = (hmd, gamepads) => { - onsetstate && onsetstate(hmd, gamepads); - }; - object.animate = thetaValue => { - uniforms.theta.value = thetaValue; - }; - object.update = () => { - // XXX - }; - object.destroy = () => { - onsetstate = null; - - if (object.skinMesh) { - object.skinMesh.parent.remove(object.skinMesh); - object.skinMesh = null; - } - }; - object.skinMesh = null; - return object; -}; scene.add(defaultGltf); const _loadDefaultGltf = () => { From a312df693e321b9e501ba49f8eb17cd8cc33656a Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Wed, 9 Oct 2019 00:19:28 -0400 Subject: [PATCH 002/562] Add three-ik.js --- three-ik.js | 1183 +++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 1183 insertions(+) create mode 100644 three-ik.js diff --git a/three-ik.js b/three-ik.js new file mode 100644 index 0000000..5af851c --- /dev/null +++ b/three-ik.js @@ -0,0 +1,1183 @@ +(function (global, factory) { + typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('three')) : + typeof define === 'function' && define.amd ? define(['exports', 'three'], factory) : + (factory((global.IK = {}),global.THREE)); +}(this, (function (exports,three) { 'use strict'; + +var t1 = new three.Vector3(); +var t2 = new three.Vector3(); +var t3 = new three.Vector3(); +var m1 = new three.Matrix4(); +function getWorldPosition(object, target) { + return target.setFromMatrixPosition(object.matrixWorld); +} + +function getCentroid(positions, target) { + target.set(0, 0, 0); + var _iteratorNormalCompletion = true; + var _didIteratorError = false; + var _iteratorError = undefined; + try { + for (var _iterator = positions[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) { + var position = _step.value; + target.add(position); + } + } catch (err) { + _didIteratorError = true; + _iteratorError = err; + } finally { + try { + if (!_iteratorNormalCompletion && _iterator.return) { + _iterator.return(); + } + } finally { + if (_didIteratorError) { + throw _iteratorError; + } + } + } + target.divideScalar(positions.length); + return target; +} +function setQuaternionFromDirection(direction, up, target) { + var x = t1; + var y = t2; + var z = t3; + var m = m1; + var el = m1.elements; + z.copy(direction); + x.crossVectors(up, z); + if (x.lengthSq() === 0) { + if (Math.abs(up.z) === 1) { + z.x += 0.0001; + } else { + z.z += 0.0001; + } + z.normalize(); + x.crossVectors(up, z); + } + x.normalize(); + y.crossVectors(z, x); + el[0] = x.x;el[4] = y.x;el[8] = z.x; + el[1] = x.y;el[5] = y.y;el[9] = z.y; + el[2] = x.z;el[6] = y.z;el[10] = z.z; + target.setFromRotationMatrix(m); +} +function transformPoint(vector, matrix, target) { + var e = matrix.elements; + var x = vector.x * e[0] + vector.y * e[4] + vector.z * e[8] + e[12]; + var y = vector.x * e[1] + vector.y * e[5] + vector.z * e[9] + e[13]; + var z = vector.x * e[2] + vector.y * e[6] + vector.z * e[10] + e[14]; + var w = vector.x * e[3] + vector.y * e[7] + vector.z * e[11] + e[15]; + target.set(x / w, y / w, z / w); +} + +var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { + return typeof obj; +} : function (obj) { + return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; +}; + + + + + +var asyncGenerator = function () { + function AwaitValue(value) { + this.value = value; + } + + function AsyncGenerator(gen) { + var front, back; + + function send(key, arg) { + return new Promise(function (resolve, reject) { + var request = { + key: key, + arg: arg, + resolve: resolve, + reject: reject, + next: null + }; + + if (back) { + back = back.next = request; + } else { + front = back = request; + resume(key, arg); + } + }); + } + + function resume(key, arg) { + try { + var result = gen[key](arg); + var value = result.value; + + if (value instanceof AwaitValue) { + Promise.resolve(value.value).then(function (arg) { + resume("next", arg); + }, function (arg) { + resume("throw", arg); + }); + } else { + settle(result.done ? "return" : "normal", result.value); + } + } catch (err) { + settle("throw", err); + } + } + + function settle(type, value) { + switch (type) { + case "return": + front.resolve({ + value: value, + done: true + }); + break; + + case "throw": + front.reject(value); + break; + + default: + front.resolve({ + value: value, + done: false + }); + break; + } + + front = front.next; + + if (front) { + resume(front.key, front.arg); + } else { + back = null; + } + } + + this._invoke = send; + + if (typeof gen.return !== "function") { + this.return = undefined; + } + } + + if (typeof Symbol === "function" && Symbol.asyncIterator) { + AsyncGenerator.prototype[Symbol.asyncIterator] = function () { + return this; + }; + } + + AsyncGenerator.prototype.next = function (arg) { + return this._invoke("next", arg); + }; + + AsyncGenerator.prototype.throw = function (arg) { + return this._invoke("throw", arg); + }; + + AsyncGenerator.prototype.return = function (arg) { + return this._invoke("return", arg); + }; + + return { + wrap: function (fn) { + return function () { + return new AsyncGenerator(fn.apply(this, arguments)); + }; + }, + await: function (value) { + return new AwaitValue(value); + } + }; +}(); + + + + + +var classCallCheck = function (instance, Constructor) { + if (!(instance instanceof Constructor)) { + throw new TypeError("Cannot call a class as a function"); + } +}; + +var createClass = function () { + function defineProperties(target, props) { + for (var i = 0; i < props.length; i++) { + var descriptor = props[i]; + descriptor.enumerable = descriptor.enumerable || false; + descriptor.configurable = true; + if ("value" in descriptor) descriptor.writable = true; + Object.defineProperty(target, descriptor.key, descriptor); + } + } + + return function (Constructor, protoProps, staticProps) { + if (protoProps) defineProperties(Constructor.prototype, protoProps); + if (staticProps) defineProperties(Constructor, staticProps); + return Constructor; + }; +}(); + + + + + + + +var get = function get(object, property, receiver) { + if (object === null) object = Function.prototype; + var desc = Object.getOwnPropertyDescriptor(object, property); + + if (desc === undefined) { + var parent = Object.getPrototypeOf(object); + + if (parent === null) { + return undefined; + } else { + return get(parent, property, receiver); + } + } else if ("value" in desc) { + return desc.value; + } else { + var getter = desc.get; + + if (getter === undefined) { + return undefined; + } + + return getter.call(receiver); + } +}; + +var inherits = function (subClass, superClass) { + if (typeof superClass !== "function" && superClass !== null) { + throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); + } + + subClass.prototype = Object.create(superClass && superClass.prototype, { + constructor: { + value: subClass, + enumerable: false, + writable: true, + configurable: true + } + }); + if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; +}; + + + + + + + + + + + +var possibleConstructorReturn = function (self, call) { + if (!self) { + throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); + } + + return call && (typeof call === "object" || typeof call === "function") ? call : self; +}; + + + + + +var slicedToArray = function () { + function sliceIterator(arr, i) { + var _arr = []; + var _n = true; + var _d = false; + var _e = undefined; + + try { + for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { + _arr.push(_s.value); + + if (i && _arr.length === i) break; + } + } catch (err) { + _d = true; + _e = err; + } finally { + try { + if (!_n && _i["return"]) _i["return"](); + } finally { + if (_d) throw _e; + } + } + + return _arr; + } + + return function (arr, i) { + if (Array.isArray(arr)) { + return arr; + } else if (Symbol.iterator in Object(arr)) { + return sliceIterator(arr, i); + } else { + throw new TypeError("Invalid attempt to destructure non-iterable instance"); + } + }; +}(); + +var Z_AXIS = new three.Vector3(0, 0, 1); +var DEG2RAD = three.Math.DEG2RAD; +var RAD2DEG = three.Math.RAD2DEG; +var IKBallConstraint = function () { + function IKBallConstraint(angle) { + classCallCheck(this, IKBallConstraint); + this.angle = angle; + } + createClass(IKBallConstraint, [{ + key: '_apply', + value: function _apply(joint) { + var direction = new three.Vector3().copy(joint._getDirection()); + var parentDirection = joint._localToWorldDirection(new three.Vector3().copy(Z_AXIS)).normalize(); + var currentAngle = direction.angleTo(parentDirection) * RAD2DEG; + if (this.angle / 2 < currentAngle) { + direction.normalize(); + var correctionAxis = new three.Vector3().crossVectors(parentDirection, direction).normalize(); + parentDirection.applyAxisAngle(correctionAxis, this.angle * DEG2RAD * 0.5); + joint._setDirection(parentDirection); + return true; + } + return false; + } + }]); + return IKBallConstraint; +}(); + +var Y_AXIS = new three.Vector3(0, 1, 0); +var IKJoint = function () { + function IKJoint(bone) { + var _ref = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}, + constraints = _ref.constraints; + classCallCheck(this, IKJoint); + this.constraints = constraints || []; + this.bone = bone; + this.distance = 0; + this._originalDirection = new three.Vector3(); + this._direction = new three.Vector3(); + this._worldPosition = new three.Vector3(); + this._isSubBase = false; + this._subBasePositions = null; + this.isIKJoint = true; + this._updateWorldPosition(); + } + createClass(IKJoint, [{ + key: '_setIsSubBase', + value: function _setIsSubBase() { + this._isSubBase = true; + this._subBasePositions = []; + } + }, { + key: '_applySubBasePositions', + value: function _applySubBasePositions() { + if (this._subBasePositions.length === 0) { + return; + } + getCentroid(this._subBasePositions, this._worldPosition); + this._subBasePositions.length = 0; + } + }, { + key: '_applyConstraints', + value: function _applyConstraints() { + if (!this.constraints) { + return; + } + var constraintApplied = false; + var _iteratorNormalCompletion = true; + var _didIteratorError = false; + var _iteratorError = undefined; + try { + for (var _iterator = this.constraints[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) { + var constraint = _step.value; + if (constraint && constraint._apply) { + var applied = constraint._apply(this); + constraintApplied = constraintApplied || applied; + } + } + } catch (err) { + _didIteratorError = true; + _iteratorError = err; + } finally { + try { + if (!_iteratorNormalCompletion && _iterator.return) { + _iterator.return(); + } + } finally { + if (_didIteratorError) { + throw _iteratorError; + } + } + } + return constraintApplied; + } + }, { + key: '_setDistance', + value: function _setDistance(distance) { + this.distance = distance; + } + }, { + key: '_getDirection', + value: function _getDirection() { + return this._direction; + } + }, { + key: '_setDirection', + value: function _setDirection(direction) { + this._direction.copy(direction); + } + }, { + key: '_getDistance', + value: function _getDistance() { + return this.distance; + } + }, { + key: '_updateMatrixWorld', + value: function _updateMatrixWorld() { + this.bone.updateMatrixWorld(true); + } + }, { + key: '_getWorldPosition', + value: function _getWorldPosition() { + return this._worldPosition; + } + }, { + key: '_getWorldDirection', + value: function _getWorldDirection(joint) { + return new three.Vector3().subVectors(this._getWorldPosition(), joint._getWorldPosition()).normalize(); + } + }, { + key: '_updateWorldPosition', + value: function _updateWorldPosition() { + getWorldPosition(this.bone, this._worldPosition); + } + }, { + key: '_setWorldPosition', + value: function _setWorldPosition(position) { + this._worldPosition.copy(position); + } + }, { + key: '_localToWorldDirection', + value: function _localToWorldDirection(direction) { + if (this.bone.parent) { + var parent = this.bone.parent.matrixWorld; + direction.transformDirection(parent); + } + return direction; + } + }, { + key: '_worldToLocalDirection', + value: function _worldToLocalDirection(direction) { + if (this.bone.parent) { + var inverseParent = new three.Matrix4().getInverse(this.bone.parent.matrixWorld); + direction.transformDirection(inverseParent); + } + return direction; + } + }, { + key: '_applyWorldPosition', + value: function _applyWorldPosition() { + var direction = new three.Vector3().copy(this._direction); + var position = new three.Vector3().copy(this._getWorldPosition()); + var parent = this.bone.parent; + if (parent) { + this._updateMatrixWorld(); + var inverseParent = new three.Matrix4().getInverse(this.bone.parent.matrixWorld); + transformPoint(position, inverseParent, position); + this.bone.position.copy(position); + this._updateMatrixWorld(); + this._worldToLocalDirection(direction); + setQuaternionFromDirection(direction, Y_AXIS, this.bone.quaternion); + } else { + this.bone.position.copy(position); + } + this.bone.updateMatrix(); + this._updateMatrixWorld(); + } + }, { + key: '_getWorldDistance', + value: function _getWorldDistance(joint) { + return this._worldPosition.distanceTo(joint.isIKJoint ? joint._getWorldPosition() : getWorldPosition(joint, new three.Vector3())); + } + }]); + return IKJoint; +}(); + +var IKChain = function () { + function IKChain() { + classCallCheck(this, IKChain); + this.isIKChain = true; + this.totalLengths = 0; + this.base = null; + this.effector = null; + this.effectorIndex = null; + this.chains = new Map(); + this.origin = null; + this.iterations = 100; + this.tolerance = 0.01; + this._depth = -1; + this._targetPosition = new three.Vector3(); + } + createClass(IKChain, [{ + key: 'add', + value: function add(joint) { + var _ref = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}, + target = _ref.target; + if (this.effector) { + throw new Error('Cannot add additional joints to a chain with an end effector.'); + } + if (!joint.isIKJoint) { + if (joint.isBone) { + joint = new IKJoint(joint); + } else { + throw new Error('Invalid joint in an IKChain. Must be an IKJoint or a THREE.Bone.'); + } + } + this.joints = this.joints || []; + this.joints.push(joint); + if (this.joints.length === 1) { + this.base = this.joints[0]; + this.origin = new three.Vector3().copy(this.base._getWorldPosition()); + } + else { + var previousJoint = this.joints[this.joints.length - 2]; + previousJoint._updateMatrixWorld(); + previousJoint._updateWorldPosition(); + joint._updateWorldPosition(); + var distance = previousJoint._getWorldDistance(joint); + if (distance === 0) { + throw new Error('bone with 0 distance between adjacent bone found'); + } + joint._setDistance(distance); + joint._updateWorldPosition(); + var direction = previousJoint._getWorldDirection(joint); + previousJoint._originalDirection = new three.Vector3().copy(direction); + joint._originalDirection = new three.Vector3().copy(direction); + this.totalLengths += distance; + } + if (target) { + this.effector = joint; + this.effectorIndex = joint; + this.target = target; + } + return this; + } + }, { + key: '_hasEffector', + value: function _hasEffector() { + return !!this.effector; + } + }, { + key: '_getDistanceFromTarget', + value: function _getDistanceFromTarget() { + return this._hasEffector() ? this.effector._getWorldDistance(this.target) : -1; + } + }, { + key: 'connect', + value: function connect(chain) { + if (!chain.isIKChain) { + throw new Error('Invalid connection in an IKChain. Must be an IKChain.'); + } + if (!chain.base.isIKJoint) { + throw new Error('Connecting chain does not have a base joint.'); + } + var index = this.joints.indexOf(chain.base); + if (this.target && index === this.joints.length - 1) { + throw new Error('Cannot append a chain to an end joint in a chain with a target.'); + } + if (index === -1) { + throw new Error('Cannot connect chain that does not have a base joint in parent chain.'); + } + this.joints[index]._setIsSubBase(); + var chains = this.chains.get(index); + if (!chains) { + chains = []; + this.chains.set(index, chains); + } + chains.push(chain); + return this; + } + }, { + key: '_updateJointWorldPositions', + value: function _updateJointWorldPositions() { + var _iteratorNormalCompletion = true; + var _didIteratorError = false; + var _iteratorError = undefined; + try { + for (var _iterator = this.joints[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) { + var joint = _step.value; + joint._updateWorldPosition(); + } + } catch (err) { + _didIteratorError = true; + _iteratorError = err; + } finally { + try { + if (!_iteratorNormalCompletion && _iterator.return) { + _iterator.return(); + } + } finally { + if (_didIteratorError) { + throw _iteratorError; + } + } + } + } + }, { + key: '_forward', + value: function _forward() { + this.origin.copy(this.base._getWorldPosition()); + if (this.target) { + this._targetPosition.setFromMatrixPosition(this.target.matrixWorld); + this.effector._setWorldPosition(this._targetPosition); + } else if (!this.joints[this.joints.length - 1]._isSubBase) { + return; + } + for (var i = 1; i < this.joints.length; i++) { + var joint = this.joints[i]; + if (joint._isSubBase) { + joint._applySubBasePositions(); + } + } + for (var _i = this.joints.length - 1; _i > 0; _i--) { + var _joint = this.joints[_i]; + var prevJoint = this.joints[_i - 1]; + var direction = prevJoint._getWorldDirection(_joint); + var worldPosition = direction.multiplyScalar(_joint.distance).add(_joint._getWorldPosition()); + if (prevJoint === this.base && this.base._isSubBase) { + this.base._subBasePositions.push(worldPosition); + } else { + prevJoint._setWorldPosition(worldPosition); + } + } + } + }, { + key: '_backward', + value: function _backward() { + if (!this.base._isSubBase) { + this.base._setWorldPosition(this.origin); + } + for (var i = 0; i < this.joints.length - 1; i++) { + var joint = this.joints[i]; + var nextJoint = this.joints[i + 1]; + var jointWorldPosition = joint._getWorldPosition(); + var direction = nextJoint._getWorldDirection(joint); + joint._setDirection(direction); + joint._applyConstraints(); + direction.copy(joint._direction); + if (!(this.base === joint && joint._isSubBase)) { + joint._applyWorldPosition(); + } + nextJoint._setWorldPosition(direction.multiplyScalar(nextJoint.distance).add(jointWorldPosition)); + if (i === this.joints.length - 2) { + if (nextJoint !== this.effector) { + nextJoint._setDirection(direction); + } + nextJoint._applyWorldPosition(); + } + } + return this._getDistanceFromTarget(); + } + }]); + return IKChain; +}(); + +var IK = function () { + function IK() { + classCallCheck(this, IK); + this.chains = []; + this._needsRecalculated = true; + this.isIK = true; + this._orderedChains = null; + } + createClass(IK, [{ + key: 'add', + value: function add(chain) { + if (!chain.isIKChain) { + throw new Error('Argument is not an IKChain.'); + } + this.chains.push(chain); + } + }, { + key: 'recalculate', + value: function recalculate() { + this._orderedChains = []; + var _iteratorNormalCompletion = true; + var _didIteratorError = false; + var _iteratorError = undefined; + try { + for (var _iterator = this.chains[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) { + var rootChain = _step.value; + var orderedChains = []; + this._orderedChains.push(orderedChains); + var chainsToSave = [rootChain]; + while (chainsToSave.length) { + var chain = chainsToSave.shift(); + orderedChains.push(chain); + var _iteratorNormalCompletion2 = true; + var _didIteratorError2 = false; + var _iteratorError2 = undefined; + try { + for (var _iterator2 = chain.chains.values()[Symbol.iterator](), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) { + var subChains = _step2.value; + var _iteratorNormalCompletion3 = true; + var _didIteratorError3 = false; + var _iteratorError3 = undefined; + try { + for (var _iterator3 = subChains[Symbol.iterator](), _step3; !(_iteratorNormalCompletion3 = (_step3 = _iterator3.next()).done); _iteratorNormalCompletion3 = true) { + var subChain = _step3.value; + if (chainsToSave.indexOf(subChain) !== -1) { + throw new Error('Recursive chain structure detected.'); + } + chainsToSave.push(subChain); + } + } catch (err) { + _didIteratorError3 = true; + _iteratorError3 = err; + } finally { + try { + if (!_iteratorNormalCompletion3 && _iterator3.return) { + _iterator3.return(); + } + } finally { + if (_didIteratorError3) { + throw _iteratorError3; + } + } + } + } + } catch (err) { + _didIteratorError2 = true; + _iteratorError2 = err; + } finally { + try { + if (!_iteratorNormalCompletion2 && _iterator2.return) { + _iterator2.return(); + } + } finally { + if (_didIteratorError2) { + throw _iteratorError2; + } + } + } + } + } + } catch (err) { + _didIteratorError = true; + _iteratorError = err; + } finally { + try { + if (!_iteratorNormalCompletion && _iterator.return) { + _iterator.return(); + } + } finally { + if (_didIteratorError) { + throw _iteratorError; + } + } + } + } + }, { + key: 'solve', + value: function solve() { + if (!this._orderedChains) { + this.recalculate(); + } + var _iteratorNormalCompletion4 = true; + var _didIteratorError4 = false; + var _iteratorError4 = undefined; + try { + for (var _iterator4 = this._orderedChains[Symbol.iterator](), _step4; !(_iteratorNormalCompletion4 = (_step4 = _iterator4.next()).done); _iteratorNormalCompletion4 = true) { + var subChains = _step4.value; + var iterations = 1; + while (iterations > 0) { + for (var i = subChains.length - 1; i >= 0; i--) { + subChains[i]._updateJointWorldPositions(); + } + for (var _i = subChains.length - 1; _i >= 0; _i--) { + subChains[_i]._forward(); + } + var withinTolerance = true; + for (var _i2 = 0; _i2 < subChains.length; _i2++) { + var distanceFromTarget = subChains[_i2]._backward(); + if (distanceFromTarget > this.tolerance) { + withinTolerance = false; + } + } + if (withinTolerance) { + break; + } + iterations--; + + } + } + } catch (err) { + _didIteratorError4 = true; + _iteratorError4 = err; + } finally { + try { + if (!_iteratorNormalCompletion4 && _iterator4.return) { + _iterator4.return(); + } + } finally { + if (_didIteratorError4) { + throw _iteratorError4; + } + } + } + } + }, { + key: 'getRootBone', + value: function getRootBone() { + return this.chains[0].base.bone; + } + }]); + return IK; +}(); + +var BoneHelper = function (_Object3D) { + inherits(BoneHelper, _Object3D); + function BoneHelper(height, boneSize, axesSize) { + classCallCheck(this, BoneHelper); + var _this = possibleConstructorReturn(this, (BoneHelper.__proto__ || Object.getPrototypeOf(BoneHelper)).call(this)); + if (height !== 0) { + var geo = new three.ConeBufferGeometry(boneSize, height, 4); + geo.applyMatrix(new three.Matrix4().makeRotationAxis(new three.Vector3(1, 0, 0), Math.PI / 2)); + _this.boneMesh = new three.Mesh(geo, new three.MeshBasicMaterial({ + color: 0xff0000, + wireframe: true, + depthTest: false, + depthWrite: false + })); + } else { + _this.boneMesh = new three.Object3D(); + } + _this.boneMesh.position.z = height / 2; + _this.add(_this.boneMesh); + _this.axesHelper = new three.AxesHelper(axesSize); + _this.add(_this.axesHelper); + return _this; + } + return BoneHelper; +}(three.Object3D); +var IKHelper = function (_Object3D2) { + inherits(IKHelper, _Object3D2); + function IKHelper(ik) { + var _ref = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}, + color = _ref.color, + showBones = _ref.showBones, + boneSize = _ref.boneSize, + showAxes = _ref.showAxes, + axesSize = _ref.axesSize, + wireframe = _ref.wireframe; + classCallCheck(this, IKHelper); + var _this2 = possibleConstructorReturn(this, (IKHelper.__proto__ || Object.getPrototypeOf(IKHelper)).call(this)); + boneSize = boneSize || 0.1; + axesSize = axesSize || 0.2; + if (!ik.isIK) { + throw new Error('IKHelper must receive an IK instance.'); + } + _this2.ik = ik; + _this2._meshes = new Map(); + var _iteratorNormalCompletion = true; + var _didIteratorError = false; + var _iteratorError = undefined; + try { + for (var _iterator = _this2.ik.chains[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) { + var rootChain = _step.value; + var chainsToMeshify = [rootChain]; + while (chainsToMeshify.length) { + var chain = chainsToMeshify.shift(); + for (var i = 0; i < chain.joints.length; i++) { + var joint = chain.joints[i]; + var nextJoint = chain.joints[i + 1]; + var distance = nextJoint ? nextJoint.distance : 0; + if (chain.base === joint && chain !== rootChain) { + continue; + } + var mesh = new BoneHelper(distance, boneSize, axesSize); + mesh.matrixAutoUpdate = false; + _this2._meshes.set(joint, mesh); + _this2.add(mesh); + } + var _iteratorNormalCompletion2 = true; + var _didIteratorError2 = false; + var _iteratorError2 = undefined; + try { + for (var _iterator2 = chain.chains.values()[Symbol.iterator](), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) { + var subChains = _step2.value; + var _iteratorNormalCompletion3 = true; + var _didIteratorError3 = false; + var _iteratorError3 = undefined; + try { + for (var _iterator3 = subChains[Symbol.iterator](), _step3; !(_iteratorNormalCompletion3 = (_step3 = _iterator3.next()).done); _iteratorNormalCompletion3 = true) { + var subChain = _step3.value; + chainsToMeshify.push(subChain); + } + } catch (err) { + _didIteratorError3 = true; + _iteratorError3 = err; + } finally { + try { + if (!_iteratorNormalCompletion3 && _iterator3.return) { + _iterator3.return(); + } + } finally { + if (_didIteratorError3) { + throw _iteratorError3; + } + } + } + } + } catch (err) { + _didIteratorError2 = true; + _iteratorError2 = err; + } finally { + try { + if (!_iteratorNormalCompletion2 && _iterator2.return) { + _iterator2.return(); + } + } finally { + if (_didIteratorError2) { + throw _iteratorError2; + } + } + } + } + } + } catch (err) { + _didIteratorError = true; + _iteratorError = err; + } finally { + try { + if (!_iteratorNormalCompletion && _iterator.return) { + _iterator.return(); + } + } finally { + if (_didIteratorError) { + throw _iteratorError; + } + } + } + _this2.showBones = showBones !== undefined ? showBones : true; + _this2.showAxes = showAxes !== undefined ? showAxes : true; + _this2.wireframe = wireframe !== undefined ? wireframe : true; + _this2.color = color || new three.Color(0xff0077); + return _this2; + } + createClass(IKHelper, [{ + key: 'updateMatrixWorld', + value: function updateMatrixWorld(force) { + var _iteratorNormalCompletion4 = true; + var _didIteratorError4 = false; + var _iteratorError4 = undefined; + try { + for (var _iterator4 = this._meshes[Symbol.iterator](), _step4; !(_iteratorNormalCompletion4 = (_step4 = _iterator4.next()).done); _iteratorNormalCompletion4 = true) { + var _ref2 = _step4.value; + var _ref3 = slicedToArray(_ref2, 2); + var joint = _ref3[0]; + var mesh = _ref3[1]; + mesh.matrix.copy(joint.bone.matrixWorld); + } + } catch (err) { + _didIteratorError4 = true; + _iteratorError4 = err; + } finally { + try { + if (!_iteratorNormalCompletion4 && _iterator4.return) { + _iterator4.return(); + } + } finally { + if (_didIteratorError4) { + throw _iteratorError4; + } + } + } + get(IKHelper.prototype.__proto__ || Object.getPrototypeOf(IKHelper.prototype), 'updateMatrixWorld', this).call(this, force); + } + }, { + key: 'showBones', + get: function get$$1() { + return this._showBones; + }, + set: function set$$1(showBones) { + if (showBones === this._showBones) { + return; + } + var _iteratorNormalCompletion5 = true; + var _didIteratorError5 = false; + var _iteratorError5 = undefined; + try { + for (var _iterator5 = this._meshes[Symbol.iterator](), _step5; !(_iteratorNormalCompletion5 = (_step5 = _iterator5.next()).done); _iteratorNormalCompletion5 = true) { + var _ref4 = _step5.value; + var _ref5 = slicedToArray(_ref4, 2); + var mesh = _ref5[1]; + if (showBones) { + mesh.add(mesh.boneMesh); + } else { + mesh.remove(mesh.boneMesh); + } + } + } catch (err) { + _didIteratorError5 = true; + _iteratorError5 = err; + } finally { + try { + if (!_iteratorNormalCompletion5 && _iterator5.return) { + _iterator5.return(); + } + } finally { + if (_didIteratorError5) { + throw _iteratorError5; + } + } + } + this._showBones = showBones; + } + }, { + key: 'showAxes', + get: function get$$1() { + return this._showAxes; + }, + set: function set$$1(showAxes) { + if (showAxes === this._showAxes) { + return; + } + var _iteratorNormalCompletion6 = true; + var _didIteratorError6 = false; + var _iteratorError6 = undefined; + try { + for (var _iterator6 = this._meshes[Symbol.iterator](), _step6; !(_iteratorNormalCompletion6 = (_step6 = _iterator6.next()).done); _iteratorNormalCompletion6 = true) { + var _ref6 = _step6.value; + var _ref7 = slicedToArray(_ref6, 2); + var mesh = _ref7[1]; + if (showAxes) { + mesh.add(mesh.axesHelper); + } else { + mesh.remove(mesh.axesHelper); + } + } + } catch (err) { + _didIteratorError6 = true; + _iteratorError6 = err; + } finally { + try { + if (!_iteratorNormalCompletion6 && _iterator6.return) { + _iterator6.return(); + } + } finally { + if (_didIteratorError6) { + throw _iteratorError6; + } + } + } + this._showAxes = showAxes; + } + }, { + key: 'wireframe', + get: function get$$1() { + return this._wireframe; + }, + set: function set$$1(wireframe) { + if (wireframe === this._wireframe) { + return; + } + var _iteratorNormalCompletion7 = true; + var _didIteratorError7 = false; + var _iteratorError7 = undefined; + try { + for (var _iterator7 = this._meshes[Symbol.iterator](), _step7; !(_iteratorNormalCompletion7 = (_step7 = _iterator7.next()).done); _iteratorNormalCompletion7 = true) { + var _ref8 = _step7.value; + var _ref9 = slicedToArray(_ref8, 2); + var mesh = _ref9[1]; + if (mesh.boneMesh.material) { + mesh.boneMesh.material.wireframe = wireframe; + } + } + } catch (err) { + _didIteratorError7 = true; + _iteratorError7 = err; + } finally { + try { + if (!_iteratorNormalCompletion7 && _iterator7.return) { + _iterator7.return(); + } + } finally { + if (_didIteratorError7) { + throw _iteratorError7; + } + } + } + this._wireframe = wireframe; + } + }, { + key: 'color', + get: function get$$1() { + return this._color; + }, + set: function set$$1(color) { + if (this._color && this._color.equals(color)) { + return; + } + color = color && color.isColor ? color : new three.Color(color); + var _iteratorNormalCompletion8 = true; + var _didIteratorError8 = false; + var _iteratorError8 = undefined; + try { + for (var _iterator8 = this._meshes[Symbol.iterator](), _step8; !(_iteratorNormalCompletion8 = (_step8 = _iterator8.next()).done); _iteratorNormalCompletion8 = true) { + var _ref10 = _step8.value; + var _ref11 = slicedToArray(_ref10, 2); + var mesh = _ref11[1]; + if (mesh.boneMesh.material) { + mesh.boneMesh.material.color = color; + } + } + } catch (err) { + _didIteratorError8 = true; + _iteratorError8 = err; + } finally { + try { + if (!_iteratorNormalCompletion8 && _iterator8.return) { + _iterator8.return(); + } + } finally { + if (_didIteratorError8) { + throw _iteratorError8; + } + } + } + this._color = color; + } + }]); + return IKHelper; +}(three.Object3D); + +if (typeof window !== 'undefined' && _typeof(window.THREE) === 'object') { + window.THREE.IK = IK; + window.THREE.IKChain = IKChain; + window.THREE.IKJoint = IKJoint; + window.THREE.IKBallConstraint = IKBallConstraint; + window.THREE.IKHelper = IKHelper; +} + +exports.IK = IK; +exports.IKChain = IKChain; +exports.IKJoint = IKJoint; +exports.IKBallConstraint = IKBallConstraint; +exports.IKHelper = IKHelper; + +Object.defineProperty(exports, '__esModule', { value: true }); + +}))); From b3068f160c7e2f427f95f74f13583f5b8a5e82fa Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Wed, 9 Oct 2019 00:19:46 -0400 Subject: [PATCH 003/562] Include three-ik.js --- app.html | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/app.html b/app.html index cb11323..b55d9f4 100644 --- a/app.html +++ b/app.html @@ -16,7 +16,8 @@ - + + From f65a87ca527ab06af0e57858025169dd12a0cf2d Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Wed, 9 Oct 2019 00:19:58 -0400 Subject: [PATCH 004/562] Add model imports to app.html --- app.html | 3 +++ 1 file changed, 3 insertions(+) diff --git a/app.html b/app.html index b55d9f4..ed7a244 100644 --- a/app.html +++ b/app.html @@ -26,6 +26,9 @@ --> + + @@ -25,9 +27,9 @@ - --> - + @@ -420,6 +420,7 @@ zForward: true, group, }; + Object.assign(o, resource); avatarMesh.riggedModel = new RiggedModel(model, o); // avatarMesh.riggedModel.rebind(o.group); From ead45181c9f4eff4b9bdaf42aef334093fc97aaa Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Thu, 10 Oct 2019 03:51:23 -0400 Subject: [PATCH 013/562] Dead code cleanup --- app.html | 12 ------------ 1 file changed, 12 deletions(-) diff --git a/app.html b/app.html index 10c7bb6..7b513c9 100644 --- a/app.html +++ b/app.html @@ -518,8 +518,6 @@ if (avatarMesh && avatarMesh.riggedModel) { const {hmd, gamepads} = data; - // this.avatarMesh.position.fromArray(hmd.position).sub(localVector.set(0, 1.5, 0)); - avatarMesh.riggedModel.targets.Head.position.fromArray(hmd.position); avatarMesh.riggedModel.targets.Head.quaternion.fromArray(hmd.quaternion); @@ -528,14 +526,6 @@ avatarMesh.riggedModel.targets.LeftHand.quaternion.fromArray(leftGamepad.quaternion); avatarMesh.riggedModel.targets.RightHand.position.fromArray(rightGamepad.position); avatarMesh.riggedModel.targets.RightHand.quaternion.fromArray(rightGamepad.quaternion); - - /* localQuaternion.fromArray(hmd.quaternion); - localEuler.setFromQuaternion(localQuaternion, localEuler.order); - localEuler.y += Math.PI; - localEuler.x *= -1; - this.avatarMesh.quaternion.setFromEuler(localEuler); - - avatarMesh.riggedModel.setState(hmd, gamepads); */ } } else if (landState && method === 'initState') { const {state} = data; @@ -6172,11 +6162,9 @@ const _updateAvatarMeshes = () => { for (let i = 0; i < avatarMeshes.length; i++) { const avatarMesh = avatarMeshes[i]; - // avatarMesh.riggedModel && avatarMesh.riggedModel.update(); const timeDelta = clock.getDelta() * 1000; const time = clock.elapsedTime * 1000; avatarMesh.riggedModel && avatarMesh.riggedModel.tick(time, timeDelta); - // avatarMesh.riggedModel && avatarMesh.riggedModel.fallback(avatarMesh.riggedModel.targets, time, timeDelta); } }; _updateAvatarMeshes(); From 7b284fbfbb10781fd5f55a975478269770364e98 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Thu, 10 Oct 2019 03:52:01 -0400 Subject: [PATCH 014/562] Add initial vrm model scheme --- proto/remapJointNames.js | 35 +++++++++++++++++++++++++++++++++++ 1 file changed, 35 insertions(+) diff --git a/proto/remapJointNames.js b/proto/remapJointNames.js index 39da92a..1191f2b 100644 --- a/proto/remapJointNames.js +++ b/proto/remapJointNames.js @@ -35,6 +35,40 @@ Object.assign(schemes, { .replace('Chest', 'Spine1'); }, }, + vrm: { + __proto__: schemes, + //Hips,Spine,Chest,Neck,Head,LeftEye,RightEye,ShoulderL,Upper_armL,Lower_armL,HandL,Upper_pointerL,Mid_pointerL,Upper_midL,Mid_midL,Upper_ringL,Mid_ringL,Upper_pinkieL,Lower_pinkieL,Upper_thumbL,Lower_thumbL,ShoulderR,Upper_armR,Lower_armR,HandR,Upper_pointerR,Mid_pointerR,Upper_midR,Mid_midR,Upper_ringR,Mid_ringR,Upper_pinkieR,Lower_pinkieR,Upper_thumbR,Lower_thumbR,Upper_legL,Lower_legL,FootL,ToeL,Upper_tail,Lower_tail,Upper_legR,Lower_legR,FootR,ToeR + re: /^J_Bip_C_Hips$/, + remap: function(from) { + return from + .replace('J_Bip_C_Hips', 'Hips') + .replace('J_Bip_C_Spine', 'Spine') + .replace('J_Bip_C_Chest', 'Spine1') + .replace('J_Bip_C_Neck', 'Neck') + .replace('J_Bip_C_Head', 'Head') + + .replace('J_Adj_L_FaceEye', 'LeftEye') + .replace('J_Adj_R_FaceEye', 'RightEye') + + .replace('J_Bip_L_Shoulder', 'LeftShoulder') + .replace('J_Bip_L_UpperArm', 'LeftArm') + .replace('J_Bip_L_LowerArm', 'LeftForeArm') + .replace('J_Bip_L_Hand', 'LeftHand') + + .replace('J_Bip_R_Shoulder', 'RightShoulder') + .replace('J_Bip_R_UpperArm', 'RightArm') + .replace('J_Bip_R_LowerArm', 'RightForeArm') + .replace('J_Bip_R_Hand', 'RightHand') + + .replace('J_Bip_L_UpperLeg', 'LeftUpLeg') + .replace('J_Bip_L_LowerLeg', 'LeftLeg') + .replace('J_Bip_L_Foot', 'LeftFoot') + + .replace('J_Bip_R_UpperLeg', 'RightUpLeg') + .replace('J_Bip_R_LowerLeg', 'RightLeg') + .replace('J_Bip_R_Foot', 'RightFoot'); + }, + }, side_winder: { __proto__: schemes, re: /^Left_wrist$/, @@ -86,6 +120,7 @@ function remapJointNames(skeleton, remapper) { schemes.mixamo, schemes.side_winder, schemes.knuckles, + schemes.vrm, ].map(function(x) { return typeof x === 'function' ? x : (x && x.match(skeleton)); }).filter(Boolean); From 9a752cd9ad8a7c2234ddac3b37ce6f186cc7a2e6 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Thu, 10 Oct 2019 03:52:46 -0400 Subject: [PATCH 015/562] Add initial anne.glb model --- proto/model.patches.js | 15 +++++++++++++++ 1 file changed, 15 insertions(+) diff --git a/proto/model.patches.js b/proto/model.patches.js index 6c9a4dd..52dc16d 100644 --- a/proto/model.patches.js +++ b/proto/model.patches.js @@ -147,6 +147,21 @@ var model_patches = { RightHand: [0,180,0], }, }, + anne: { + url: 'anne.glb', + preRotations: { + /* Armature: [0,180,0], */ + // Head: [-90,180,0], + // Spine: [-180,0,0], + // Spine1: [-180,0,0], + // Armature: [-180, 0, 0], + // Hips: [-180,0,0], + /* LeftEye: [90,0,0], + RightEye: [90,0,0], + LeftHand: [0,180,0], + RightHand: [0,180,0], */ + }, + } }; try { self.model_patches = model_patches; } catch(e) {} From 1748c53a71d12035c2f1cad777ee95d0df643bdb Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Thu, 10 Oct 2019 03:53:25 -0400 Subject: [PATCH 016/562] Add initial anne model load hack --- app.html | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app.html b/app.html index 7b513c9..9bf2755 100644 --- a/app.html +++ b/app.html @@ -400,8 +400,8 @@ avatarMesh.riggedModel = null; const loader = new THREE.GLTFLoader(); // const u = 'https://modulesio.github.io/models/miku.glb'; - // const u = 'model4.glb'; const resource = model_patches.model4; + // const resource = model_patches.anne; loader.load(resource.url, object => { if (avatarMesh) { const model = object.scene; From 8469414600ba022ef651f563740a4ac9a1dd840b Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Thu, 10 Oct 2019 03:53:58 -0400 Subject: [PATCH 017/562] Use anne model in app.html --- app.html | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/app.html b/app.html index 9bf2755..8ea336d 100644 --- a/app.html +++ b/app.html @@ -400,8 +400,8 @@ avatarMesh.riggedModel = null; const loader = new THREE.GLTFLoader(); // const u = 'https://modulesio.github.io/models/miku.glb'; - const resource = model_patches.model4; - // const resource = model_patches.anne; + // const resource = model_patches.model4; + const resource = model_patches.anne; loader.load(resource.url, object => { if (avatarMesh) { const model = object.scene; From 51227e7d54732ad68ce3c407e47134255b1c6c95 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 11:35:39 -0400 Subject: [PATCH 018/562] Add VRArmIK C# source --- vrarmik/ArmTransforms.cs | 81 ++++ vrarmik/ArmTransforms.cs.meta | 13 + vrarmik/AvatarPlatformOffset.cs | 19 + vrarmik/AvatarPlatformOffset.cs.meta | 11 + vrarmik/AvatarVRTrackingReferences.cs | 79 ++++ vrarmik/AvatarVRTrackingReferences.cs.meta | 13 + vrarmik/LocalVrTrackingInput.cs | 21 + vrarmik/LocalVrTrackingInput.cs.meta | 13 + vrarmik/PoseManager.cs | 92 +++++ vrarmik/PoseManager.cs.meta | 13 + vrarmik/ShoulderPoser.cs | 293 ++++++++++++++ vrarmik/ShoulderPoser.cs.meta | 13 + vrarmik/ShoulderTransforms.cs | 51 +++ vrarmik/ShoulderTransforms.cs.meta | 13 + vrarmik/StaticOffsetTransform.cs | 116 ++++++ vrarmik/StaticOffsetTransform.cs.meta | 13 + vrarmik/Utils.meta | 10 + vrarmik/Utils/Editor.meta | 10 + .../Utils/Editor/LabelOverrideInspector.cs | 13 + .../Editor/LabelOverrideInspector.cs.meta | 13 + vrarmik/Utils/Editor/ReadOnlyInspector.cs | 39 ++ .../Utils/Editor/ReadOnlyInspector.cs.meta | 13 + vrarmik/Utils/Extensions.cs | 301 ++++++++++++++ vrarmik/Utils/Extensions.cs.meta | 13 + vrarmik/Utils/InspectorAttributes.cs | 14 + vrarmik/Utils/InspectorAttributes.cs.meta | 13 + vrarmik/Utils/VectorHelpers.cs | 19 + vrarmik/Utils/VectorHelpers.cs.meta | 13 + vrarmik/VRArmIK.cs | 379 ++++++++++++++++++ vrarmik/VRArmIK.cs.meta | 13 + vrarmik/VRTrackingReferences.cs | 11 + vrarmik/VRTrackingReferences.cs.meta | 13 + 32 files changed, 1741 insertions(+) create mode 100644 vrarmik/ArmTransforms.cs create mode 100644 vrarmik/ArmTransforms.cs.meta create mode 100644 vrarmik/AvatarPlatformOffset.cs create mode 100644 vrarmik/AvatarPlatformOffset.cs.meta create mode 100644 vrarmik/AvatarVRTrackingReferences.cs create mode 100644 vrarmik/AvatarVRTrackingReferences.cs.meta create mode 100644 vrarmik/LocalVrTrackingInput.cs create mode 100644 vrarmik/LocalVrTrackingInput.cs.meta create mode 100644 vrarmik/PoseManager.cs create mode 100644 vrarmik/PoseManager.cs.meta create mode 100644 vrarmik/ShoulderPoser.cs create mode 100644 vrarmik/ShoulderPoser.cs.meta create mode 100644 vrarmik/ShoulderTransforms.cs create mode 100644 vrarmik/ShoulderTransforms.cs.meta create mode 100644 vrarmik/StaticOffsetTransform.cs create mode 100644 vrarmik/StaticOffsetTransform.cs.meta create mode 100644 vrarmik/Utils.meta create mode 100644 vrarmik/Utils/Editor.meta create mode 100644 vrarmik/Utils/Editor/LabelOverrideInspector.cs create mode 100644 vrarmik/Utils/Editor/LabelOverrideInspector.cs.meta create mode 100644 vrarmik/Utils/Editor/ReadOnlyInspector.cs create mode 100644 vrarmik/Utils/Editor/ReadOnlyInspector.cs.meta create mode 100644 vrarmik/Utils/Extensions.cs create mode 100644 vrarmik/Utils/Extensions.cs.meta create mode 100644 vrarmik/Utils/InspectorAttributes.cs create mode 100644 vrarmik/Utils/InspectorAttributes.cs.meta create mode 100644 vrarmik/Utils/VectorHelpers.cs create mode 100644 vrarmik/Utils/VectorHelpers.cs.meta create mode 100644 vrarmik/VRArmIK.cs create mode 100644 vrarmik/VRArmIK.cs.meta create mode 100644 vrarmik/VRTrackingReferences.cs create mode 100644 vrarmik/VRTrackingReferences.cs.meta diff --git a/vrarmik/ArmTransforms.cs b/vrarmik/ArmTransforms.cs new file mode 100644 index 0000000..a9ff175 --- /dev/null +++ b/vrarmik/ArmTransforms.cs @@ -0,0 +1,81 @@ +using UnityEngine; + +namespace VRArmIK +{ + public class ArmTransforms : MonoBehaviour + { + public Transform upperArm, lowerArm, wrist1, wrist2, hand; + + public float upperArmLength => distance(upperArm, lowerArm); + public float lowerArmLength => distance(lowerArm, hand); + public float armLength => upperArmLength + lowerArmLength; + + public bool armLengthByScale = false; + public Vector3 scaleAxis = Vector3.one; + public float scaleHandFactor = .7f; + + float distance(Transform a, Transform b) => (a.position - b.position).magnitude; + + void Start() + { + PoseManager.Instance.onCalibrate += updateArmLengths; + updateArmLengths(); + } + + void updateArmLengths() + { + var shoulderWidth = (upperArm.position - lowerArm.position).magnitude; + var _armLength = (PoseManager.Instance.playerWidthWrist - shoulderWidth) / 2f; + setArmLength(_armLength); + } + + public void setUpperArmLength(float length) + { + if (armLengthByScale) + { + float oldLowerArmLength = distance(lowerArm, hand); + + Vector3 newScale = upperArm.localScale - Vector3.Scale(upperArm.localScale, scaleAxis).magnitude * scaleAxis; + float scaleFactor = Vector3.Scale(upperArm.localScale, scaleAxis).magnitude / upperArmLength * length; + newScale += scaleAxis * scaleFactor; + upperArm.localScale = newScale; + + setLowerArmLength(oldLowerArmLength); + } + else + { + Vector3 pos = lowerArm.localPosition; + pos.x = Mathf.Sign(pos.x) * length; + lowerArm.localPosition = pos; + } + } + + public void setLowerArmLength(float length) + { + if (armLengthByScale) + { + } + else + { + Vector3 pos = hand.localPosition; + pos.x = Mathf.Sign(pos.x) * length; + hand.localPosition = pos; + } + } + + public void setArmLength(float length) + { + float upperArmFactor = .48f; + if (armLengthByScale) + { + upperArm.localScale = upperArm.localScale / armLength * length; + hand.localScale = Vector3.one / (1f - (1f - scaleHandFactor) * (1f - upperArm.localScale.x)); + } + else + { + setUpperArmLength(length * upperArmFactor); + setLowerArmLength(length * (1f - upperArmFactor)); + } + } + } +} \ No newline at end of file diff --git a/vrarmik/ArmTransforms.cs.meta b/vrarmik/ArmTransforms.cs.meta new file mode 100644 index 0000000..5233ee2 --- /dev/null +++ b/vrarmik/ArmTransforms.cs.meta @@ -0,0 +1,13 @@ +fileFormatVersion: 2 +guid: ec032af568a757340beb92f8fea7c65a +timeCreated: 1528126260 +licenseType: Free +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/AvatarPlatformOffset.cs b/vrarmik/AvatarPlatformOffset.cs new file mode 100644 index 0000000..cbc916b --- /dev/null +++ b/vrarmik/AvatarPlatformOffset.cs @@ -0,0 +1,19 @@ +using System.Collections; +using System.Collections.Generic; +using UnityEngine; +using VRArmIK; + +public class AvatarPlatformOffset : MonoBehaviour +{ + public bool correctVrPlatformOffsetOnStart = true; + + void Start() + { + if (correctVrPlatformOffsetOnStart) + { + var p = transform.position; + p.y -= PoseManager.Instance.vrSystemOffsetHeight; + transform.position = p; + } + } +} diff --git a/vrarmik/AvatarPlatformOffset.cs.meta b/vrarmik/AvatarPlatformOffset.cs.meta new file mode 100644 index 0000000..b6f1e27 --- /dev/null +++ b/vrarmik/AvatarPlatformOffset.cs.meta @@ -0,0 +1,11 @@ +fileFormatVersion: 2 +guid: ebc3d0b087754e543af552b66b64c187 +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/AvatarVRTrackingReferences.cs b/vrarmik/AvatarVRTrackingReferences.cs new file mode 100644 index 0000000..bb3bf38 --- /dev/null +++ b/vrarmik/AvatarVRTrackingReferences.cs @@ -0,0 +1,79 @@ +using UnityEngine; + +namespace VRArmIK +{ + public class AvatarVRTrackingReferences : MonoBehaviour + { + public StaticOffsetTransform head, hmd, leftHand, rightHand; + + void Start() + { + initTransforms(); + } + + [ContextMenu("init transforms")] + public void initTransforms() + { + createTransforms(); + connectTransforms(); + } + + void setStaticOffsetSettings(StaticOffsetTransform s) + { + s.referenceLocalPosition = false; + s.referenceLocalRotation = false; + s.applyLocalPosition = true; + s.applyLocalRotation = true; + s.applyPosition = true; + s.applyRotation = true; + s.applyForwardOffsetAfterRotationOffset = false; + } + + + void createTransform(ref StaticOffsetTransform t, string name) + { + if (t == null) + { + t = new GameObject(name).AddComponent(); + t.transform.parent = transform; + setStaticOffsetSettings(t); + } + } + + void createHandTransform(ref Transform t, string name, Transform parent) + { + if (t == null) + { + t = new GameObject(name).transform; + t.transform.localPosition = Vector3.zero; + t.transform.parent = parent; + } + } + + void createTransforms() + { + createTransform(ref head, nameof(head)); + createTransform(ref leftHand, nameof(leftHand)); + createTransform(ref rightHand, nameof(rightHand)); + createTransform(ref hmd, nameof(hmd)); + } + + void connectTransforms() + { + StaticOffsetTransform sot = this.GetOrAddComponent(); + if (sot.reference == null) + { + sot.reference = transform.parent; + } + + head.reference = head.reference != null ? head.reference : PoseManager.Instance.vrTransforms.head; + hmd.reference = hmd.reference != null ? hmd.reference : PoseManager.Instance.vrTransforms.hmd; + leftHand.reference = leftHand.reference != null + ? leftHand.reference + : PoseManager.Instance.vrTransforms.leftHand; + rightHand.reference = rightHand.reference != null + ? rightHand.reference + : PoseManager.Instance.vrTransforms.rightHand; + } + } +} \ No newline at end of file diff --git a/vrarmik/AvatarVRTrackingReferences.cs.meta b/vrarmik/AvatarVRTrackingReferences.cs.meta new file mode 100644 index 0000000..cfdb77c --- /dev/null +++ b/vrarmik/AvatarVRTrackingReferences.cs.meta @@ -0,0 +1,13 @@ +fileFormatVersion: 2 +guid: 4ead1ab284cddee4ba96204f5eb416da +timeCreated: 1528126940 +licenseType: Free +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/LocalVrTrackingInput.cs b/vrarmik/LocalVrTrackingInput.cs new file mode 100644 index 0000000..abaea21 --- /dev/null +++ b/vrarmik/LocalVrTrackingInput.cs @@ -0,0 +1,21 @@ +using System.Collections; +using System.Collections.Generic; +using UnityEngine; +using UnityEngine.VR; + +namespace VRArmIK +{ + public class LocalVrTrackingInput : MonoBehaviour + { + public UnityEngine.XR.XRNode node; + + void Update() + { + if (UnityEngine.XR.InputTracking.GetLocalPosition(node).magnitude < 0.000001f) + return; + + transform.localPosition = UnityEngine.XR.InputTracking.GetLocalPosition(node); + transform.localRotation = UnityEngine.XR.InputTracking.GetLocalRotation(node); + } + } +} \ No newline at end of file diff --git a/vrarmik/LocalVrTrackingInput.cs.meta b/vrarmik/LocalVrTrackingInput.cs.meta new file mode 100644 index 0000000..9d7d5af --- /dev/null +++ b/vrarmik/LocalVrTrackingInput.cs.meta @@ -0,0 +1,13 @@ +fileFormatVersion: 2 +guid: 6deb6fe736bb1fa4f920aa26496ebab4 +timeCreated: 1528128704 +licenseType: Free +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/PoseManager.cs b/vrarmik/PoseManager.cs new file mode 100644 index 0000000..62e14d3 --- /dev/null +++ b/vrarmik/PoseManager.cs @@ -0,0 +1,92 @@ +using UnityEngine; +using UnityEngine.XR; + +namespace VRArmIK +{ + [ExecuteInEditMode] + public class PoseManager : MonoBehaviour + { + public static PoseManager Instance = null; + public VRTrackingReferences vrTransforms; + + public delegate void OnCalibrateListener(); + + public event OnCalibrateListener onCalibrate; + + // Oculus uses a different reference position -> 0 is the reference head position if the user is standing in the middle of the room. + // In OpenVR, the 0 position is the ground position and the user is then at (0, playerHeightHmd, 0) if he is in the middle of the room, so I need to correct this for shoulder calculation + public float vrSystemOffsetHeight = 0.0f; + + public const float referencePlayerHeightHmd = 1.7f; + public const float referencePlayerWidthWrist = 1.39f; + public float playerHeightHmd = 1.70f; + public float playerWidthWrist = 1.39f; + public float playerWidthShoulders = 0.31f; + public bool loadPlayerSizeOnAwake = false; + + void OnEnable() + { + if (Instance == null) + { + Instance = this; + } + else if (Instance != null) + { + Debug.LogError("Multiple Instances of PoseManager in Scene"); + } + } + + void Awake() + { + if (loadPlayerSizeOnAwake) + { + loadPlayerSize(); + } + var device = XRSettings.loadedDeviceName; + vrSystemOffsetHeight = string.IsNullOrEmpty(device) || device == "OpenVR" ? 0 : playerHeightHmd; + } + + void Start() + { + onCalibrate += OnCalibrate; + } + + [ContextMenu("calibrate")] + void OnCalibrate() + { + playerHeightHmd = Camera.main.transform.position.y; + } + + void loadPlayerWidthShoulders() + { + playerWidthShoulders = PlayerPrefs.GetFloat("VRArmIK_PlayerWidthShoulders", 0.31f); + } + + public void savePlayerWidthShoulders(float width) + { + PlayerPrefs.SetFloat("VRArmIK_PlayerWidthShoulders", width); + } + + [ContextMenu("setArmLength")] + public void calibrateIK() + { + playerWidthWrist = (vrTransforms.leftHand.position - vrTransforms.rightHand.position).magnitude; + playerHeightHmd = vrTransforms.hmd.position.y; + savePlayerSize(playerHeightHmd, playerWidthWrist); + } + + public void savePlayerSize(float heightHmd, float widthWrist) + { + PlayerPrefs.SetFloat("VRArmIK_PlayerHeightHmd", heightHmd); + PlayerPrefs.SetFloat("VRArmIK_PlayerWidthWrist", widthWrist); + loadPlayerSize(); + onCalibrate?.Invoke(); + } + + public void loadPlayerSize() + { + playerHeightHmd = PlayerPrefs.GetFloat("VRArmIK_PlayerHeightHmd", referencePlayerHeightHmd); + playerWidthWrist = PlayerPrefs.GetFloat("VRArmIK_PlayerWidthWrist", referencePlayerWidthWrist); + } + } +} \ No newline at end of file diff --git a/vrarmik/PoseManager.cs.meta b/vrarmik/PoseManager.cs.meta new file mode 100644 index 0000000..8ccf569 --- /dev/null +++ b/vrarmik/PoseManager.cs.meta @@ -0,0 +1,13 @@ +fileFormatVersion: 2 +guid: 944b5c1596f68324a998b2ef2c5a363d +timeCreated: 1528126646 +licenseType: Free +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/ShoulderPoser.cs b/vrarmik/ShoulderPoser.cs new file mode 100644 index 0000000..bbde39b --- /dev/null +++ b/vrarmik/ShoulderPoser.cs @@ -0,0 +1,293 @@ +using UnityEngine; + +namespace VRArmIK +{ + public class ShoulderPoser : MonoBehaviour + { + public ShoulderTransforms shoulder; + public VRTrackingReferences vrTrackingReferences; + public AvatarVRTrackingReferences avatarTrackingReferences; + + public float headNeckDistance = 0.03f; + public Vector3 neckShoulderDistance = new Vector3(0f, -.1f, -0.02f); + + public float maxDeltaHeadRotation = 80f; + + [LabelOverride("max forward rotation")] + public float distinctShoulderRotationLimitForward = 33f; + + [LabelOverride("max backward rotation")] + public float distinctShoulderRotationLimitBackward = 0f; + + [LabelOverride("max upward rotation")] public float distinctShoulderRotationLimitUpward = 33f; + public float distinctShoulderRotationMultiplier = 30; + + public float rightRotationStartHeight = 0f; + public float rightRotationHeightFactor = 142f; + public float rightRotationHeadRotationFactor = 0.3f; + public float rightRotationHeadRotationOffset = -20f; + + public Vector3 headNeckDirectionVector = new Vector3(0f, -1f, -.05f); + public float startShoulderDislocationBefore = 0.005f; + + [Header("Features")] public bool ignoreYPos = true; + public bool autoDetectHandsBehindHead = true; + public bool clampRotationToHead = true; + public bool enableDistinctShoulderRotation = true; + public bool enableShoulderDislocation = true; + + + [Header("Debug")] + [DisplayOnly] + [SerializeField] + bool handsBehindHead = false; + + [DisplayOnly] [SerializeField] bool clampingHeadRotation = false; +#pragma warning disable 414 + [DisplayOnly] [SerializeField] bool shoulderDislocated = false; +#pragma warning restore 414 + public float shoulderRightRotation; + + + Vector3 lastAngle = Vector3.zero; + + + Vector3 leftShoulderAnkerStartLocalPosition, rightShoulderAnkerStartLocalPosition; + + void Start() + { + if (vrTrackingReferences == null) + vrTrackingReferences = PoseManager.Instance.vrTransforms; + + leftShoulderAnkerStartLocalPosition = shoulder.transform.InverseTransformPoint(shoulder.leftShoulderAnchor.position); + rightShoulderAnkerStartLocalPosition = + shoulder.transform.InverseTransformPoint(shoulder.rightShoulderAnchor.position); + } + + void onCalibrate() + { + shoulder.leftArm.setArmLength((avatarTrackingReferences.leftHand.transform.position - shoulder.leftShoulderAnchor.position) + .magnitude); + shoulder.rightArm.setArmLength((avatarTrackingReferences.rightHand.transform.position - shoulder.rightShoulderAnchor.position) + .magnitude); + } + + protected virtual void Update() + { + shoulder.transform.rotation = Quaternion.identity; + positionShoulder(); + rotateShoulderUp(); + rotateShoulderRight(); + + if (enableDistinctShoulderRotation) + { + rotateLeftShoulder(); + rotateRightShoulder(); + } + + if (enableShoulderDislocation) + { + clampShoulderHandDistance(); + } + else + { + shoulder.leftArm.transform.localPosition = Vector3.zero; + shoulder.rightArm.transform.localPosition = Vector3.zero; + } + + Debug.DrawRay(shoulder.transform.position, shoulder.transform.forward); + } + + protected virtual void rotateLeftShoulder() + { + rotateShoulderUp(shoulder.leftShoulder, shoulder.leftArm, avatarTrackingReferences.leftHand.transform, + leftShoulderAnkerStartLocalPosition, 1f); + + } + + protected virtual void rotateRightShoulder() + { + rotateShoulderUp(shoulder.rightShoulder, shoulder.rightArm, avatarTrackingReferences.rightHand.transform, + rightShoulderAnkerStartLocalPosition, -1f); + } + + void rotateShoulderUp(Transform shoulderSide, ArmTransforms arm, Transform targetHand, + Vector3 initialShoulderLocalPos, float angleSign) + { + Vector3 initialShoulderPos = shoulder.transform.TransformPoint(initialShoulderLocalPos); + Vector3 handShoulderOffset = targetHand.position - initialShoulderPos; + float armLength = arm.armLength; + + Vector3 targetAngle = Vector3.zero; + + float forwardDistanceRatio = Vector3.Dot(handShoulderOffset, shoulder.transform.forward) / armLength; + float upwardDistanceRatio = Vector3.Dot(handShoulderOffset, shoulder.transform.up) / armLength; + if (forwardDistanceRatio > 0f) + { + targetAngle.y = Mathf.Clamp((forwardDistanceRatio - 0.5f) * distinctShoulderRotationMultiplier, 0f, + distinctShoulderRotationLimitForward); + } + else + { + targetAngle.y = Mathf.Clamp(-(forwardDistanceRatio + 0.08f) * distinctShoulderRotationMultiplier * 10f, + -distinctShoulderRotationLimitBackward, 0f); + } + + targetAngle.z = Mathf.Clamp(-(upwardDistanceRatio - 0.5f) * distinctShoulderRotationMultiplier, + -distinctShoulderRotationLimitUpward, 0f); + + shoulderSide.localEulerAngles = targetAngle * angleSign; + } + + + void positionShoulder() + { + Vector3 headNeckOffset = avatarTrackingReferences.hmd.transform.rotation * headNeckDirectionVector; + Vector3 targetPosition = avatarTrackingReferences.head.transform.position + headNeckOffset * headNeckDistance; + shoulder.transform.localPosition = + shoulder.transform.parent.InverseTransformPoint(targetPosition) + neckShoulderDistance; + } + + protected virtual void rotateShoulderUp() + { + float angle = getCombinedDirectionAngleUp(); + + Vector3 targetRotation = new Vector3(0f, angle, 0f); + + if (autoDetectHandsBehindHead) + { + detectHandsBehindHead(ref targetRotation); + } + + if (clampRotationToHead) + { + clampHeadRotationDeltaUp(ref targetRotation); + } + + shoulder.transform.eulerAngles = targetRotation; + } + + protected virtual void rotateShoulderRight() + { + float heightDiff = vrTrackingReferences.hmd.transform.position.y - PoseManager.Instance.vrSystemOffsetHeight; + float relativeHeightDiff = -heightDiff / PoseManager.Instance.playerHeightHmd; + + float headRightRotation = VectorHelpers.getAngleBetween(shoulder.transform.forward, + avatarTrackingReferences.hmd.transform.forward, + Vector3.up, shoulder.transform.right) + rightRotationHeadRotationOffset; + float heightFactor = Mathf.Clamp(relativeHeightDiff - rightRotationStartHeight, 0f, 1f); + shoulderRightRotation = heightFactor * rightRotationHeightFactor; + shoulderRightRotation += Mathf.Clamp(headRightRotation * rightRotationHeadRotationFactor * heightFactor, 0f, 50f); + + shoulderRightRotation = Mathf.Clamp(shoulderRightRotation, 0f, 50f); + + Quaternion deltaRot = Quaternion.AngleAxis(shoulderRightRotation, shoulder.transform.right); + + + shoulder.transform.rotation = deltaRot * shoulder.transform.rotation; + positionShoulderRelative(); + } + + protected void positionShoulderRelative() + { + Quaternion deltaRot = Quaternion.AngleAxis(shoulderRightRotation, shoulder.transform.right); + Vector3 shoulderHeadDiff = shoulder.transform.position - avatarTrackingReferences.head.transform.position; + shoulder.transform.position = deltaRot * shoulderHeadDiff + avatarTrackingReferences.head.transform.position; + } + + float getCombinedDirectionAngleUp() + { + Transform leftHand = avatarTrackingReferences.leftHand.transform, rightHand = avatarTrackingReferences.rightHand.transform; + + Vector3 distanceLeftHand = leftHand.position - shoulder.transform.position, + distanceRightHand = rightHand.position - shoulder.transform.position; + + if (ignoreYPos) + { + distanceLeftHand.y = 0; + distanceRightHand.y = 0; + } + + Vector3 directionLeftHand = distanceLeftHand.normalized, + directionRightHand = distanceRightHand.normalized; + + Vector3 combinedDirection = directionLeftHand + directionRightHand; + + return Mathf.Atan2(combinedDirection.x, combinedDirection.z) * 180f / Mathf.PI; + } + + void detectHandsBehindHead(ref Vector3 targetRotation) + { + float delta = Mathf.Abs(targetRotation.y - lastAngle.y + 360f) % 360f; + if (delta > 150f && delta < 210f && lastAngle.magnitude > 0.000001f && !clampingHeadRotation) + { + handsBehindHead = !handsBehindHead; + } + + lastAngle = targetRotation; + + if (handsBehindHead) + { + targetRotation.y += 180f; + } + } + + void clampHeadRotationDeltaUp(ref Vector3 targetRotation) + { + float headUpRotation = (avatarTrackingReferences.head.transform.eulerAngles.y + 360f) % 360f; + float targetUpRotation = (targetRotation.y + 360f) % 360f; + + float delta = headUpRotation - targetUpRotation; + + if (delta > maxDeltaHeadRotation && delta < 180f || delta < -180f && delta >= -360f + maxDeltaHeadRotation) + { + targetRotation.y = headUpRotation - maxDeltaHeadRotation; + clampingHeadRotation = true; + } + else if (delta < -maxDeltaHeadRotation && delta > -180 || delta > 180f && delta < 360f - maxDeltaHeadRotation) + { + targetRotation.y = headUpRotation + maxDeltaHeadRotation; + clampingHeadRotation = true; + } + else + { + clampingHeadRotation = false; + } + } + + void clampShoulderHandDistance() + { + Vector3 leftHandVector = avatarTrackingReferences.leftHand.transform.position - shoulder.leftShoulderAnchor.position; + Vector3 rightHandVector = avatarTrackingReferences.rightHand.transform.position - shoulder.rightShoulderAnchor.position; + float leftShoulderHandDistance = leftHandVector.magnitude, rightShoulderHandDistance = rightHandVector.magnitude; + shoulderDislocated = false; + + float startBeforeFactor = (1f - startShoulderDislocationBefore); + + if (leftShoulderHandDistance > shoulder.leftArm.armLength * startBeforeFactor) + { + shoulderDislocated = true; + shoulder.leftArm.transform.position = shoulder.leftShoulderAnchor.position + + leftHandVector.normalized * + (leftShoulderHandDistance - shoulder.leftArm.armLength * startBeforeFactor); + } + else + { + shoulder.leftArm.transform.localPosition = Vector3.zero; + } + + if (rightShoulderHandDistance > shoulder.rightArm.armLength * startBeforeFactor) + { + shoulderDislocated = true; + shoulder.rightArm.transform.position = shoulder.rightShoulderAnchor.position + + rightHandVector.normalized * + (rightShoulderHandDistance - + shoulder.rightArm.armLength * startBeforeFactor); + } + else + { + shoulder.rightArm.transform.localPosition = Vector3.zero; + } + } + } +} \ No newline at end of file diff --git a/vrarmik/ShoulderPoser.cs.meta b/vrarmik/ShoulderPoser.cs.meta new file mode 100644 index 0000000..fe93a44 --- /dev/null +++ b/vrarmik/ShoulderPoser.cs.meta @@ -0,0 +1,13 @@ +fileFormatVersion: 2 +guid: e31d3179dbae073468ca3ac97ec5c295 +timeCreated: 1528126260 +licenseType: Free +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/ShoulderTransforms.cs b/vrarmik/ShoulderTransforms.cs new file mode 100644 index 0000000..283147e --- /dev/null +++ b/vrarmik/ShoulderTransforms.cs @@ -0,0 +1,51 @@ +using UnityEngine; + +namespace VRArmIK +{ + public class ShoulderTransforms : MonoBehaviour + { + public Transform leftShoulder, rightShoulder; + public Transform leftShoulderRenderer, rightShoulderRenderer; + public Transform leftShoulderAnchor, rightShoulderAnchor; + public ArmTransforms leftArmDummy, rightArmDummy; + public ArmTransforms leftArm, rightArm; + + void Awake() + { + if (leftArm == null) + { + leftArm = Instantiate(leftArmDummy, leftShoulderAnchor.position, leftShoulderAnchor.rotation, leftShoulderAnchor); + var armIk = leftArm.GetComponentInChildren(); + armIk.shoulder = this; + armIk.shoulderPoser = GetComponent(); + armIk.target = armIk.shoulderPoser.avatarTrackingReferences.leftHand.transform; + } + if (rightArm == null) + { + rightArm = Instantiate(rightArmDummy, leftShoulderAnchor.position, rightShoulderAnchor.rotation, rightShoulderAnchor); + var armIk = rightArm.GetComponentInChildren(); + armIk.shoulder = this; + armIk.shoulderPoser = GetComponent(); + armIk.target = armIk.shoulderPoser.avatarTrackingReferences.rightHand.transform; + } + } + + void Start() + { + setShoulderWidth(PoseManager.Instance.playerWidthShoulders); + } + + public void setShoulderWidth(float width) + { + Vector3 localScale = new Vector3(width * .5f, .05f, .05f); + Vector3 localPosition = new Vector3(width * .25f, 0f, 0f); + + leftShoulderRenderer.localScale = localScale; + leftShoulderRenderer.localPosition = -localPosition; + + rightShoulderRenderer.localScale = localScale; + rightShoulderRenderer.localPosition = localPosition; + } + } + +} \ No newline at end of file diff --git a/vrarmik/ShoulderTransforms.cs.meta b/vrarmik/ShoulderTransforms.cs.meta new file mode 100644 index 0000000..fab1c6c --- /dev/null +++ b/vrarmik/ShoulderTransforms.cs.meta @@ -0,0 +1,13 @@ +fileFormatVersion: 2 +guid: 952d2a04990ab5d4e95d91263da82740 +timeCreated: 1528126260 +licenseType: Free +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/StaticOffsetTransform.cs b/vrarmik/StaticOffsetTransform.cs new file mode 100644 index 0000000..8d816a9 --- /dev/null +++ b/vrarmik/StaticOffsetTransform.cs @@ -0,0 +1,116 @@ +using UnityEngine; + +namespace VRArmIK +{ + public class StaticOffsetTransform : MonoBehaviour + { + public enum EulerOrder + { + XYZ, + XZY, + YXZ, + YZX, + ZXY, + ZYX + }; + + public Transform reference = null; + public Vector3 offsetPosition; + public Vector3 offsetRotation; + public Vector3 orientationalOffset; + public Vector3 referenceRotationMultiplicator = Vector3.one; + + public EulerOrder axisOrder; + + public bool referenceLocalPosition = false, referenceLocalRotation = false; + public bool applyLocalPosition = false, applyLocalRotation = false; + public bool applyPosition = true, applyRotation = true; + public bool applyForwardOffsetAfterRotationOffset = false; + + + public static Vector3 switchAxis(Vector3 r, EulerOrder order) + { + switch (order) + { + case EulerOrder.XYZ: + return new Vector3(r.x, r.y, r.z); + case EulerOrder.XZY: + return new Vector3(r.x, r.z, r.y); + case EulerOrder.YXZ: + return new Vector3(r.y, r.x, r.z); + case EulerOrder.YZX: + return new Vector3(r.y, r.z, r.x); + case EulerOrder.ZXY: + return new Vector3(r.z, r.x, r.y); + case EulerOrder.ZYX: + return new Vector3(r.z, r.y, r.x); + + default: + return r; + } + } + + void Awake() + { + updatePosition(); + } + + void Update() + { + updatePosition(); + } + + void updatePosition() + { + if (reference == null) + return; + + Vector3 rot = switchAxis(referenceLocalRotation ? reference.localEulerAngles : reference.eulerAngles, axisOrder) + + offsetRotation; + rot.Scale(referenceRotationMultiplicator); + + Vector3 pos = referenceLocalPosition ? reference.localPosition : reference.position; + + + if (applyForwardOffsetAfterRotationOffset) + { + pos += Quaternion.Euler(rot) * Vector3.right * orientationalOffset.x; + pos += Quaternion.Euler(rot) * Vector3.up * orientationalOffset.y; + pos += Quaternion.Euler(rot) * Vector3.forward * orientationalOffset.z; + } + else + { + pos += reference.right * orientationalOffset.x; + pos += reference.up * orientationalOffset.y; + pos += reference.forward * orientationalOffset.z; + } + + pos += offsetPosition; + + if (applyPosition) + { + if (applyLocalPosition) + { + transform.localPosition = pos; + } + else + { + transform.position = pos; + } + } + + + if (applyRotation) + { + if (applyLocalRotation) + { + transform.localEulerAngles = rot; + } + else + { + transform.eulerAngles = rot; + } + } + } + } +} \ No newline at end of file diff --git a/vrarmik/StaticOffsetTransform.cs.meta b/vrarmik/StaticOffsetTransform.cs.meta new file mode 100644 index 0000000..4489a65 --- /dev/null +++ b/vrarmik/StaticOffsetTransform.cs.meta @@ -0,0 +1,13 @@ +fileFormatVersion: 2 +guid: fbf684afebadf9e448a63ab5775ad7c8 +timeCreated: 1528126990 +licenseType: Free +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/Utils.meta b/vrarmik/Utils.meta new file mode 100644 index 0000000..59a4e8c --- /dev/null +++ b/vrarmik/Utils.meta @@ -0,0 +1,10 @@ +fileFormatVersion: 2 +guid: 0e4d0a507fe834b4c8e75a936e694622 +folderAsset: yes +timeCreated: 1528126363 +licenseType: Free +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/Utils/Editor.meta b/vrarmik/Utils/Editor.meta new file mode 100644 index 0000000..99d9d7a --- /dev/null +++ b/vrarmik/Utils/Editor.meta @@ -0,0 +1,10 @@ +fileFormatVersion: 2 +guid: ff05c1415d8afee45b3616972c5e311b +folderAsset: yes +timeCreated: 1528126517 +licenseType: Free +DefaultImporter: + externalObjects: {} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/Utils/Editor/LabelOverrideInspector.cs b/vrarmik/Utils/Editor/LabelOverrideInspector.cs new file mode 100644 index 0000000..a808af9 --- /dev/null +++ b/vrarmik/Utils/Editor/LabelOverrideInspector.cs @@ -0,0 +1,13 @@ +using UnityEngine; +using UnityEditor; + +[CustomPropertyDrawer(typeof(LabelOverride))] +public class ThisPropertyDrawer : PropertyDrawer +{ + public override void OnGUI(Rect position, SerializedProperty property, GUIContent label) + { + var propertyAttribute = attribute as LabelOverride; + label.text = propertyAttribute?.label; + EditorGUI.PropertyField(position, property, label); + } +} \ No newline at end of file diff --git a/vrarmik/Utils/Editor/LabelOverrideInspector.cs.meta b/vrarmik/Utils/Editor/LabelOverrideInspector.cs.meta new file mode 100644 index 0000000..564a570 --- /dev/null +++ b/vrarmik/Utils/Editor/LabelOverrideInspector.cs.meta @@ -0,0 +1,13 @@ +fileFormatVersion: 2 +guid: 3a301e6db22cc5747b54c20deee95547 +timeCreated: 1528126534 +licenseType: Free +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/Utils/Editor/ReadOnlyInspector.cs b/vrarmik/Utils/Editor/ReadOnlyInspector.cs new file mode 100644 index 0000000..3443cbb --- /dev/null +++ b/vrarmik/Utils/Editor/ReadOnlyInspector.cs @@ -0,0 +1,39 @@ +using System.ComponentModel; +using UnityEditor; +using UnityEngine; + +[CustomPropertyDrawer(typeof(DisplayOnlyAttribute))] +public class DisplayOnlyInspector : PropertyDrawer +{ + public override void OnGUI(Rect position, SerializedProperty prop, GUIContent label) + { + string valueStr; + + switch (prop.propertyType) + { + case SerializedPropertyType.Integer: + valueStr = prop.intValue.ToString(); + break; + case SerializedPropertyType.Boolean: + valueStr = prop.boolValue.ToString(); + break; + case SerializedPropertyType.Float: + valueStr = prop.floatValue.ToString("0.00000"); + break; + case SerializedPropertyType.String: + valueStr = prop.stringValue; + break; + case SerializedPropertyType.Vector2: + valueStr = prop.vector2Value.ToString(); + break; + case SerializedPropertyType.Vector3: + valueStr = prop.vector3Value.ToString(); + break; + default: + valueStr = "(not supported)"; + break; + } + + EditorGUI.LabelField(position, label.text, valueStr); + } +} \ No newline at end of file diff --git a/vrarmik/Utils/Editor/ReadOnlyInspector.cs.meta b/vrarmik/Utils/Editor/ReadOnlyInspector.cs.meta new file mode 100644 index 0000000..5944573 --- /dev/null +++ b/vrarmik/Utils/Editor/ReadOnlyInspector.cs.meta @@ -0,0 +1,13 @@ +fileFormatVersion: 2 +guid: 995d6e718b3990448939a03e8c25bbf2 +timeCreated: 1528126534 +licenseType: Free +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/Utils/Extensions.cs b/vrarmik/Utils/Extensions.cs new file mode 100644 index 0000000..c2df576 --- /dev/null +++ b/vrarmik/Utils/Extensions.cs @@ -0,0 +1,301 @@ +using System; +using System.Collections; +using System.Collections.Generic; +using System.Linq; +using UnityEngine; +using UnityEngine.UI; +using Random = UnityEngine.Random; + + +namespace VRArmIK +{ + + public static class LinqExtensions + { + public static void ForEach(this IEnumerable source, Action action) + { + foreach (var item in source) + action(item); + } + } + + public static class UtilsExtensions + { + public static string F(this string self, params object[] objects) + { + return string.Format(self, objects); + } + } + + public static class TransformExtensions + { + public static void SetParentForReal(this Transform self, Transform parent) + { + self.SetParent(parent); + self.localScale = Vector3.one; + } + } + + public static class MonoBehaviourExtensions + { + public static T GetOrAddComponent(this Component self) where T : Component + { + T component = self.GetComponent(); + return component != null ? component : self.gameObject.AddComponent(); + } + + public static T GetOrAddComponentInChildren(this MonoBehaviour self) where T : MonoBehaviour + { + T component = self.GetComponentInChildren(); + return component != null ? component : self.gameObject.AddComponent(); + } + } + + + public static class LayerMaskExtensions + { + public static LayerMask Create(params string[] layerNames) + { + return NamesToMask(layerNames); + } + + public static LayerMask Create(params int[] layerNumbers) + { + return LayerNumbersToMask(layerNumbers); + } + + public static LayerMask NamesToMask(params string[] layerNames) + { + LayerMask ret = (LayerMask)0; + foreach (var name in layerNames) + { + ret |= (1 << LayerMask.NameToLayer(name)); + } + + return ret; + } + + public static LayerMask LayerNumbersToMask(params int[] layerNumbers) + { + LayerMask ret = (LayerMask)0; + foreach (var layer in layerNumbers) + { + ret |= (1 << layer); + } + return ret; + } + + public static LayerMask Inverse(this LayerMask original) + { + return ~original; + } + + public static LayerMask AddToMask(this LayerMask original, params string[] layerNames) + { + return original | NamesToMask(layerNames); + } + + public static LayerMask RemoveFromMask(this LayerMask original, params string[] layerNames) + { + return original & ~NamesToMask(layerNames); + } + + public static string[] MaskToNames(this LayerMask original) + { + var output = new List(); + + for (int i = 0; i < 32; ++i) + { + int shifted = 1 << i; + if ((original & shifted) == shifted) + { + string layerName = LayerMask.LayerToName(i); + if (!string.IsNullOrEmpty(layerName)) + { + output.Add(layerName); + } + } + } + return output.ToArray(); + } + + public static string MaskToString(this LayerMask original) + { + return MaskToString(original, ", "); + } + + public static string MaskToString(this LayerMask original, string delimiter) + { + return string.Join(delimiter, MaskToNames(original)); + } + } + + public static class VectorExtensions + { + public static Vector3 toVector3(this Vector2 self) + { + return new Vector3(self.x, self.y); + } + + public static Vector2 xy(this Vector3 self) + { + return new Vector2(self.x, self.y); + } + + public static Vector2 xz(this Vector3 self) + { + return new Vector2(self.x, self.z); + } + + public static Vector2 yz(this Vector3 self) + { + return new Vector2(self.y, self.y); + } + } + + public static class ListExtensions + { + public static T random(this List self) => self[(int)(self.Count * Random.value)]; + public static T next(this List self, int currentIndex, bool loop = false) + { + currentIndex++; + if (!loop && currentIndex >= self.Count) + { + throw new IndexOutOfRangeException(); + } + + currentIndex %= self.Count; + return self[currentIndex]; + } + + public static T next(this List self, T current, bool loop = false) + { + return self.next(self.IndexOf(current), loop); + } + } + + + public static class RandomExtensions + { + public static Vector2 insideUnitCircle(this System.Random self) + { + float radius = (float)self.NextDouble(); + float angle = (float)self.NextDouble() * Mathf.PI * 2; + + return new Vector2(Mathf.Cos(angle) * radius, Mathf.Sin(angle) * radius); + } + + public static float range(this System.Random self, float min, float max) + { + return (float)(self.NextDouble() * (max - min) + min); + } + + public static Vector3 insideUnitCube(this System.Random self) + { + return new Vector3(self.range(-1f, 1f), self.range(-1f, 1f), self.range(-1f, 1f)); + } + } + + public static class EnumExtenstions + { + public class Enum where T : struct, IConvertible + { + public static int Count + { + get + { + if (!typeof(T).IsEnum) + throw new ArgumentException("T must be an enumerated type"); + + return Enum.GetNames(typeof(T)).Length; + } + } + } + + public static T Next(this T src) where T : struct + { + if (!typeof(T).IsEnum) throw new ArgumentException($"Argumnent {typeof(T).FullName} is not an Enum"); + + T[] Arr = (T[])Enum.GetValues(src.GetType()); + int j = Array.IndexOf(Arr, src) + 1; + return Arr.Length == j ? Arr[0] : Arr[j]; + } + + public static bool IsLast(this T src) where T : struct + { + if (!typeof(T).IsEnum) throw new ArgumentException($"Argumnent {typeof(T).FullName} is not an Enum"); + + T[] Arr = (T[])Enum.GetValues(src.GetType()); + int j = Array.IndexOf(Arr, src); + return Arr.Length == j + 1; + } + } + + public static class UIExtensions + { + public static void setNormalButtonColor(this Button self, Color color) + { + + var buttonColors = self.colors; + buttonColors.normalColor = color; + self.colors = buttonColors; + } + + public static IEnumerator fadeIn(this CanvasGroup self, float duration) + { + float timePassed = 0.0f; + + while (timePassed < duration) + { + self.alpha = timePassed / duration; + timePassed += Time.unscaledDeltaTime; + yield return null; + } + + self.alpha = 1f; + } + + public static IEnumerator fadeOut(this CanvasGroup self, float duration) + { + float timePassed = 0.0f; + + while (timePassed < duration) + { + self.alpha = 1f - timePassed / duration; + timePassed += Time.unscaledDeltaTime; + yield return null; + } + + self.alpha = 0f; + } + } + + public static class IENumerableExtensions + { + public static T random(this IEnumerable self) + { + if (self.Count() == 0) + { + throw new IndexOutOfRangeException(); + } + return self.ElementAt(Random.Range(0, self.Count())); + } + } + + public static class FloatExtensions + { + public static float toSignedEulerAngle(this float self) + { + float result = self.toPositiveEulerAngle(); + if (result > 180f) + result = result - 360f; + return result; + } + + public static float toPositiveEulerAngle(this float self) + { + float result = (self % 360f + 360f) % 360f; + return result; + } + } +} \ No newline at end of file diff --git a/vrarmik/Utils/Extensions.cs.meta b/vrarmik/Utils/Extensions.cs.meta new file mode 100644 index 0000000..8db612c --- /dev/null +++ b/vrarmik/Utils/Extensions.cs.meta @@ -0,0 +1,13 @@ +fileFormatVersion: 2 +guid: dd40c275b2b3c5042bee83817b9d5277 +timeCreated: 1528126366 +licenseType: Free +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/Utils/InspectorAttributes.cs b/vrarmik/Utils/InspectorAttributes.cs new file mode 100644 index 0000000..d319b07 --- /dev/null +++ b/vrarmik/Utils/InspectorAttributes.cs @@ -0,0 +1,14 @@ +using UnityEngine; + +public class DisplayOnlyAttribute : PropertyAttribute +{ +} +public class LabelOverride : PropertyAttribute +{ + public string label; + public LabelOverride(string label) + { + this.label = label; + } + +} \ No newline at end of file diff --git a/vrarmik/Utils/InspectorAttributes.cs.meta b/vrarmik/Utils/InspectorAttributes.cs.meta new file mode 100644 index 0000000..3e7f3ba --- /dev/null +++ b/vrarmik/Utils/InspectorAttributes.cs.meta @@ -0,0 +1,13 @@ +fileFormatVersion: 2 +guid: a729eda56804e2043ac3fc58960a13bb +timeCreated: 1528126528 +licenseType: Free +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/Utils/VectorHelpers.cs b/vrarmik/Utils/VectorHelpers.cs new file mode 100644 index 0000000..fb69917 --- /dev/null +++ b/vrarmik/Utils/VectorHelpers.cs @@ -0,0 +1,19 @@ +using UnityEngine; + +public static class VectorHelpers +{ + public static float axisAngle(Vector3 v, Vector3 forward, Vector3 axis) + { + Vector3 right = Vector3.Cross(axis, forward); + forward = Vector3.Cross(right, axis); + return Mathf.Atan2(Vector3.Dot(v, right), Vector3.Dot(v, forward)) * Mathf.Rad2Deg; + } + + public static float getAngleBetween(Vector3 a, Vector3 b, Vector3 forward, Vector3 axis) + { + float angleA = axisAngle(a, forward, axis); + float angleB = axisAngle(b, forward, axis); + + return Mathf.DeltaAngle(angleA, angleB); + } +} diff --git a/vrarmik/Utils/VectorHelpers.cs.meta b/vrarmik/Utils/VectorHelpers.cs.meta new file mode 100644 index 0000000..2c65cb4 --- /dev/null +++ b/vrarmik/Utils/VectorHelpers.cs.meta @@ -0,0 +1,13 @@ +fileFormatVersion: 2 +guid: 56770d9500874ac4d8980df3b398f7ee +timeCreated: 1528126637 +licenseType: Free +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/VRArmIK.cs b/vrarmik/VRArmIK.cs new file mode 100644 index 0000000..a8f3b87 --- /dev/null +++ b/vrarmik/VRArmIK.cs @@ -0,0 +1,379 @@ +using UnityEngine; +using UnityEngine.UI; + +namespace VRArmIK +{ + + public class VRArmIK : MonoBehaviour + { + [System.Serializable] + public class ArmIKElbowSettings + { + public bool calcElbowAngle = true; + public bool clampElbowAngle = true; + public bool softClampElbowAngle = true; + public float maxAngle = 175f, minAngle = 13f, softClampRange = 10f; + public float offsetAngle = 135f; + public float yWeight = -60f; + public float zWeightTop = 260, zWeightBottom = -100, zBorderY = -.25f, zDistanceStart = .6f; + public float xWeight = -50f, xDistanceStart = .1f; + } + + [System.Serializable] + public class BeforePositioningSettings + { + public bool correctElbowOutside = true; + public float weight = -0.5f; + public float startBelowZ = .4f; + public float startAboveY = 0.1f; + } + + [System.Serializable] + public class ElbowCorrectionSettings + { + public bool useFixedElbowWhenNearShoulder = true; + public float startBelowDistance = .5f; + public float startBelowY = 0.1f; + public float weight = 2f; + public Vector3 localElbowPos = new Vector3(0.3f, -1f, -2f); + } + + [System.Serializable] + public class HandSettings + { + public bool useWristRotation = true; + public bool rotateElbowWithHandRight = true; + public bool rotateElbowWithHandForward = true; + public float handDeltaPow = 1.5f, handDeltaFactor = -.3f, handDeltaOffset = 45f; + // todo fix rotateElbowWithHandForward with factor != 1 -> horrible jumps. good value would be between [0.4, 0.6] + public float handDeltaForwardPow = 2f, handDeltaForwardFactor = 1f, handDeltaForwardOffset = 0f, handDeltaForwardDeadzone = .3f; + public float rotateElbowWithHandDelay = .08f; + } + + + public ArmTransforms arm; + public ShoulderTransforms shoulder; + public ShoulderPoser shoulderPoser; + public Transform target; + public bool left = true; + + public ArmIKElbowSettings elbowSettings; + public BeforePositioningSettings beforePositioningSettings; + public ElbowCorrectionSettings elbowCorrectionSettings; + public HandSettings handSettings; + + Vector3 nextLowerArmAngle; + + Quaternion upperArmStartRotation, lowerArmStartRotation, wristStartRotation, handStartRotation; + + float interpolatedDeltaElbow; + float interpolatedDeltaElbowForward; + + void Awake() + { + upperArmStartRotation = arm.upperArm.rotation; + lowerArmStartRotation = arm.lowerArm.rotation; + wristStartRotation = Quaternion.identity; + if (arm.wrist1 != null) + wristStartRotation = arm.wrist1.rotation; + handStartRotation = arm.hand.rotation; + } + + void OnEnable() + { + setUpperArmRotation(Quaternion.identity); + setLowerArmRotation(Quaternion.identity); + setHandRotation(Quaternion.identity); + } + + void LateUpdate() + { + updateUpperArmPosition(); + calcElbowInnerAngle(); + rotateShoulder(); + correctElbowRotation(); + if (elbowSettings.calcElbowAngle) + { + positionElbow(); + if (elbowCorrectionSettings.useFixedElbowWhenNearShoulder) + correctElbowAfterPositioning(); + if (handSettings.rotateElbowWithHandRight) + rotateElbowWithHandRight(); + if (handSettings.rotateElbowWithHandForward) + rotateElbowWithHandFoward(); + rotateHand(); + } + } + + public void updateArmAndTurnElbowUp() + { + updateUpperArmPosition(); + calcElbowInnerAngle(); + rotateShoulder(); + correctElbowRotation(); + } + + void updateUpperArmPosition() + { + //arm.upperArm.position = shoulderAnker.transform.position; + } + + void calcElbowInnerAngle() + { + Vector3 eulerAngles = new Vector3(); + float targetShoulderDistance = (target.position - upperArmPos).magnitude; + float innerAngle; + + if (targetShoulderDistance > arm.armLength) + { + innerAngle = 0f; + } + else + { + innerAngle = Mathf.Acos(Mathf.Clamp((Mathf.Pow(arm.upperArmLength, 2f) + Mathf.Pow(arm.lowerArmLength, 2f) - + Mathf.Pow(targetShoulderDistance, 2f)) / (2f * arm.upperArmLength * arm.lowerArmLength), -1f, 1f)) * Mathf.Rad2Deg; + if (left) + innerAngle = 180f - innerAngle; + else + innerAngle = 180f + innerAngle; + if (float.IsNaN(innerAngle)) + { + innerAngle = 180f; + } + } + + eulerAngles.y = innerAngle; + nextLowerArmAngle = eulerAngles; + } + + //source: https://github.com/NickHardeman/ofxIKArm/blob/master/src/ofxIKArm.cpp + void rotateShoulder() + { + Vector3 eulerAngles = new Vector3(); + Vector3 targetShoulderDirection = (target.position - upperArmPos).normalized; + float targetShoulderDistance = (target.position - upperArmPos).magnitude; + + eulerAngles.y = (left ? -1f : 1f) * + Mathf.Acos(Mathf.Clamp((Mathf.Pow(targetShoulderDistance, 2f) + Mathf.Pow(arm.upperArmLength, 2f) - + Mathf.Pow(arm.lowerArmLength, 2f)) / (2f * targetShoulderDistance * arm.upperArmLength), -1f, 1f)) * Mathf.Rad2Deg; + if (float.IsNaN(eulerAngles.y)) + eulerAngles.y = 0f; + + + Quaternion shoulderRightRotation = Quaternion.FromToRotation(armDirection, targetShoulderDirection); + setUpperArmRotation(shoulderRightRotation); + arm.upperArm.rotation = Quaternion.AngleAxis(eulerAngles.y, lowerArmRotation * Vector3.up) * arm.upperArm.rotation; + setLowerArmLocalRotation(Quaternion.Euler(nextLowerArmAngle)); + } + + float getElbowTargetAngle() + { + Vector3 localHandPosNormalized = shoulderAnker.InverseTransformPoint(handPos) / arm.armLength; + + // angle from Y + var angle = elbowSettings.yWeight * localHandPosNormalized.y + elbowSettings.offsetAngle; + + // angle from Z + /*angle += Mathf.Lerp(elbowSettings.zWeightBottom, elbowSettings.zWeightTop, Mathf.Clamp01((localHandPosNormalized.y + 1f) - elbowSettings.zBorderY)) * + (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0f));*/ + if (localHandPosNormalized.y > 0) + angle += elbowSettings.zWeightTop * (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0f) * Mathf.Max(localHandPosNormalized.y, 0f)); + else + angle += elbowSettings.zWeightBottom * (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0f) * Mathf.Max(-localHandPosNormalized.y, 0f)); + + + // angle from X + angle += elbowSettings.xWeight * Mathf.Max(localHandPosNormalized.x * (left ? 1.0f : -1.0f) + elbowSettings.xDistanceStart, 0f); + + if (elbowSettings.clampElbowAngle) + { + if (elbowSettings.softClampElbowAngle) + { + if (angle < elbowSettings.minAngle + elbowSettings.softClampRange) + { + float a = elbowSettings.minAngle + elbowSettings.softClampRange - angle; + angle = elbowSettings.minAngle + elbowSettings.softClampRange * (1f - Mathf.Log(1f + a) * 3f); + } + } + else + { + angle = Mathf.Clamp(angle, elbowSettings.minAngle, elbowSettings.maxAngle); + } + } + + if (left) + angle *= -1f; + + return angle; + } + + void correctElbowRotation() + { + var s = beforePositioningSettings; + + Vector3 localTargetPos = shoulderAnker.InverseTransformPoint(target.position) / arm.armLength; + float elbowOutsideFactor = Mathf.Clamp01( + Mathf.Clamp01((s.startBelowZ - localTargetPos.z) / + Mathf.Abs(s.startBelowZ) * .5f) * + Mathf.Clamp01((localTargetPos.y - s.startAboveY) / + Mathf.Abs(s.startAboveY)) * + Mathf.Clamp01(1f - localTargetPos.x * (left ? -1f : 1f)) + ) * s.weight; + + Vector3 shoulderHandDirection = (upperArmPos - handPos).normalized; + Vector3 targetDir = shoulder.transform.rotation * (Vector3.up + (s.correctElbowOutside ? (armDirection + Vector3.forward * -.2f) * elbowOutsideFactor : Vector3.zero)); + Vector3 cross = Vector3.Cross(shoulderHandDirection, targetDir * 1000f); + + Vector3 upperArmUp = upperArmRotation * Vector3.up; + + float elbowTargetUp = Vector3.Dot(upperArmUp, targetDir); + float elbowAngle = Vector3.Angle(cross, upperArmUp) + (left ? 0f : 180f); + Quaternion rotation = Quaternion.AngleAxis(elbowAngle * Mathf.Sign(elbowTargetUp), shoulderHandDirection); + arm.upperArm.rotation = rotation * arm.upperArm.rotation; + } + + /// + /// reduces calculation problems when hand is moving around shoulder XZ coordinates -> forces elbow to be outside of body + /// + void correctElbowAfterPositioning() + { + var s = elbowCorrectionSettings; + Vector3 localTargetPos = shoulderAnker.InverseTransformPoint(target.position) / arm.armLength; + Vector3 shoulderHandDirection = (upperArmPos - handPos).normalized; + Vector3 elbowPos = s.localElbowPos; + + if (left) + elbowPos.x *= -1f; + + Vector3 targetDir = shoulder.transform.rotation * elbowPos.normalized; + Vector3 cross = Vector3.Cross(shoulderHandDirection, targetDir); + + Vector3 upperArmUp = upperArmRotation * Vector3.up; + + + Vector3 distance = target.position - upperArmPos; + distance = distance.magnitude * shoulder.transform.InverseTransformDirection(distance / distance.magnitude); + + float weight = Mathf.Clamp01(Mathf.Clamp01((s.startBelowDistance - distance.xz().magnitude / arm.armLength) / + s.startBelowDistance) * s.weight + Mathf.Clamp01((-distance.z + .1f) * 3)) * + Mathf.Clamp01((s.startBelowY - localTargetPos.y) / + s.startBelowY); + + float elbowTargetUp = Vector3.Dot(upperArmUp, targetDir); + float elbowAngle2 = Vector3.Angle(cross, upperArmUp) + (left ? 0f : 180f); + Quaternion rotation = Quaternion.AngleAxis((elbowAngle2 * Mathf.Sign(elbowTargetUp)).toSignedEulerAngle() * Mathf.Clamp(weight, 0, 1f), shoulderHandDirection); + arm.upperArm.rotation = rotation * arm.upperArm.rotation; + } + + public void rotateElbow(float angle) + { + Vector3 shoulderHandDirection = (upperArmPos - handPos).normalized; + + Quaternion rotation = Quaternion.AngleAxis(angle, shoulderHandDirection); + setUpperArmRotation(rotation * upperArmRotation); + } + + //source: https://github.com/NickHardeman/ofxIKArm/blob/master/src/ofxIKArm.cpp + void positionElbow() + { + float targetElbowAngle = getElbowTargetAngle(); + rotateElbow(targetElbowAngle); + } + + + void rotateElbowWithHandRight() + { + var s = handSettings; + Vector3 handUpVec = target.rotation * Vector3.up; + float forwardAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.right, target.rotation * Vector3.right, + lowerArmRotation * Vector3.up, lowerArmRotation * Vector3.forward); + + // todo reduce influence if hand local forward rotation is high (hand tilted inside) + Quaternion handForwardRotation = Quaternion.AngleAxis(-forwardAngle, lowerArmRotation * Vector3.forward); + handUpVec = handForwardRotation * handUpVec; + + float elbowTargetAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.up, handUpVec, + lowerArmRotation * Vector3.forward, lowerArmRotation * armDirection); + + float deltaElbow = (elbowTargetAngle + (left ? -s.handDeltaOffset : s.handDeltaOffset)) / 180f; + + deltaElbow = Mathf.Sign(deltaElbow) * Mathf.Pow(Mathf.Abs(deltaElbow), s.handDeltaPow) * 180f * s.handDeltaFactor; + interpolatedDeltaElbow = + Mathf.LerpAngle(interpolatedDeltaElbow, deltaElbow, Time.deltaTime / s.rotateElbowWithHandDelay); + rotateElbow(interpolatedDeltaElbow); + } + + void rotateElbowWithHandFoward() + { + var s = handSettings; + Vector3 handRightVec = target.rotation * armDirection; + + float elbowTargetAngleForward = VectorHelpers.getAngleBetween(lowerArmRotation * armDirection, handRightVec, + lowerArmRotation * Vector3.up, lowerArmRotation * Vector3.forward); + + float deltaElbowForward = (elbowTargetAngleForward + (left ? -s.handDeltaForwardOffset : s.handDeltaForwardOffset)) / 180f; + + if (Mathf.Abs(deltaElbowForward) < s.handDeltaForwardDeadzone) + deltaElbowForward = 0f; + else + { + deltaElbowForward = (deltaElbowForward - Mathf.Sign(deltaElbowForward) * s.handDeltaForwardDeadzone) / (1f - s.handDeltaForwardDeadzone); + } + + deltaElbowForward = Mathf.Sign(deltaElbowForward) * Mathf.Pow(Mathf.Abs(deltaElbowForward), s.handDeltaForwardPow) * 180f; + interpolatedDeltaElbowForward = Mathf.LerpAngle(interpolatedDeltaElbowForward, deltaElbowForward, Time.deltaTime / s.rotateElbowWithHandDelay); + + float signedInterpolated = interpolatedDeltaElbowForward.toSignedEulerAngle(); + rotateElbow(signedInterpolated * s.handDeltaForwardFactor); + } + + public void rotateHand() + { + if (handSettings.useWristRotation) + { + Vector3 handUpVec = target.rotation * Vector3.up; + float forwardAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.right, target.rotation * Vector3.right, + lowerArmRotation * Vector3.up, lowerArmRotation * Vector3.forward); + + // todo reduce influence if hand local forward rotation is high (hand tilted inside) + Quaternion handForwardRotation = Quaternion.AngleAxis(-forwardAngle, lowerArmRotation * Vector3.forward); + handUpVec = handForwardRotation * handUpVec; + + float elbowTargetAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.up, handUpVec, + lowerArmRotation * Vector3.forward, lowerArmRotation * armDirection); + + elbowTargetAngle = Mathf.Clamp(elbowTargetAngle, -90f, 90f); + if (arm.wrist1 != null) + setWrist1Rotation(Quaternion.AngleAxis(elbowTargetAngle * .3f, lowerArmRotation * armDirection) * lowerArmRotation); + if (arm.wrist2 != null) + setWrist2Rotation(Quaternion.AngleAxis(elbowTargetAngle * .8f, lowerArmRotation * armDirection) * lowerArmRotation); + } + setHandRotation(target.rotation); + } + + Vector3 removeShoulderRightRotation(Vector3 direction) => Quaternion.AngleAxis(-shoulderPoser.shoulderRightRotation, shoulder.transform.right) * direction; + + Vector3 armDirection => left ? Vector3.left : Vector3.right; + Vector3 upperArmPos => arm.upperArm.position; + Vector3 lowerArmPos => arm.lowerArm.position; + Vector3 handPos => arm.hand.position; + Transform shoulderAnker => left ? shoulder.leftShoulderAnchor : shoulder.rightShoulderAnchor; + + Quaternion upperArmRotation => arm.upperArm.rotation * Quaternion.Inverse(upperArmStartRotation); + Quaternion lowerArmRotation => arm.lowerArm.rotation * Quaternion.Inverse(lowerArmStartRotation); + Quaternion handRotation => arm.hand.rotation * Quaternion.Inverse(handStartRotation); + + void setUpperArmRotation(Quaternion rotation) => arm.upperArm.rotation = rotation * upperArmStartRotation; + void setLowerArmRotation(Quaternion rotation) => arm.lowerArm.rotation = rotation * lowerArmStartRotation; + void setLowerArmLocalRotation(Quaternion rotation) => arm.lowerArm.rotation = upperArmRotation * rotation * lowerArmStartRotation; + void setWrist1Rotation(Quaternion rotation) => arm.wrist1.rotation = rotation * wristStartRotation; + void setWrist2Rotation(Quaternion rotation) => arm.wrist2.rotation = rotation * wristStartRotation; + void setWristLocalRotation(Quaternion rotation) => arm.wrist1.rotation = arm.lowerArm.rotation * rotation * wristStartRotation; + + void setHandRotation(Quaternion rotation) => + arm.hand.rotation = arm.hand.rotation = rotation * handStartRotation; + } + + + +} \ No newline at end of file diff --git a/vrarmik/VRArmIK.cs.meta b/vrarmik/VRArmIK.cs.meta new file mode 100644 index 0000000..83cc958 --- /dev/null +++ b/vrarmik/VRArmIK.cs.meta @@ -0,0 +1,13 @@ +fileFormatVersion: 2 +guid: 2042600d9ae259848ad6a09a5402a741 +timeCreated: 1528126260 +licenseType: Free +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: diff --git a/vrarmik/VRTrackingReferences.cs b/vrarmik/VRTrackingReferences.cs new file mode 100644 index 0000000..c703d31 --- /dev/null +++ b/vrarmik/VRTrackingReferences.cs @@ -0,0 +1,11 @@ +using UnityEngine; + +namespace VRArmIK +{ + + public class VRTrackingReferences : MonoBehaviour + { + public Transform leftController, rightController, hmd; + public Transform leftHand, rightHand, head; + } +} \ No newline at end of file diff --git a/vrarmik/VRTrackingReferences.cs.meta b/vrarmik/VRTrackingReferences.cs.meta new file mode 100644 index 0000000..ff3b893 --- /dev/null +++ b/vrarmik/VRTrackingReferences.cs.meta @@ -0,0 +1,13 @@ +fileFormatVersion: 2 +guid: 3ba5e054e87d9864db4d157ceca462f3 +timeCreated: 1528126807 +licenseType: Free +MonoImporter: + externalObjects: {} + serializedVersion: 2 + defaultReferences: [] + executionOrder: 0 + icon: {instanceID: 0} + userData: + assetBundleName: + assetBundleVariant: From dec545326864540cc0f2492f4b3f8e29dd05c2c3 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 12:03:59 -0400 Subject: [PATCH 019/562] Initial vrarmik porting work --- vrarmik/ArmTransforms.cs | 37 +++++----- vrarmik/ShoulderPoser.cs | 108 +++++++++++++---------------- vrarmik/ShoulderTransforms.cs | 26 +++---- vrarmik/VRArmIK.cs | 125 ++++++++++++++++++++-------------- 4 files changed, 147 insertions(+), 149 deletions(-) diff --git a/vrarmik/ArmTransforms.cs b/vrarmik/ArmTransforms.cs index a9ff175..c7dc298 100644 --- a/vrarmik/ArmTransforms.cs +++ b/vrarmik/ArmTransforms.cs @@ -1,35 +1,33 @@ -using UnityEngine; - -namespace VRArmIK -{ - public class ArmTransforms : MonoBehaviour +class ArmTransforms { - public Transform upperArm, lowerArm, wrist1, wrist2, hand; + Transform upperArm, lowerArm, wrist1, wrist2, hand; - public float upperArmLength => distance(upperArm, lowerArm); - public float lowerArmLength => distance(lowerArm, hand); - public float armLength => upperArmLength + lowerArmLength; + float upperArmLength => distance(upperArm, lowerArm); + float lowerArmLength => distance(lowerArm, hand); + float armLength => upperArmLength + lowerArmLength; - public bool armLengthByScale = false; - public Vector3 scaleAxis = Vector3.one; - public float scaleHandFactor = .7f; + bool armLengthByScale = false; + Vector3 scaleAxis = Vector3.one; + float scaleHandFactor = .7f; - float distance(Transform a, Transform b) => (a.position - b.position).magnitude; + distance(Transform a, Transform b) { + return (a.position - b.position).magnitude; + } - void Start() + Start() { PoseManager.Instance.onCalibrate += updateArmLengths; updateArmLengths(); } - void updateArmLengths() + updateArmLengths() { var shoulderWidth = (upperArm.position - lowerArm.position).magnitude; var _armLength = (PoseManager.Instance.playerWidthWrist - shoulderWidth) / 2f; setArmLength(_armLength); } - public void setUpperArmLength(float length) + setUpperArmLength(float length) { if (armLengthByScale) { @@ -50,7 +48,7 @@ public void setUpperArmLength(float length) } } - public void setLowerArmLength(float length) + setLowerArmLength(float length) { if (armLengthByScale) { @@ -63,7 +61,7 @@ public void setLowerArmLength(float length) } } - public void setArmLength(float length) + setArmLength(float length) { float upperArmFactor = .48f; if (armLengthByScale) @@ -77,5 +75,4 @@ public void setArmLength(float length) setLowerArmLength(length * (1f - upperArmFactor)); } } - } -} \ No newline at end of file + } \ No newline at end of file diff --git a/vrarmik/ShoulderPoser.cs b/vrarmik/ShoulderPoser.cs index bbde39b..a68cc0c 100644 --- a/vrarmik/ShoulderPoser.cs +++ b/vrarmik/ShoulderPoser.cs @@ -1,52 +1,41 @@ -using UnityEngine; - -namespace VRArmIK -{ - public class ShoulderPoser : MonoBehaviour +class ShoulderPoser { - public ShoulderTransforms shoulder; - public VRTrackingReferences vrTrackingReferences; - public AvatarVRTrackingReferences avatarTrackingReferences; + ShoulderTransforms shoulder; + VRTrackingReferences vrTrackingReferences; + AvatarVRTrackingReferences avatarTrackingReferences; - public float headNeckDistance = 0.03f; - public Vector3 neckShoulderDistance = new Vector3(0f, -.1f, -0.02f); + float headNeckDistance = 0.03f; + Vector3 neckShoulderDistance = new Vector3(0f, -.1f, -0.02f); - public float maxDeltaHeadRotation = 80f; + float maxDeltaHeadRotation = 80f; - [LabelOverride("max forward rotation")] - public float distinctShoulderRotationLimitForward = 33f; + float distinctShoulderRotationLimitForward = 33f; - [LabelOverride("max backward rotation")] - public float distinctShoulderRotationLimitBackward = 0f; + float distinctShoulderRotationLimitBackward = 0f; - [LabelOverride("max upward rotation")] public float distinctShoulderRotationLimitUpward = 33f; - public float distinctShoulderRotationMultiplier = 30; + float distinctShoulderRotationLimitUpward = 33f; + float distinctShoulderRotationMultiplier = 30; - public float rightRotationStartHeight = 0f; - public float rightRotationHeightFactor = 142f; - public float rightRotationHeadRotationFactor = 0.3f; - public float rightRotationHeadRotationOffset = -20f; + float rightRotationStartHeight = 0f; + float rightRotationHeightFactor = 142f; + float rightRotationHeadRotationFactor = 0.3f; + float rightRotationHeadRotationOffset = -20f; - public Vector3 headNeckDirectionVector = new Vector3(0f, -1f, -.05f); - public float startShoulderDislocationBefore = 0.005f; + Vector3 headNeckDirectionVector = new Vector3(0f, -1f, -.05f); + float startShoulderDislocationBefore = 0.005f; - [Header("Features")] public bool ignoreYPos = true; - public bool autoDetectHandsBehindHead = true; - public bool clampRotationToHead = true; - public bool enableDistinctShoulderRotation = true; - public bool enableShoulderDislocation = true; + bool ignoreYPos = true; + bool autoDetectHandsBehindHead = true; + bool clampRotationToHead = true; + bool enableDistinctShoulderRotation = true; + bool enableShoulderDislocation = true; - [Header("Debug")] - [DisplayOnly] - [SerializeField] bool handsBehindHead = false; - [DisplayOnly] [SerializeField] bool clampingHeadRotation = false; -#pragma warning disable 414 - [DisplayOnly] [SerializeField] bool shoulderDislocated = false; -#pragma warning restore 414 - public float shoulderRightRotation; + bool clampingHeadRotation = false; + bool shoulderDislocated = false; + float shoulderRightRotation; Vector3 lastAngle = Vector3.zero; @@ -54,7 +43,7 @@ public class ShoulderPoser : MonoBehaviour Vector3 leftShoulderAnkerStartLocalPosition, rightShoulderAnkerStartLocalPosition; - void Start() + Start() { if (vrTrackingReferences == null) vrTrackingReferences = PoseManager.Instance.vrTransforms; @@ -64,7 +53,7 @@ void Start() shoulder.transform.InverseTransformPoint(shoulder.rightShoulderAnchor.position); } - void onCalibrate() + onCalibrate() { shoulder.leftArm.setArmLength((avatarTrackingReferences.leftHand.transform.position - shoulder.leftShoulderAnchor.position) .magnitude); @@ -72,7 +61,7 @@ void onCalibrate() .magnitude); } - protected virtual void Update() + Update() { shoulder.transform.rotation = Quaternion.identity; positionShoulder(); @@ -98,20 +87,20 @@ protected virtual void Update() Debug.DrawRay(shoulder.transform.position, shoulder.transform.forward); } - protected virtual void rotateLeftShoulder() + rotateLeftShoulder() { rotateShoulderUp(shoulder.leftShoulder, shoulder.leftArm, avatarTrackingReferences.leftHand.transform, leftShoulderAnkerStartLocalPosition, 1f); } - protected virtual void rotateRightShoulder() + rotateRightShoulder() { rotateShoulderUp(shoulder.rightShoulder, shoulder.rightArm, avatarTrackingReferences.rightHand.transform, rightShoulderAnkerStartLocalPosition, -1f); } - void rotateShoulderUp(Transform shoulderSide, ArmTransforms arm, Transform targetHand, + rotateShoulderUp(Transform shoulderSide, ArmTransforms arm, Transform targetHand, Vector3 initialShoulderLocalPos, float angleSign) { Vector3 initialShoulderPos = shoulder.transform.TransformPoint(initialShoulderLocalPos); @@ -140,7 +129,7 @@ void rotateShoulderUp(Transform shoulderSide, ArmTransforms arm, Transform targe } - void positionShoulder() + positionShoulder() { Vector3 headNeckOffset = avatarTrackingReferences.hmd.transform.rotation * headNeckDirectionVector; Vector3 targetPosition = avatarTrackingReferences.head.transform.position + headNeckOffset * headNeckDistance; @@ -148,7 +137,7 @@ void positionShoulder() shoulder.transform.parent.InverseTransformPoint(targetPosition) + neckShoulderDistance; } - protected virtual void rotateShoulderUp() + rotateShoulderUp() { float angle = getCombinedDirectionAngleUp(); @@ -167,13 +156,13 @@ protected virtual void rotateShoulderUp() shoulder.transform.eulerAngles = targetRotation; } - protected virtual void rotateShoulderRight() + rotateShoulderRight() { float heightDiff = vrTrackingReferences.hmd.transform.position.y - PoseManager.Instance.vrSystemOffsetHeight; float relativeHeightDiff = -heightDiff / PoseManager.Instance.playerHeightHmd; - float headRightRotation = VectorHelpers.getAngleBetween(shoulder.transform.forward, - avatarTrackingReferences.hmd.transform.forward, + float headRightRotation = VectorHelpers.getAngleBetween(shoulder.transform.forward, + avatarTrackingReferences.hmd.transform.forward, Vector3.up, shoulder.transform.right) + rightRotationHeadRotationOffset; float heightFactor = Mathf.Clamp(relativeHeightDiff - rightRotationStartHeight, 0f, 1f); shoulderRightRotation = heightFactor * rightRotationHeightFactor; @@ -188,14 +177,14 @@ protected virtual void rotateShoulderRight() positionShoulderRelative(); } - protected void positionShoulderRelative() + positionShoulderRelative() { Quaternion deltaRot = Quaternion.AngleAxis(shoulderRightRotation, shoulder.transform.right); Vector3 shoulderHeadDiff = shoulder.transform.position - avatarTrackingReferences.head.transform.position; shoulder.transform.position = deltaRot * shoulderHeadDiff + avatarTrackingReferences.head.transform.position; } - float getCombinedDirectionAngleUp() + getCombinedDirectionAngleUp() { Transform leftHand = avatarTrackingReferences.leftHand.transform, rightHand = avatarTrackingReferences.rightHand.transform; @@ -216,7 +205,7 @@ float getCombinedDirectionAngleUp() return Mathf.Atan2(combinedDirection.x, combinedDirection.z) * 180f / Mathf.PI; } - void detectHandsBehindHead(ref Vector3 targetRotation) + detectHandsBehindHead(ref Vector3 targetRotation) { float delta = Mathf.Abs(targetRotation.y - lastAngle.y + 360f) % 360f; if (delta > 150f && delta < 210f && lastAngle.magnitude > 0.000001f && !clampingHeadRotation) @@ -232,8 +221,8 @@ void detectHandsBehindHead(ref Vector3 targetRotation) } } - void clampHeadRotationDeltaUp(ref Vector3 targetRotation) - { + clampHeadRotationDeltaUp(ref Vector3 targetRotation) + { float headUpRotation = (avatarTrackingReferences.head.transform.eulerAngles.y + 360f) % 360f; float targetUpRotation = (targetRotation.y + 360f) % 360f; @@ -255,7 +244,7 @@ void clampHeadRotationDeltaUp(ref Vector3 targetRotation) } } - void clampShoulderHandDistance() + clampShoulderHandDistance() { Vector3 leftHandVector = avatarTrackingReferences.leftHand.transform.position - shoulder.leftShoulderAnchor.position; Vector3 rightHandVector = avatarTrackingReferences.rightHand.transform.position - shoulder.rightShoulderAnchor.position; @@ -267,8 +256,8 @@ void clampShoulderHandDistance() if (leftShoulderHandDistance > shoulder.leftArm.armLength * startBeforeFactor) { shoulderDislocated = true; - shoulder.leftArm.transform.position = shoulder.leftShoulderAnchor.position + - leftHandVector.normalized * + shoulder.leftArm.transform.position = shoulder.leftShoulderAnchor.position + + leftHandVector.normalized * (leftShoulderHandDistance - shoulder.leftArm.armLength * startBeforeFactor); } else @@ -279,9 +268,9 @@ void clampShoulderHandDistance() if (rightShoulderHandDistance > shoulder.rightArm.armLength * startBeforeFactor) { shoulderDislocated = true; - shoulder.rightArm.transform.position = shoulder.rightShoulderAnchor.position + - rightHandVector.normalized * - (rightShoulderHandDistance - + shoulder.rightArm.transform.position = shoulder.rightShoulderAnchor.position + + rightHandVector.normalized * + (rightShoulderHandDistance - shoulder.rightArm.armLength * startBeforeFactor); } else @@ -289,5 +278,4 @@ void clampShoulderHandDistance() shoulder.rightArm.transform.localPosition = Vector3.zero; } } - } -} \ No newline at end of file + } \ No newline at end of file diff --git a/vrarmik/ShoulderTransforms.cs b/vrarmik/ShoulderTransforms.cs index 283147e..65b53f2 100644 --- a/vrarmik/ShoulderTransforms.cs +++ b/vrarmik/ShoulderTransforms.cs @@ -1,16 +1,12 @@ -using UnityEngine; - -namespace VRArmIK -{ - public class ShoulderTransforms : MonoBehaviour +class ShoulderTransforms { - public Transform leftShoulder, rightShoulder; - public Transform leftShoulderRenderer, rightShoulderRenderer; - public Transform leftShoulderAnchor, rightShoulderAnchor; - public ArmTransforms leftArmDummy, rightArmDummy; - public ArmTransforms leftArm, rightArm; + Transform leftShoulder, rightShoulder; + Transform leftShoulderRenderer, rightShoulderRenderer; + Transform leftShoulderAnchor, rightShoulderAnchor; + ArmTransforms leftArmDummy, rightArmDummy; + ArmTransforms leftArm, rightArm; - void Awake() + Awake() { if (leftArm == null) { @@ -30,12 +26,12 @@ void Awake() } } - void Start() + Start() { setShoulderWidth(PoseManager.Instance.playerWidthShoulders); } - public void setShoulderWidth(float width) + setShoulderWidth(float width) { Vector3 localScale = new Vector3(width * .5f, .05f, .05f); Vector3 localPosition = new Vector3(width * .25f, 0f, 0f); @@ -46,6 +42,4 @@ public void setShoulderWidth(float width) rightShoulderRenderer.localScale = localScale; rightShoulderRenderer.localPosition = localPosition; } - } - -} \ No newline at end of file + } \ No newline at end of file diff --git a/vrarmik/VRArmIK.cs b/vrarmik/VRArmIK.cs index a8f3b87..16cd951 100644 --- a/vrarmik/VRArmIK.cs +++ b/vrarmik/VRArmIK.cs @@ -1,13 +1,10 @@ -using UnityEngine; -using UnityEngine.UI; +import ArmTransforms from './ArmTransforms.js'; +import ShoulderTransforms from './ShoulderTransforms.js'; +import ShoulderPoser from './ShoulderPoser.js'; -namespace VRArmIK -{ - - public class VRArmIK : MonoBehaviour + public class VRArmIK { - [System.Serializable] - public class ArmIKElbowSettings + class ArmIKElbowSettings { public bool calcElbowAngle = true; public bool clampElbowAngle = true; @@ -19,8 +16,7 @@ public class ArmIKElbowSettings public float xWeight = -50f, xDistanceStart = .1f; } - [System.Serializable] - public class BeforePositioningSettings + class BeforePositioningSettings { public bool correctElbowOutside = true; public float weight = -0.5f; @@ -28,8 +24,7 @@ public class BeforePositioningSettings public float startAboveY = 0.1f; } - [System.Serializable] - public class ElbowCorrectionSettings + class ElbowCorrectionSettings { public bool useFixedElbowWhenNearShoulder = true; public float startBelowDistance = .5f; @@ -38,8 +33,7 @@ public class ElbowCorrectionSettings public Vector3 localElbowPos = new Vector3(0.3f, -1f, -2f); } - [System.Serializable] - public class HandSettings + class HandSettings { public bool useWristRotation = true; public bool rotateElbowWithHandRight = true; @@ -50,7 +44,6 @@ public class HandSettings public float rotateElbowWithHandDelay = .08f; } - public ArmTransforms arm; public ShoulderTransforms shoulder; public ShoulderPoser shoulderPoser; @@ -69,7 +62,7 @@ public class HandSettings float interpolatedDeltaElbow; float interpolatedDeltaElbowForward; - void Awake() + Awake() { upperArmStartRotation = arm.upperArm.rotation; lowerArmStartRotation = arm.lowerArm.rotation; @@ -79,14 +72,14 @@ void Awake() handStartRotation = arm.hand.rotation; } - void OnEnable() + OnEnable() { setUpperArmRotation(Quaternion.identity); setLowerArmRotation(Quaternion.identity); setHandRotation(Quaternion.identity); } - void LateUpdate() + LateUpdate() { updateUpperArmPosition(); calcElbowInnerAngle(); @@ -105,7 +98,7 @@ void LateUpdate() } } - public void updateArmAndTurnElbowUp() + updateArmAndTurnElbowUp() { updateUpperArmPosition(); calcElbowInnerAngle(); @@ -113,12 +106,12 @@ public void updateArmAndTurnElbowUp() correctElbowRotation(); } - void updateUpperArmPosition() + updateUpperArmPosition() { //arm.upperArm.position = shoulderAnker.transform.position; } - void calcElbowInnerAngle() + calcElbowInnerAngle() { Vector3 eulerAngles = new Vector3(); float targetShoulderDistance = (target.position - upperArmPos).magnitude; @@ -147,7 +140,7 @@ void calcElbowInnerAngle() } //source: https://github.com/NickHardeman/ofxIKArm/blob/master/src/ofxIKArm.cpp - void rotateShoulder() + rotateShoulder() { Vector3 eulerAngles = new Vector3(); Vector3 targetShoulderDirection = (target.position - upperArmPos).normalized; @@ -166,7 +159,7 @@ void rotateShoulder() setLowerArmLocalRotation(Quaternion.Euler(nextLowerArmAngle)); } - float getElbowTargetAngle() + getElbowTargetAngle() { Vector3 localHandPosNormalized = shoulderAnker.InverseTransformPoint(handPos) / arm.armLength; @@ -207,7 +200,7 @@ float getElbowTargetAngle() return angle; } - void correctElbowRotation() + correctElbowRotation() { var s = beforePositioningSettings; @@ -235,7 +228,7 @@ void correctElbowRotation() /// /// reduces calculation problems when hand is moving around shoulder XZ coordinates -> forces elbow to be outside of body /// - void correctElbowAfterPositioning() + correctElbowAfterPositioning() { var s = elbowCorrectionSettings; Vector3 localTargetPos = shoulderAnker.InverseTransformPoint(target.position) / arm.armLength; @@ -265,7 +258,7 @@ void correctElbowAfterPositioning() arm.upperArm.rotation = rotation * arm.upperArm.rotation; } - public void rotateElbow(float angle) + rotateElbow(float angle) { Vector3 shoulderHandDirection = (upperArmPos - handPos).normalized; @@ -274,14 +267,14 @@ public void rotateElbow(float angle) } //source: https://github.com/NickHardeman/ofxIKArm/blob/master/src/ofxIKArm.cpp - void positionElbow() + positionElbow() { float targetElbowAngle = getElbowTargetAngle(); rotateElbow(targetElbowAngle); } - void rotateElbowWithHandRight() + rotateElbowWithHandRight() { var s = handSettings; Vector3 handUpVec = target.rotation * Vector3.up; @@ -303,7 +296,7 @@ void rotateElbowWithHandRight() rotateElbow(interpolatedDeltaElbow); } - void rotateElbowWithHandFoward() + rotateElbowWithHandFoward() { var s = handSettings; Vector3 handRightVec = target.rotation * armDirection; @@ -327,7 +320,7 @@ void rotateElbowWithHandFoward() rotateElbow(signedInterpolated * s.handDeltaForwardFactor); } - public void rotateHand() + rotateHand() { if (handSettings.useWristRotation) { @@ -351,29 +344,55 @@ public void rotateHand() setHandRotation(target.rotation); } - Vector3 removeShoulderRightRotation(Vector3 direction) => Quaternion.AngleAxis(-shoulderPoser.shoulderRightRotation, shoulder.transform.right) * direction; - - Vector3 armDirection => left ? Vector3.left : Vector3.right; - Vector3 upperArmPos => arm.upperArm.position; - Vector3 lowerArmPos => arm.lowerArm.position; - Vector3 handPos => arm.hand.position; - Transform shoulderAnker => left ? shoulder.leftShoulderAnchor : shoulder.rightShoulderAnchor; - - Quaternion upperArmRotation => arm.upperArm.rotation * Quaternion.Inverse(upperArmStartRotation); - Quaternion lowerArmRotation => arm.lowerArm.rotation * Quaternion.Inverse(lowerArmStartRotation); - Quaternion handRotation => arm.hand.rotation * Quaternion.Inverse(handStartRotation); - - void setUpperArmRotation(Quaternion rotation) => arm.upperArm.rotation = rotation * upperArmStartRotation; - void setLowerArmRotation(Quaternion rotation) => arm.lowerArm.rotation = rotation * lowerArmStartRotation; - void setLowerArmLocalRotation(Quaternion rotation) => arm.lowerArm.rotation = upperArmRotation * rotation * lowerArmStartRotation; - void setWrist1Rotation(Quaternion rotation) => arm.wrist1.rotation = rotation * wristStartRotation; - void setWrist2Rotation(Quaternion rotation) => arm.wrist2.rotation = rotation * wristStartRotation; - void setWristLocalRotation(Quaternion rotation) => arm.wrist1.rotation = arm.lowerArm.rotation * rotation * wristStartRotation; - - void setHandRotation(Quaternion rotation) => - arm.hand.rotation = arm.hand.rotation = rotation * handStartRotation; - } + removeShoulderRightRotation(Vector3 direction) { + return Quaternion.AngleAxis(-shoulderPoser.shoulderRightRotation, shoulder.transform.right) * direction; + } + get armDirection() { + return left ? Vector3.left : Vector3.right; + } + get upperArmPos() { + return arm.upperArm.position; + } + get lowerArmPos() { + return arm.lowerArm.position; + } + get handPos() { + return arm.hand.position; + } + get shoulderAnker() { + return left ? shoulder.leftShoulderAnchor : shoulder.rightShoulderAnchor; + } + get upperArmRotation() { + return arm.upperArm.rotation * Quaternion.Inverse(upperArmStartRotation); + } + get lowerArmRotation() { + return arm.lowerArm.rotation * Quaternion.Inverse(lowerArmStartRotation); + } + get handRotation() { + return arm.hand.rotation * Quaternion.Inverse(handStartRotation); + } -} \ No newline at end of file + setUpperArmRotation(Quaternion rotation) { + return arm.upperArm.rotation = rotation * upperArmStartRotation; + } + setLowerArmRotation(Quaternion rotation) { + return arm.lowerArm.rotation = rotation * lowerArmStartRotation; + } + setLowerArmLocalRotation(Quaternion rotation) { + return arm.lowerArm.rotation = upperArmRotation * rotation * lowerArmStartRotation; + } + setWrist1Rotation(Quaternion rotation) { + return arm.wrist1.rotation = rotation * wristStartRotation; + } + setWrist2Rotation(Quaternion rotation) { + return arm.wrist2.rotation = rotation * wristStartRotation; + } + setWristLocalRotation(Quaternion rotation) { + return arm.wrist1.rotation = arm.lowerArm.rotation * rotation * wristStartRotation; + } + setHandRotation(Quaternion rotation) { + return arm.hand.rotation = arm.hand.rotation = rotation * handStartRotation; + } + } \ No newline at end of file From 40993518193453be8ff736fb35291cf1e55ea8c9 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 12:04:38 -0400 Subject: [PATCH 020/562] Remove vrarmik meta files --- vrarmik/ArmTransforms.cs.meta | 13 ------------- vrarmik/AvatarPlatformOffset.cs.meta | 11 ----------- vrarmik/AvatarVRTrackingReferences.cs.meta | 13 ------------- vrarmik/LocalVrTrackingInput.cs.meta | 13 ------------- vrarmik/PoseManager.cs.meta | 13 ------------- vrarmik/ShoulderPoser.cs.meta | 13 ------------- vrarmik/ShoulderTransforms.cs.meta | 13 ------------- vrarmik/StaticOffsetTransform.cs.meta | 13 ------------- vrarmik/Utils.meta | 10 ---------- vrarmik/VRArmIK.cs.meta | 13 ------------- vrarmik/VRTrackingReferences.cs.meta | 13 ------------- 11 files changed, 138 deletions(-) delete mode 100644 vrarmik/ArmTransforms.cs.meta delete mode 100644 vrarmik/AvatarPlatformOffset.cs.meta delete mode 100644 vrarmik/AvatarVRTrackingReferences.cs.meta delete mode 100644 vrarmik/LocalVrTrackingInput.cs.meta delete mode 100644 vrarmik/PoseManager.cs.meta delete mode 100644 vrarmik/ShoulderPoser.cs.meta delete mode 100644 vrarmik/ShoulderTransforms.cs.meta delete mode 100644 vrarmik/StaticOffsetTransform.cs.meta delete mode 100644 vrarmik/Utils.meta delete mode 100644 vrarmik/VRArmIK.cs.meta delete mode 100644 vrarmik/VRTrackingReferences.cs.meta diff --git a/vrarmik/ArmTransforms.cs.meta b/vrarmik/ArmTransforms.cs.meta deleted file mode 100644 index 5233ee2..0000000 --- a/vrarmik/ArmTransforms.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: ec032af568a757340beb92f8fea7c65a -timeCreated: 1528126260 -licenseType: Free -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/vrarmik/AvatarPlatformOffset.cs.meta b/vrarmik/AvatarPlatformOffset.cs.meta deleted file mode 100644 index b6f1e27..0000000 --- a/vrarmik/AvatarPlatformOffset.cs.meta +++ /dev/null @@ -1,11 +0,0 @@ -fileFormatVersion: 2 -guid: ebc3d0b087754e543af552b66b64c187 -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/vrarmik/AvatarVRTrackingReferences.cs.meta b/vrarmik/AvatarVRTrackingReferences.cs.meta deleted file mode 100644 index cfdb77c..0000000 --- a/vrarmik/AvatarVRTrackingReferences.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: 4ead1ab284cddee4ba96204f5eb416da -timeCreated: 1528126940 -licenseType: Free -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/vrarmik/LocalVrTrackingInput.cs.meta b/vrarmik/LocalVrTrackingInput.cs.meta deleted file mode 100644 index 9d7d5af..0000000 --- a/vrarmik/LocalVrTrackingInput.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: 6deb6fe736bb1fa4f920aa26496ebab4 -timeCreated: 1528128704 -licenseType: Free -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/vrarmik/PoseManager.cs.meta b/vrarmik/PoseManager.cs.meta deleted file mode 100644 index 8ccf569..0000000 --- a/vrarmik/PoseManager.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: 944b5c1596f68324a998b2ef2c5a363d -timeCreated: 1528126646 -licenseType: Free -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/vrarmik/ShoulderPoser.cs.meta b/vrarmik/ShoulderPoser.cs.meta deleted file mode 100644 index fe93a44..0000000 --- a/vrarmik/ShoulderPoser.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: e31d3179dbae073468ca3ac97ec5c295 -timeCreated: 1528126260 -licenseType: Free -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/vrarmik/ShoulderTransforms.cs.meta b/vrarmik/ShoulderTransforms.cs.meta deleted file mode 100644 index fab1c6c..0000000 --- a/vrarmik/ShoulderTransforms.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: 952d2a04990ab5d4e95d91263da82740 -timeCreated: 1528126260 -licenseType: Free -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/vrarmik/StaticOffsetTransform.cs.meta b/vrarmik/StaticOffsetTransform.cs.meta deleted file mode 100644 index 4489a65..0000000 --- a/vrarmik/StaticOffsetTransform.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: fbf684afebadf9e448a63ab5775ad7c8 -timeCreated: 1528126990 -licenseType: Free -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/vrarmik/Utils.meta b/vrarmik/Utils.meta deleted file mode 100644 index 59a4e8c..0000000 --- a/vrarmik/Utils.meta +++ /dev/null @@ -1,10 +0,0 @@ -fileFormatVersion: 2 -guid: 0e4d0a507fe834b4c8e75a936e694622 -folderAsset: yes -timeCreated: 1528126363 -licenseType: Free -DefaultImporter: - externalObjects: {} - userData: - assetBundleName: - assetBundleVariant: diff --git a/vrarmik/VRArmIK.cs.meta b/vrarmik/VRArmIK.cs.meta deleted file mode 100644 index 83cc958..0000000 --- a/vrarmik/VRArmIK.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: 2042600d9ae259848ad6a09a5402a741 -timeCreated: 1528126260 -licenseType: Free -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/vrarmik/VRTrackingReferences.cs.meta b/vrarmik/VRTrackingReferences.cs.meta deleted file mode 100644 index ff3b893..0000000 --- a/vrarmik/VRTrackingReferences.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: 3ba5e054e87d9864db4d157ceca462f3 -timeCreated: 1528126807 -licenseType: Free -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: From 6b0fc06fc331c4e0885bddca768c560b5b4c4ffc Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 12:08:32 -0400 Subject: [PATCH 021/562] Rename vrarmik .cs -> .js --- vrarmik/{ArmTransforms.cs => ArmTransforms.js} | 0 vrarmik/{AvatarPlatformOffset.cs => AvatarPlatformOffset.js} | 0 ...vatarVRTrackingReferences.cs => AvatarVRTrackingReferences.js} | 0 vrarmik/{LocalVrTrackingInput.cs => LocalVrTrackingInput.js} | 0 vrarmik/{PoseManager.cs => PoseManager.js} | 0 vrarmik/{ShoulderPoser.cs => ShoulderPoser.js} | 0 vrarmik/{ShoulderTransforms.cs => ShoulderTransforms.js} | 0 vrarmik/{StaticOffsetTransform.cs => StaticOffsetTransform.js} | 0 .../{LabelOverrideInspector.cs => LabelOverrideInspector.js} | 0 .../Utils/Editor/{ReadOnlyInspector.cs => ReadOnlyInspector.js} | 0 vrarmik/Utils/{Extensions.cs => Extensions.js} | 0 vrarmik/Utils/{InspectorAttributes.cs => InspectorAttributes.js} | 0 vrarmik/Utils/{VectorHelpers.cs => VectorHelpers.js} | 0 vrarmik/{VRArmIK.cs => VRArmIK.js} | 0 vrarmik/{VRTrackingReferences.cs => VRTrackingReferences.js} | 0 15 files changed, 0 insertions(+), 0 deletions(-) rename vrarmik/{ArmTransforms.cs => ArmTransforms.js} (100%) rename vrarmik/{AvatarPlatformOffset.cs => AvatarPlatformOffset.js} (100%) rename vrarmik/{AvatarVRTrackingReferences.cs => AvatarVRTrackingReferences.js} (100%) rename vrarmik/{LocalVrTrackingInput.cs => LocalVrTrackingInput.js} (100%) rename vrarmik/{PoseManager.cs => PoseManager.js} (100%) rename vrarmik/{ShoulderPoser.cs => ShoulderPoser.js} (100%) rename vrarmik/{ShoulderTransforms.cs => ShoulderTransforms.js} (100%) rename vrarmik/{StaticOffsetTransform.cs => StaticOffsetTransform.js} (100%) rename vrarmik/Utils/Editor/{LabelOverrideInspector.cs => LabelOverrideInspector.js} (100%) rename vrarmik/Utils/Editor/{ReadOnlyInspector.cs => ReadOnlyInspector.js} (100%) rename vrarmik/Utils/{Extensions.cs => Extensions.js} (100%) rename vrarmik/Utils/{InspectorAttributes.cs => InspectorAttributes.js} (100%) rename vrarmik/Utils/{VectorHelpers.cs => VectorHelpers.js} (100%) rename vrarmik/{VRArmIK.cs => VRArmIK.js} (100%) rename vrarmik/{VRTrackingReferences.cs => VRTrackingReferences.js} (100%) diff --git a/vrarmik/ArmTransforms.cs b/vrarmik/ArmTransforms.js similarity index 100% rename from vrarmik/ArmTransforms.cs rename to vrarmik/ArmTransforms.js diff --git a/vrarmik/AvatarPlatformOffset.cs b/vrarmik/AvatarPlatformOffset.js similarity index 100% rename from vrarmik/AvatarPlatformOffset.cs rename to vrarmik/AvatarPlatformOffset.js diff --git a/vrarmik/AvatarVRTrackingReferences.cs b/vrarmik/AvatarVRTrackingReferences.js similarity index 100% rename from vrarmik/AvatarVRTrackingReferences.cs rename to vrarmik/AvatarVRTrackingReferences.js diff --git a/vrarmik/LocalVrTrackingInput.cs b/vrarmik/LocalVrTrackingInput.js similarity index 100% rename from vrarmik/LocalVrTrackingInput.cs rename to vrarmik/LocalVrTrackingInput.js diff --git a/vrarmik/PoseManager.cs b/vrarmik/PoseManager.js similarity index 100% rename from vrarmik/PoseManager.cs rename to vrarmik/PoseManager.js diff --git a/vrarmik/ShoulderPoser.cs b/vrarmik/ShoulderPoser.js similarity index 100% rename from vrarmik/ShoulderPoser.cs rename to vrarmik/ShoulderPoser.js diff --git a/vrarmik/ShoulderTransforms.cs b/vrarmik/ShoulderTransforms.js similarity index 100% rename from vrarmik/ShoulderTransforms.cs rename to vrarmik/ShoulderTransforms.js diff --git a/vrarmik/StaticOffsetTransform.cs b/vrarmik/StaticOffsetTransform.js similarity index 100% rename from vrarmik/StaticOffsetTransform.cs rename to vrarmik/StaticOffsetTransform.js diff --git a/vrarmik/Utils/Editor/LabelOverrideInspector.cs b/vrarmik/Utils/Editor/LabelOverrideInspector.js similarity index 100% rename from vrarmik/Utils/Editor/LabelOverrideInspector.cs rename to vrarmik/Utils/Editor/LabelOverrideInspector.js diff --git a/vrarmik/Utils/Editor/ReadOnlyInspector.cs b/vrarmik/Utils/Editor/ReadOnlyInspector.js similarity index 100% rename from vrarmik/Utils/Editor/ReadOnlyInspector.cs rename to vrarmik/Utils/Editor/ReadOnlyInspector.js diff --git a/vrarmik/Utils/Extensions.cs b/vrarmik/Utils/Extensions.js similarity index 100% rename from vrarmik/Utils/Extensions.cs rename to vrarmik/Utils/Extensions.js diff --git a/vrarmik/Utils/InspectorAttributes.cs b/vrarmik/Utils/InspectorAttributes.js similarity index 100% rename from vrarmik/Utils/InspectorAttributes.cs rename to vrarmik/Utils/InspectorAttributes.js diff --git a/vrarmik/Utils/VectorHelpers.cs b/vrarmik/Utils/VectorHelpers.js similarity index 100% rename from vrarmik/Utils/VectorHelpers.cs rename to vrarmik/Utils/VectorHelpers.js diff --git a/vrarmik/VRArmIK.cs b/vrarmik/VRArmIK.js similarity index 100% rename from vrarmik/VRArmIK.cs rename to vrarmik/VRArmIK.js diff --git a/vrarmik/VRTrackingReferences.cs b/vrarmik/VRTrackingReferences.js similarity index 100% rename from vrarmik/VRTrackingReferences.cs rename to vrarmik/VRTrackingReferences.js From 4bc9052463302d4d721d64309c1bb03a72f68662 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 12:10:37 -0400 Subject: [PATCH 022/562] Clean up VRArmIK classes --- vrarmik/VRArmIK.js | 86 ++++++++++++++++++++++++---------------------- 1 file changed, 44 insertions(+), 42 deletions(-) diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index 16cd951..cc1066a 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -2,48 +2,48 @@ import ShoulderTransforms from './ShoulderTransforms.js'; import ShoulderPoser from './ShoulderPoser.js'; - public class VRArmIK +class ArmIKElbowSettings +{ + public bool calcElbowAngle = true; + public bool clampElbowAngle = true; + public bool softClampElbowAngle = true; + public float maxAngle = 175f, minAngle = 13f, softClampRange = 10f; + public float offsetAngle = 135f; + public float yWeight = -60f; + public float zWeightTop = 260, zWeightBottom = -100, zBorderY = -.25f, zDistanceStart = .6f; + public float xWeight = -50f, xDistanceStart = .1f; +} + +class BeforePositioningSettings +{ + public bool correctElbowOutside = true; + public float weight = -0.5f; + public float startBelowZ = .4f; + public float startAboveY = 0.1f; +} + +class ElbowCorrectionSettings +{ + public bool useFixedElbowWhenNearShoulder = true; + public float startBelowDistance = .5f; + public float startBelowY = 0.1f; + public float weight = 2f; + public Vector3 localElbowPos = new Vector3(0.3f, -1f, -2f); +} + +class HandSettings +{ + public bool useWristRotation = true; + public bool rotateElbowWithHandRight = true; + public bool rotateElbowWithHandForward = true; + public float handDeltaPow = 1.5f, handDeltaFactor = -.3f, handDeltaOffset = 45f; + // todo fix rotateElbowWithHandForward with factor != 1 -> horrible jumps. good value would be between [0.4, 0.6] + public float handDeltaForwardPow = 2f, handDeltaForwardFactor = 1f, handDeltaForwardOffset = 0f, handDeltaForwardDeadzone = .3f; + public float rotateElbowWithHandDelay = .08f; +} + + class VRArmIK { - class ArmIKElbowSettings - { - public bool calcElbowAngle = true; - public bool clampElbowAngle = true; - public bool softClampElbowAngle = true; - public float maxAngle = 175f, minAngle = 13f, softClampRange = 10f; - public float offsetAngle = 135f; - public float yWeight = -60f; - public float zWeightTop = 260, zWeightBottom = -100, zBorderY = -.25f, zDistanceStart = .6f; - public float xWeight = -50f, xDistanceStart = .1f; - } - - class BeforePositioningSettings - { - public bool correctElbowOutside = true; - public float weight = -0.5f; - public float startBelowZ = .4f; - public float startAboveY = 0.1f; - } - - class ElbowCorrectionSettings - { - public bool useFixedElbowWhenNearShoulder = true; - public float startBelowDistance = .5f; - public float startBelowY = 0.1f; - public float weight = 2f; - public Vector3 localElbowPos = new Vector3(0.3f, -1f, -2f); - } - - class HandSettings - { - public bool useWristRotation = true; - public bool rotateElbowWithHandRight = true; - public bool rotateElbowWithHandForward = true; - public float handDeltaPow = 1.5f, handDeltaFactor = -.3f, handDeltaOffset = 45f; - // todo fix rotateElbowWithHandForward with factor != 1 -> horrible jumps. good value would be between [0.4, 0.6] - public float handDeltaForwardPow = 2f, handDeltaForwardFactor = 1f, handDeltaForwardOffset = 0f, handDeltaForwardDeadzone = .3f; - public float rotateElbowWithHandDelay = .08f; - } - public ArmTransforms arm; public ShoulderTransforms shoulder; public ShoulderPoser shoulderPoser; @@ -395,4 +395,6 @@ import ShoulderPoser from './ShoulderPoser.js'; setHandRotation(Quaternion rotation) { return arm.hand.rotation = arm.hand.rotation = rotation * handStartRotation; } - } \ No newline at end of file + } + +export default VRArmIK; \ No newline at end of file From c64c0ead5af5dd575091634a01ace79b9a260ab6 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 12:12:59 -0400 Subject: [PATCH 023/562] Debugging class exports --- vrarmik/ArmTransforms.js | 4 +++- vrarmik/ShoulderPoser.js | 4 +++- vrarmik/ShoulderTransforms.js | 4 +++- 3 files changed, 9 insertions(+), 3 deletions(-) diff --git a/vrarmik/ArmTransforms.js b/vrarmik/ArmTransforms.js index c7dc298..2747be0 100644 --- a/vrarmik/ArmTransforms.js +++ b/vrarmik/ArmTransforms.js @@ -75,4 +75,6 @@ setLowerArmLength(length * (1f - upperArmFactor)); } } - } \ No newline at end of file + } + +export default ArmTransforms; \ No newline at end of file diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index a68cc0c..bc6cf61 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -278,4 +278,6 @@ shoulder.rightArm.transform.localPosition = Vector3.zero; } } - } \ No newline at end of file + } + +export default ShoulderPoser; \ No newline at end of file diff --git a/vrarmik/ShoulderTransforms.js b/vrarmik/ShoulderTransforms.js index 65b53f2..e1cfc28 100644 --- a/vrarmik/ShoulderTransforms.js +++ b/vrarmik/ShoulderTransforms.js @@ -42,4 +42,6 @@ rightShoulderRenderer.localScale = localScale; rightShoulderRenderer.localPosition = localPosition; } - } \ No newline at end of file + } + +export default ShoulderTransforms; \ No newline at end of file From df54b25584474dfa69cc7fd545f9a265f3652b66 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 12:29:57 -0400 Subject: [PATCH 024/562] JSify VRArmIK.js --- vrarmik/VRArmIK.js | 172 ++++++++++++++++++++++++++------------------- 1 file changed, 98 insertions(+), 74 deletions(-) diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index cc1066a..dafb873 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -4,63 +4,87 @@ import ShoulderPoser from './ShoulderPoser.js'; class ArmIKElbowSettings { - public bool calcElbowAngle = true; - public bool clampElbowAngle = true; - public bool softClampElbowAngle = true; - public float maxAngle = 175f, minAngle = 13f, softClampRange = 10f; - public float offsetAngle = 135f; - public float yWeight = -60f; - public float zWeightTop = 260, zWeightBottom = -100, zBorderY = -.25f, zDistanceStart = .6f; - public float xWeight = -50f, xDistanceStart = .1f; + constructor() { + this.calcElbowAngle = true; + this.clampElbowAngle = true; + this.softClampElbowAngle = true; + this.maxAngle = 175; + this.minAngle = 13; + this.softClampRange = 10; + this.offsetAngle = 135; + this.yWeight = -60; + this.zWeightTop = 260; + this.zWeightBottom = -100; + this.zBorderY = -.25; + this.zDistanceStart = .6; + this.xWeight = -50; + this.xDistanceStart = .1; + } } class BeforePositioningSettings { - public bool correctElbowOutside = true; - public float weight = -0.5f; - public float startBelowZ = .4f; - public float startAboveY = 0.1f; + constructor() { + this.correctElbowOutside = true; + this.weight = -0.5; + this.startBelowZ = .4; + this.startAboveY = 0.1; + } } class ElbowCorrectionSettings { - public bool useFixedElbowWhenNearShoulder = true; - public float startBelowDistance = .5f; - public float startBelowY = 0.1f; - public float weight = 2f; - public Vector3 localElbowPos = new Vector3(0.3f, -1f, -2f); + constructor() { + this.useFixedElbowWhenNearShoulder = true; + this.startBelowDistance = .5; + this.startBelowY = 0.1; + this.weight = 2; + this.localElbowPos = new Vector3(0.3, -1, -2); + } } class HandSettings { - public bool useWristRotation = true; - public bool rotateElbowWithHandRight = true; - public bool rotateElbowWithHandForward = true; - public float handDeltaPow = 1.5f, handDeltaFactor = -.3f, handDeltaOffset = 45f; - // todo fix rotateElbowWithHandForward with factor != 1 -> horrible jumps. good value would be between [0.4, 0.6] - public float handDeltaForwardPow = 2f, handDeltaForwardFactor = 1f, handDeltaForwardOffset = 0f, handDeltaForwardDeadzone = .3f; - public float rotateElbowWithHandDelay = .08f; + constructor() { + this.useWristRotation = true; + this.rotateElbowWithHandRight = true; + this.rotateElbowWithHandForward = true; + this.handDeltaPow = 1.5; + this.handDeltaFactor = -.3; + this.handDeltaOffset = 45; + // todo fix rotateElbowWithHandForward with factor != 1 -> horrible jumps. good value would be between [0.4, 0.6] + this.handDeltaForwardPow = 2; + this.handDeltaForwardFactor = 1; + this.handDeltaForwardOffset = 0; + this.handDeltaForwardDeadzone = .3; + this.rotateElbowWithHandDelay = .08; + } } class VRArmIK { - public ArmTransforms arm; - public ShoulderTransforms shoulder; - public ShoulderPoser shoulderPoser; - public Transform target; - public bool left = true; - - public ArmIKElbowSettings elbowSettings; - public BeforePositioningSettings beforePositioningSettings; - public ElbowCorrectionSettings elbowCorrectionSettings; - public HandSettings handSettings; - - Vector3 nextLowerArmAngle; - - Quaternion upperArmStartRotation, lowerArmStartRotation, wristStartRotation, handStartRotation; - - float interpolatedDeltaElbow; - float interpolatedDeltaElbowForward; + constructor() { + this.arm = new ArmTransforms(); // XXX these need to be this'd below + this.shoulder = ShoulderTransforms(); + this.shoulderPoser = new ShoulderPoser(); + this.target = new Transform(); + this.left = true; + + this.elbowSettings = new ArmIKElbowSettings(); + this.beforePositioningSettings = new BeforePositioningSettings(); + this.elbowCorrectionSettings = ElbowCorrectionSettings(); + this.handSettings = new HandSettings(); + + this.nextLowerArmAngle = new Vector3(); + + this.upperArmStartRotation = new Quaternion(); + this.lowerArmStartRotation = new Quaternion(); + this.wristStartRotation = new Quaternion(); + this.handStartRotation = new Quaternion(); + + this.interpolatedDeltaElbow = 0; + this.interpolatedDeltaElbowForward = 0; + } Awake() { @@ -119,19 +143,19 @@ class HandSettings if (targetShoulderDistance > arm.armLength) { - innerAngle = 0f; + innerAngle = 0; } else { - innerAngle = Mathf.Acos(Mathf.Clamp((Mathf.Pow(arm.upperArmLength, 2f) + Mathf.Pow(arm.lowerArmLength, 2f) - - Mathf.Pow(targetShoulderDistance, 2f)) / (2f * arm.upperArmLength * arm.lowerArmLength), -1f, 1f)) * Mathf.Rad2Deg; + innerAngle = Mathf.Acos(Mathf.Clamp((Mathf.Pow(arm.upperArmLength, 2) + Mathf.Pow(arm.lowerArmLength, 2) - + Mathf.Pow(targetShoulderDistance, 2)) / (2 * arm.upperArmLength * arm.lowerArmLength), -1, 1)) * Mathf.Rad2Deg; if (left) - innerAngle = 180f - innerAngle; + innerAngle = 180 - innerAngle; else - innerAngle = 180f + innerAngle; + innerAngle = 180 + innerAngle; if (float.IsNaN(innerAngle)) { - innerAngle = 180f; + innerAngle = 180; } } @@ -146,11 +170,11 @@ class HandSettings Vector3 targetShoulderDirection = (target.position - upperArmPos).normalized; float targetShoulderDistance = (target.position - upperArmPos).magnitude; - eulerAngles.y = (left ? -1f : 1f) * - Mathf.Acos(Mathf.Clamp((Mathf.Pow(targetShoulderDistance, 2f) + Mathf.Pow(arm.upperArmLength, 2f) - - Mathf.Pow(arm.lowerArmLength, 2f)) / (2f * targetShoulderDistance * arm.upperArmLength), -1f, 1f)) * Mathf.Rad2Deg; + eulerAngles.y = (left ? -1 : 1) * + Mathf.Acos(Mathf.Clamp((Mathf.Pow(targetShoulderDistance, 2) + Mathf.Pow(arm.upperArmLength, 2) - + Mathf.Pow(arm.lowerArmLength, 2)) / (2 * targetShoulderDistance * arm.upperArmLength), -1, 1)) * Mathf.Rad2Deg; if (float.IsNaN(eulerAngles.y)) - eulerAngles.y = 0f; + eulerAngles.y = 0; Quaternion shoulderRightRotation = Quaternion.FromToRotation(armDirection, targetShoulderDirection); @@ -170,13 +194,13 @@ class HandSettings /*angle += Mathf.Lerp(elbowSettings.zWeightBottom, elbowSettings.zWeightTop, Mathf.Clamp01((localHandPosNormalized.y + 1f) - elbowSettings.zBorderY)) * (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0f));*/ if (localHandPosNormalized.y > 0) - angle += elbowSettings.zWeightTop * (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0f) * Mathf.Max(localHandPosNormalized.y, 0f)); + angle += elbowSettings.zWeightTop * (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0) * Mathf.Max(localHandPosNormalized.y, 0)); else - angle += elbowSettings.zWeightBottom * (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0f) * Mathf.Max(-localHandPosNormalized.y, 0f)); + angle += elbowSettings.zWeightBottom * (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0) * Mathf.Max(-localHandPosNormalized.y, 0)); // angle from X - angle += elbowSettings.xWeight * Mathf.Max(localHandPosNormalized.x * (left ? 1.0f : -1.0f) + elbowSettings.xDistanceStart, 0f); + angle += elbowSettings.xWeight * Mathf.Max(localHandPosNormalized.x * (left ? 1.0 : -1.0) + elbowSettings.xDistanceStart, 0); if (elbowSettings.clampElbowAngle) { @@ -185,7 +209,7 @@ class HandSettings if (angle < elbowSettings.minAngle + elbowSettings.softClampRange) { float a = elbowSettings.minAngle + elbowSettings.softClampRange - angle; - angle = elbowSettings.minAngle + elbowSettings.softClampRange * (1f - Mathf.Log(1f + a) * 3f); + angle = elbowSettings.minAngle + elbowSettings.softClampRange * (1 - Mathf.Log(1 + a) * 3); } } else @@ -195,7 +219,7 @@ class HandSettings } if (left) - angle *= -1f; + angle *= -1; return angle; } @@ -207,20 +231,20 @@ class HandSettings Vector3 localTargetPos = shoulderAnker.InverseTransformPoint(target.position) / arm.armLength; float elbowOutsideFactor = Mathf.Clamp01( Mathf.Clamp01((s.startBelowZ - localTargetPos.z) / - Mathf.Abs(s.startBelowZ) * .5f) * + Mathf.Abs(s.startBelowZ) * .5) * Mathf.Clamp01((localTargetPos.y - s.startAboveY) / Mathf.Abs(s.startAboveY)) * - Mathf.Clamp01(1f - localTargetPos.x * (left ? -1f : 1f)) + Mathf.Clamp01(1 - localTargetPos.x * (left ? -1 : 1)) ) * s.weight; Vector3 shoulderHandDirection = (upperArmPos - handPos).normalized; - Vector3 targetDir = shoulder.transform.rotation * (Vector3.up + (s.correctElbowOutside ? (armDirection + Vector3.forward * -.2f) * elbowOutsideFactor : Vector3.zero)); - Vector3 cross = Vector3.Cross(shoulderHandDirection, targetDir * 1000f); + Vector3 targetDir = shoulder.transform.rotation * (Vector3.up + (s.correctElbowOutside ? (armDirection + Vector3.forward * -.2) * elbowOutsideFactor : Vector3.zero)); + Vector3 cross = Vector3.Cross(shoulderHandDirection, targetDir * 1000); Vector3 upperArmUp = upperArmRotation * Vector3.up; float elbowTargetUp = Vector3.Dot(upperArmUp, targetDir); - float elbowAngle = Vector3.Angle(cross, upperArmUp) + (left ? 0f : 180f); + float elbowAngle = Vector3.Angle(cross, upperArmUp) + (left ? 0 : 180); Quaternion rotation = Quaternion.AngleAxis(elbowAngle * Mathf.Sign(elbowTargetUp), shoulderHandDirection); arm.upperArm.rotation = rotation * arm.upperArm.rotation; } @@ -236,7 +260,7 @@ class HandSettings Vector3 elbowPos = s.localElbowPos; if (left) - elbowPos.x *= -1f; + elbowPos.x *= -1; Vector3 targetDir = shoulder.transform.rotation * elbowPos.normalized; Vector3 cross = Vector3.Cross(shoulderHandDirection, targetDir); @@ -248,13 +272,13 @@ class HandSettings distance = distance.magnitude * shoulder.transform.InverseTransformDirection(distance / distance.magnitude); float weight = Mathf.Clamp01(Mathf.Clamp01((s.startBelowDistance - distance.xz().magnitude / arm.armLength) / - s.startBelowDistance) * s.weight + Mathf.Clamp01((-distance.z + .1f) * 3)) * + s.startBelowDistance) * s.weight + Mathf.Clamp01((-distance.z + .1) * 3)) * Mathf.Clamp01((s.startBelowY - localTargetPos.y) / s.startBelowY); float elbowTargetUp = Vector3.Dot(upperArmUp, targetDir); - float elbowAngle2 = Vector3.Angle(cross, upperArmUp) + (left ? 0f : 180f); - Quaternion rotation = Quaternion.AngleAxis((elbowAngle2 * Mathf.Sign(elbowTargetUp)).toSignedEulerAngle() * Mathf.Clamp(weight, 0, 1f), shoulderHandDirection); + float elbowAngle2 = Vector3.Angle(cross, upperArmUp) + (left ? 0 : 180); + Quaternion rotation = Quaternion.AngleAxis((elbowAngle2 * Mathf.Sign(elbowTargetUp)).toSignedEulerAngle() * Mathf.Clamp(weight, 0, 1), shoulderHandDirection); arm.upperArm.rotation = rotation * arm.upperArm.rotation; } @@ -288,9 +312,9 @@ class HandSettings float elbowTargetAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.up, handUpVec, lowerArmRotation * Vector3.forward, lowerArmRotation * armDirection); - float deltaElbow = (elbowTargetAngle + (left ? -s.handDeltaOffset : s.handDeltaOffset)) / 180f; + float deltaElbow = (elbowTargetAngle + (left ? -s.handDeltaOffset : s.handDeltaOffset)) / 180; - deltaElbow = Mathf.Sign(deltaElbow) * Mathf.Pow(Mathf.Abs(deltaElbow), s.handDeltaPow) * 180f * s.handDeltaFactor; + deltaElbow = Mathf.Sign(deltaElbow) * Mathf.Pow(Mathf.Abs(deltaElbow), s.handDeltaPow) * 180 * s.handDeltaFactor; interpolatedDeltaElbow = Mathf.LerpAngle(interpolatedDeltaElbow, deltaElbow, Time.deltaTime / s.rotateElbowWithHandDelay); rotateElbow(interpolatedDeltaElbow); @@ -304,16 +328,16 @@ class HandSettings float elbowTargetAngleForward = VectorHelpers.getAngleBetween(lowerArmRotation * armDirection, handRightVec, lowerArmRotation * Vector3.up, lowerArmRotation * Vector3.forward); - float deltaElbowForward = (elbowTargetAngleForward + (left ? -s.handDeltaForwardOffset : s.handDeltaForwardOffset)) / 180f; + float deltaElbowForward = (elbowTargetAngleForward + (left ? -s.handDeltaForwardOffset : s.handDeltaForwardOffset)) / 180; if (Mathf.Abs(deltaElbowForward) < s.handDeltaForwardDeadzone) - deltaElbowForward = 0f; + deltaElbowForward = 0; else { - deltaElbowForward = (deltaElbowForward - Mathf.Sign(deltaElbowForward) * s.handDeltaForwardDeadzone) / (1f - s.handDeltaForwardDeadzone); + deltaElbowForward = (deltaElbowForward - Mathf.Sign(deltaElbowForward) * s.handDeltaForwardDeadzone) / (1 - s.handDeltaForwardDeadzone); } - deltaElbowForward = Mathf.Sign(deltaElbowForward) * Mathf.Pow(Mathf.Abs(deltaElbowForward), s.handDeltaForwardPow) * 180f; + deltaElbowForward = Mathf.Sign(deltaElbowForward) * Mathf.Pow(Mathf.Abs(deltaElbowForward), s.handDeltaForwardPow) * 180; interpolatedDeltaElbowForward = Mathf.LerpAngle(interpolatedDeltaElbowForward, deltaElbowForward, Time.deltaTime / s.rotateElbowWithHandDelay); float signedInterpolated = interpolatedDeltaElbowForward.toSignedEulerAngle(); @@ -335,11 +359,11 @@ class HandSettings float elbowTargetAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.up, handUpVec, lowerArmRotation * Vector3.forward, lowerArmRotation * armDirection); - elbowTargetAngle = Mathf.Clamp(elbowTargetAngle, -90f, 90f); + elbowTargetAngle = Mathf.Clamp(elbowTargetAngle, -90, 90); if (arm.wrist1 != null) - setWrist1Rotation(Quaternion.AngleAxis(elbowTargetAngle * .3f, lowerArmRotation * armDirection) * lowerArmRotation); + setWrist1Rotation(Quaternion.AngleAxis(elbowTargetAngle * .3, lowerArmRotation * armDirection) * lowerArmRotation); if (arm.wrist2 != null) - setWrist2Rotation(Quaternion.AngleAxis(elbowTargetAngle * .8f, lowerArmRotation * armDirection) * lowerArmRotation); + setWrist2Rotation(Quaternion.AngleAxis(elbowTargetAngle * .8, lowerArmRotation * armDirection) * lowerArmRotation); } setHandRotation(target.rotation); } From 28d0f2af1e38085c9d6c0d88f21e254d6fa2764f Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 12:53:01 -0400 Subject: [PATCH 025/562] More VRArmIk porting work --- vrarmik/ArmTransforms.js | 36 +- vrarmik/AvatarPlatformOffset.js | 2 +- vrarmik/AvatarVRTrackingReferences.js | 2 +- vrarmik/LocalVrTrackingInput.js | 2 +- vrarmik/PoseManager.js | 2 +- vrarmik/ShoulderPoser.js | 103 ++-- vrarmik/ShoulderTransforms.js | 101 +-- vrarmik/StaticOffsetTransform.js | 2 +- vrarmik/Utils/Extensions.js | 78 +-- vrarmik/Utils/InspectorAttributes.js | 26 +- vrarmik/Utils/VectorHelpers.js | 4 +- vrarmik/VRArmIK.js | 846 +++++++++++++------------- vrarmik/VRTrackingReferences.js | 20 +- 13 files changed, 625 insertions(+), 599 deletions(-) diff --git a/vrarmik/ArmTransforms.js b/vrarmik/ArmTransforms.js index 2747be0..31ffada 100644 --- a/vrarmik/ArmTransforms.js +++ b/vrarmik/ArmTransforms.js @@ -1,14 +1,26 @@ -class ArmTransforms +class ArmTransforms { - Transform upperArm, lowerArm, wrist1, wrist2, hand; + constructor() { + this.upperArm = new Transform(); + this.lowerArm = new Transform(); + this.wrist1 = new Transform(); + this.wrist2 = new Transform(); + this.hand = new Transform(); - float upperArmLength => distance(upperArm, lowerArm); - float lowerArmLength => distance(lowerArm, hand); - float armLength => upperArmLength + lowerArmLength; + this.armLengthByScale = false; + this.scaleAxis = Vector3.one; + this.scaleHandFactor = .7; + } - bool armLengthByScale = false; - Vector3 scaleAxis = Vector3.one; - float scaleHandFactor = .7f; + get upperArmLength() { + return distance(upperArm, lowerArm); + } + get lowerArmLength() { + return distance(lowerArm, hand); + } + get armLength() { + return upperArmLength + lowerArmLength; + } distance(Transform a, Transform b) { return (a.position - b.position).magnitude; @@ -23,7 +35,7 @@ updateArmLengths() { var shoulderWidth = (upperArm.position - lowerArm.position).magnitude; - var _armLength = (PoseManager.Instance.playerWidthWrist - shoulderWidth) / 2f; + var _armLength = (PoseManager.Instance.playerWidthWrist - shoulderWidth) / 2; setArmLength(_armLength); } @@ -63,16 +75,16 @@ setArmLength(float length) { - float upperArmFactor = .48f; + float upperArmFactor = .48; if (armLengthByScale) { upperArm.localScale = upperArm.localScale / armLength * length; - hand.localScale = Vector3.one / (1f - (1f - scaleHandFactor) * (1f - upperArm.localScale.x)); + hand.localScale = Vector3.one / (1 - (1 - scaleHandFactor) * (1 - upperArm.localScale.x)); } else { setUpperArmLength(length * upperArmFactor); - setLowerArmLength(length * (1f - upperArmFactor)); + setLowerArmLength(length * (1 - upperArmFactor)); } } } diff --git a/vrarmik/AvatarPlatformOffset.js b/vrarmik/AvatarPlatformOffset.js index cbc916b..dee5972 100644 --- a/vrarmik/AvatarPlatformOffset.js +++ b/vrarmik/AvatarPlatformOffset.js @@ -1,4 +1,4 @@ -using System.Collections; +using System.Collections; using System.Collections.Generic; using UnityEngine; using VRArmIK; diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index bb3bf38..35c0ba2 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -1,4 +1,4 @@ -using UnityEngine; +using UnityEngine; namespace VRArmIK { diff --git a/vrarmik/LocalVrTrackingInput.js b/vrarmik/LocalVrTrackingInput.js index abaea21..c663c91 100644 --- a/vrarmik/LocalVrTrackingInput.js +++ b/vrarmik/LocalVrTrackingInput.js @@ -1,4 +1,4 @@ -using System.Collections; +using System.Collections; using System.Collections.Generic; using UnityEngine; using UnityEngine.VR; diff --git a/vrarmik/PoseManager.js b/vrarmik/PoseManager.js index 62e14d3..754a768 100644 --- a/vrarmik/PoseManager.js +++ b/vrarmik/PoseManager.js @@ -1,4 +1,4 @@ -using UnityEngine; +using UnityEngine; using UnityEngine.XR; namespace VRArmIK diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index bc6cf61..532b71a 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -1,47 +1,52 @@ -class ShoulderPoser +import ShoulderTransforms from './ShoulderTransforms.js'; + +class ShoulderPoser { - ShoulderTransforms shoulder; - VRTrackingReferences vrTrackingReferences; - AvatarVRTrackingReferences avatarTrackingReferences; + constructor() { + this.shoulder = new ShoulderTransforms(); + this.vrTrackingReferences = new VRTrackingReferences(); + this.avatarTrackingReferences = AvatarVRTrackingReferences(); - float headNeckDistance = 0.03f; - Vector3 neckShoulderDistance = new Vector3(0f, -.1f, -0.02f); + this.headNeckDistance = 0.03; + this.neckShoulderDistance = new Vector3(0, -.1, -0.02); - float maxDeltaHeadRotation = 80f; + this.maxDeltaHeadRotation = 80; - float distinctShoulderRotationLimitForward = 33f; + this.distinctShoulderRotationLimitForward = 33; - float distinctShoulderRotationLimitBackward = 0f; + this.distinctShoulderRotationLimitBackward = 0; - float distinctShoulderRotationLimitUpward = 33f; - float distinctShoulderRotationMultiplier = 30; + this.distinctShoulderRotationLimitUpward = 33; + this.distinctShoulderRotationMultiplier = 30; - float rightRotationStartHeight = 0f; - float rightRotationHeightFactor = 142f; - float rightRotationHeadRotationFactor = 0.3f; - float rightRotationHeadRotationOffset = -20f; + this.rightRotationStartHeight = 0; + this.rightRotationHeightFactor = 142; + this.rightRotationHeadRotationFactor = 0.3; + this.rightRotationHeadRotationOffset = -20; - Vector3 headNeckDirectionVector = new Vector3(0f, -1f, -.05f); - float startShoulderDislocationBefore = 0.005f; + this.headNeckDirectionVector = new Vector3(0, -1, -.05); + this.startShoulderDislocationBefore = 0.005; - bool ignoreYPos = true; - bool autoDetectHandsBehindHead = true; - bool clampRotationToHead = true; - bool enableDistinctShoulderRotation = true; - bool enableShoulderDislocation = true; + this.ignoreYPos = true; + this.autoDetectHandsBehindHead = true; + this.clampRotationToHead = true; + this.enableDistinctShoulderRotation = true; + this.enableShoulderDislocation = true; - bool handsBehindHead = false; + this.handsBehindHead = false; - bool clampingHeadRotation = false; - bool shoulderDislocated = false; - float shoulderRightRotation; + this.clampingHeadRotation = false; + this.shoulderDislocated = false; + this.shoulderRightRotation; - Vector3 lastAngle = Vector3.zero; + this.lastAngle = Vector3.zero; - Vector3 leftShoulderAnkerStartLocalPosition, rightShoulderAnkerStartLocalPosition; + this.leftShoulderAnkerStartLocalPosition = new Vector3(); + this.rightShoulderAnkerStartLocalPosition = new Vector3(); + } Start() { @@ -90,14 +95,14 @@ rotateLeftShoulder() { rotateShoulderUp(shoulder.leftShoulder, shoulder.leftArm, avatarTrackingReferences.leftHand.transform, - leftShoulderAnkerStartLocalPosition, 1f); + leftShoulderAnkerStartLocalPosition, 1); } rotateRightShoulder() { rotateShoulderUp(shoulder.rightShoulder, shoulder.rightArm, avatarTrackingReferences.rightHand.transform, - rightShoulderAnkerStartLocalPosition, -1f); + rightShoulderAnkerStartLocalPosition, -1); } rotateShoulderUp(Transform shoulderSide, ArmTransforms arm, Transform targetHand, @@ -111,19 +116,19 @@ float forwardDistanceRatio = Vector3.Dot(handShoulderOffset, shoulder.transform.forward) / armLength; float upwardDistanceRatio = Vector3.Dot(handShoulderOffset, shoulder.transform.up) / armLength; - if (forwardDistanceRatio > 0f) + if (forwardDistanceRatio > 0) { - targetAngle.y = Mathf.Clamp((forwardDistanceRatio - 0.5f) * distinctShoulderRotationMultiplier, 0f, + targetAngle.y = Mathf.Clamp((forwardDistanceRatio - 0.5) * distinctShoulderRotationMultiplier, 0, distinctShoulderRotationLimitForward); } else { - targetAngle.y = Mathf.Clamp(-(forwardDistanceRatio + 0.08f) * distinctShoulderRotationMultiplier * 10f, - -distinctShoulderRotationLimitBackward, 0f); + targetAngle.y = Mathf.Clamp(-(forwardDistanceRatio + 0.08) * distinctShoulderRotationMultiplier * 10, + -distinctShoulderRotationLimitBackward, 0); } - targetAngle.z = Mathf.Clamp(-(upwardDistanceRatio - 0.5f) * distinctShoulderRotationMultiplier, - -distinctShoulderRotationLimitUpward, 0f); + targetAngle.z = Mathf.Clamp(-(upwardDistanceRatio - 0.5) * distinctShoulderRotationMultiplier, + -distinctShoulderRotationLimitUpward, 0); shoulderSide.localEulerAngles = targetAngle * angleSign; } @@ -141,7 +146,7 @@ { float angle = getCombinedDirectionAngleUp(); - Vector3 targetRotation = new Vector3(0f, angle, 0f); + Vector3 targetRotation = new Vector3(0, angle, 0); if (autoDetectHandsBehindHead) { @@ -164,11 +169,11 @@ float headRightRotation = VectorHelpers.getAngleBetween(shoulder.transform.forward, avatarTrackingReferences.hmd.transform.forward, Vector3.up, shoulder.transform.right) + rightRotationHeadRotationOffset; - float heightFactor = Mathf.Clamp(relativeHeightDiff - rightRotationStartHeight, 0f, 1f); + float heightFactor = Mathf.Clamp(relativeHeightDiff - rightRotationStartHeight, 0, 1); shoulderRightRotation = heightFactor * rightRotationHeightFactor; - shoulderRightRotation += Mathf.Clamp(headRightRotation * rightRotationHeadRotationFactor * heightFactor, 0f, 50f); + shoulderRightRotation += Mathf.Clamp(headRightRotation * rightRotationHeadRotationFactor * heightFactor, 0, 50); - shoulderRightRotation = Mathf.Clamp(shoulderRightRotation, 0f, 50f); + shoulderRightRotation = Mathf.Clamp(shoulderRightRotation, 0, 50); Quaternion deltaRot = Quaternion.AngleAxis(shoulderRightRotation, shoulder.transform.right); @@ -202,13 +207,13 @@ Vector3 combinedDirection = directionLeftHand + directionRightHand; - return Mathf.Atan2(combinedDirection.x, combinedDirection.z) * 180f / Mathf.PI; + return Mathf.Atan2(combinedDirection.x, combinedDirection.z) * 180 / Mathf.PI; } detectHandsBehindHead(ref Vector3 targetRotation) { - float delta = Mathf.Abs(targetRotation.y - lastAngle.y + 360f) % 360f; - if (delta > 150f && delta < 210f && lastAngle.magnitude > 0.000001f && !clampingHeadRotation) + float delta = Mathf.Abs(targetRotation.y - lastAngle.y + 360) % 360; + if (delta > 150 && delta < 210 && lastAngle.magnitude > 0.000001 && !clampingHeadRotation) { handsBehindHead = !handsBehindHead; } @@ -217,23 +222,23 @@ if (handsBehindHead) { - targetRotation.y += 180f; + targetRotation.y += 180; } } clampHeadRotationDeltaUp(ref Vector3 targetRotation) { - float headUpRotation = (avatarTrackingReferences.head.transform.eulerAngles.y + 360f) % 360f; - float targetUpRotation = (targetRotation.y + 360f) % 360f; + float headUpRotation = (avatarTrackingReferences.head.transform.eulerAngles.y + 360) % 360; + float targetUpRotation = (targetRotation.y + 360) % 360; float delta = headUpRotation - targetUpRotation; - if (delta > maxDeltaHeadRotation && delta < 180f || delta < -180f && delta >= -360f + maxDeltaHeadRotation) + if (delta > maxDeltaHeadRotation && delta < 180 || delta < -180 && delta >= -360 + maxDeltaHeadRotation) { targetRotation.y = headUpRotation - maxDeltaHeadRotation; clampingHeadRotation = true; } - else if (delta < -maxDeltaHeadRotation && delta > -180 || delta > 180f && delta < 360f - maxDeltaHeadRotation) + else if (delta < -maxDeltaHeadRotation && delta > -180 || delta > 180 && delta < 360 - maxDeltaHeadRotation) { targetRotation.y = headUpRotation + maxDeltaHeadRotation; clampingHeadRotation = true; @@ -251,7 +256,7 @@ float leftShoulderHandDistance = leftHandVector.magnitude, rightShoulderHandDistance = rightHandVector.magnitude; shoulderDislocated = false; - float startBeforeFactor = (1f - startShoulderDislocationBefore); + float startBeforeFactor = (1 - startShoulderDislocationBefore); if (leftShoulderHandDistance > shoulder.leftArm.armLength * startBeforeFactor) { diff --git a/vrarmik/ShoulderTransforms.js b/vrarmik/ShoulderTransforms.js index e1cfc28..d9b4be0 100644 --- a/vrarmik/ShoulderTransforms.js +++ b/vrarmik/ShoulderTransforms.js @@ -1,47 +1,56 @@ -class ShoulderTransforms - { - Transform leftShoulder, rightShoulder; - Transform leftShoulderRenderer, rightShoulderRenderer; - Transform leftShoulderAnchor, rightShoulderAnchor; - ArmTransforms leftArmDummy, rightArmDummy; - ArmTransforms leftArm, rightArm; - - Awake() - { - if (leftArm == null) - { - leftArm = Instantiate(leftArmDummy, leftShoulderAnchor.position, leftShoulderAnchor.rotation, leftShoulderAnchor); - var armIk = leftArm.GetComponentInChildren(); - armIk.shoulder = this; - armIk.shoulderPoser = GetComponent(); - armIk.target = armIk.shoulderPoser.avatarTrackingReferences.leftHand.transform; - } - if (rightArm == null) - { - rightArm = Instantiate(rightArmDummy, leftShoulderAnchor.position, rightShoulderAnchor.rotation, rightShoulderAnchor); - var armIk = rightArm.GetComponentInChildren(); - armIk.shoulder = this; - armIk.shoulderPoser = GetComponent(); - armIk.target = armIk.shoulderPoser.avatarTrackingReferences.rightHand.transform; - } - } - - Start() - { - setShoulderWidth(PoseManager.Instance.playerWidthShoulders); - } - - setShoulderWidth(float width) - { - Vector3 localScale = new Vector3(width * .5f, .05f, .05f); - Vector3 localPosition = new Vector3(width * .25f, 0f, 0f); - - leftShoulderRenderer.localScale = localScale; - leftShoulderRenderer.localPosition = -localPosition; - - rightShoulderRenderer.localScale = localScale; - rightShoulderRenderer.localPosition = localPosition; - } - } - +import ArmTransforms from './ArmTransforms.js'; + +class ShoulderTransforms + { + constructor() { + this.leftShoulder = new Transform(); + this.rightShoulder = new Transform(); + this.leftShoulderRenderer = new Transform(); + this.rightShoulderRenderer = new Transform(); + this.leftShoulderAnchor = Transform(); + this.rightShoulderAnchor = Transform(); + this.leftArmDummy = new ArmTransforms(); + this.rightArmDummy = new ArmTransforms(); + this.leftArm = new ArmTransforms(); + this.rightArm = new ArmTransforms(); + } + + Awake() + { + if (leftArm == null) + { + leftArm = Instantiate(leftArmDummy, leftShoulderAnchor.position, leftShoulderAnchor.rotation, leftShoulderAnchor); + var armIk = leftArm.GetComponentInChildren(); + armIk.shoulder = this; + armIk.shoulderPoser = GetComponent(); + armIk.target = armIk.shoulderPoser.avatarTrackingReferences.leftHand.transform; + } + if (rightArm == null) + { + rightArm = Instantiate(rightArmDummy, leftShoulderAnchor.position, rightShoulderAnchor.rotation, rightShoulderAnchor); + var armIk = rightArm.GetComponentInChildren(); + armIk.shoulder = this; + armIk.shoulderPoser = GetComponent(); + armIk.target = armIk.shoulderPoser.avatarTrackingReferences.rightHand.transform; + } + } + + Start() + { + setShoulderWidth(PoseManager.Instance.playerWidthShoulders); + } + + setShoulderWidth(float width) + { + Vector3 localScale = new Vector3(width * .5, .05, .05); + Vector3 localPosition = new Vector3(width * .25, 0, 0); + + leftShoulderRenderer.localScale = localScale; + leftShoulderRenderer.localPosition = -localPosition; + + rightShoulderRenderer.localScale = localScale; + rightShoulderRenderer.localPosition = localPosition; + } + } + export default ShoulderTransforms; \ No newline at end of file diff --git a/vrarmik/StaticOffsetTransform.js b/vrarmik/StaticOffsetTransform.js index 8d816a9..c8bc1dd 100644 --- a/vrarmik/StaticOffsetTransform.js +++ b/vrarmik/StaticOffsetTransform.js @@ -1,4 +1,4 @@ -using UnityEngine; +using UnityEngine; namespace VRArmIK { diff --git a/vrarmik/Utils/Extensions.js b/vrarmik/Utils/Extensions.js index c2df576..08b84b1 100644 --- a/vrarmik/Utils/Extensions.js +++ b/vrarmik/Utils/Extensions.js @@ -1,4 +1,4 @@ -using System; +using System; using System.Collections; using System.Collections.Generic; using System.Linq; @@ -7,23 +7,23 @@ using UnityEngine.UI; using Random = UnityEngine.Random; -namespace VRArmIK +namespace VRArmIK { public static class LinqExtensions - { - public static void ForEach(this IEnumerable source, Action action) - { - foreach (var item in source) - action(item); + { + public static void ForEach(this IEnumerable source, Action action) + { + foreach (var item in source) + action(item); } } public static class UtilsExtensions - { - public static string F(this string self, params object[] objects) - { - return string.Format(self, objects); + { + public static string F(this string self, params object[] objects) + { + return string.Format(self, objects); } } @@ -155,22 +155,22 @@ namespace VRArmIK public static class ListExtensions { - public static T random(this List self) => self[(int)(self.Count * Random.value)]; - public static T next(this List self, int currentIndex, bool loop = false) - { - currentIndex++; - if (!loop && currentIndex >= self.Count) - { - throw new IndexOutOfRangeException(); - } - - currentIndex %= self.Count; - return self[currentIndex]; - } - - public static T next(this List self, T current, bool loop = false) - { - return self.next(self.IndexOf(current), loop); + public static T random(this List self) => self[(int)(self.Count * Random.value)]; + public static T next(this List self, int currentIndex, bool loop = false) + { + currentIndex++; + if (!loop && currentIndex >= self.Count) + { + throw new IndexOutOfRangeException(); + } + + currentIndex %= self.Count; + return self[currentIndex]; + } + + public static T next(this List self, T current, bool loop = false) + { + return self.next(self.IndexOf(current), loop); } } @@ -268,18 +268,18 @@ namespace VRArmIK self.alpha = 0f; } - } - - public static class IENumerableExtensions - { - public static T random(this IEnumerable self) - { - if (self.Count() == 0) - { - throw new IndexOutOfRangeException(); - } - return self.ElementAt(Random.Range(0, self.Count())); - } + } + + public static class IENumerableExtensions + { + public static T random(this IEnumerable self) + { + if (self.Count() == 0) + { + throw new IndexOutOfRangeException(); + } + return self.ElementAt(Random.Range(0, self.Count())); + } } public static class FloatExtensions diff --git a/vrarmik/Utils/InspectorAttributes.js b/vrarmik/Utils/InspectorAttributes.js index d319b07..235f6f2 100644 --- a/vrarmik/Utils/InspectorAttributes.js +++ b/vrarmik/Utils/InspectorAttributes.js @@ -1,14 +1,14 @@ -using UnityEngine; - -public class DisplayOnlyAttribute : PropertyAttribute -{ -} -public class LabelOverride : PropertyAttribute -{ - public string label; - public LabelOverride(string label) - { - this.label = label; - } - +using UnityEngine; + +public class DisplayOnlyAttribute : PropertyAttribute +{ +} +public class LabelOverride : PropertyAttribute +{ + public string label; + public LabelOverride(string label) + { + this.label = label; + } + } \ No newline at end of file diff --git a/vrarmik/Utils/VectorHelpers.js b/vrarmik/Utils/VectorHelpers.js index fb69917..aed02c1 100644 --- a/vrarmik/Utils/VectorHelpers.js +++ b/vrarmik/Utils/VectorHelpers.js @@ -1,6 +1,6 @@ -using UnityEngine; +using UnityEngine; -public static class VectorHelpers +public static class VectorHelpers { public static float axisAngle(Vector3 v, Vector3 forward, Vector3 axis) { diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index dafb873..14ac05f 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -1,424 +1,424 @@ -import ArmTransforms from './ArmTransforms.js'; -import ShoulderTransforms from './ShoulderTransforms.js'; -import ShoulderPoser from './ShoulderPoser.js'; - -class ArmIKElbowSettings -{ - constructor() { - this.calcElbowAngle = true; - this.clampElbowAngle = true; - this.softClampElbowAngle = true; - this.maxAngle = 175; - this.minAngle = 13; - this.softClampRange = 10; - this.offsetAngle = 135; - this.yWeight = -60; - this.zWeightTop = 260; - this.zWeightBottom = -100; - this.zBorderY = -.25; - this.zDistanceStart = .6; - this.xWeight = -50; - this.xDistanceStart = .1; - } -} - -class BeforePositioningSettings -{ - constructor() { - this.correctElbowOutside = true; - this.weight = -0.5; - this.startBelowZ = .4; - this.startAboveY = 0.1; - } -} - -class ElbowCorrectionSettings -{ - constructor() { - this.useFixedElbowWhenNearShoulder = true; - this.startBelowDistance = .5; - this.startBelowY = 0.1; - this.weight = 2; - this.localElbowPos = new Vector3(0.3, -1, -2); - } -} - -class HandSettings -{ - constructor() { - this.useWristRotation = true; - this.rotateElbowWithHandRight = true; - this.rotateElbowWithHandForward = true; - this.handDeltaPow = 1.5; - this.handDeltaFactor = -.3; - this.handDeltaOffset = 45; - // todo fix rotateElbowWithHandForward with factor != 1 -> horrible jumps. good value would be between [0.4, 0.6] - this.handDeltaForwardPow = 2; - this.handDeltaForwardFactor = 1; - this.handDeltaForwardOffset = 0; - this.handDeltaForwardDeadzone = .3; - this.rotateElbowWithHandDelay = .08; - } -} - - class VRArmIK - { - constructor() { - this.arm = new ArmTransforms(); // XXX these need to be this'd below - this.shoulder = ShoulderTransforms(); - this.shoulderPoser = new ShoulderPoser(); - this.target = new Transform(); - this.left = true; - - this.elbowSettings = new ArmIKElbowSettings(); - this.beforePositioningSettings = new BeforePositioningSettings(); - this.elbowCorrectionSettings = ElbowCorrectionSettings(); - this.handSettings = new HandSettings(); - - this.nextLowerArmAngle = new Vector3(); - - this.upperArmStartRotation = new Quaternion(); - this.lowerArmStartRotation = new Quaternion(); - this.wristStartRotation = new Quaternion(); - this.handStartRotation = new Quaternion(); - - this.interpolatedDeltaElbow = 0; - this.interpolatedDeltaElbowForward = 0; - } - - Awake() - { - upperArmStartRotation = arm.upperArm.rotation; - lowerArmStartRotation = arm.lowerArm.rotation; - wristStartRotation = Quaternion.identity; - if (arm.wrist1 != null) - wristStartRotation = arm.wrist1.rotation; - handStartRotation = arm.hand.rotation; - } - - OnEnable() - { - setUpperArmRotation(Quaternion.identity); - setLowerArmRotation(Quaternion.identity); - setHandRotation(Quaternion.identity); - } - - LateUpdate() - { - updateUpperArmPosition(); - calcElbowInnerAngle(); - rotateShoulder(); - correctElbowRotation(); - if (elbowSettings.calcElbowAngle) - { - positionElbow(); - if (elbowCorrectionSettings.useFixedElbowWhenNearShoulder) - correctElbowAfterPositioning(); - if (handSettings.rotateElbowWithHandRight) - rotateElbowWithHandRight(); - if (handSettings.rotateElbowWithHandForward) - rotateElbowWithHandFoward(); - rotateHand(); - } - } - - updateArmAndTurnElbowUp() - { - updateUpperArmPosition(); - calcElbowInnerAngle(); - rotateShoulder(); - correctElbowRotation(); - } - - updateUpperArmPosition() - { - //arm.upperArm.position = shoulderAnker.transform.position; - } - - calcElbowInnerAngle() - { - Vector3 eulerAngles = new Vector3(); - float targetShoulderDistance = (target.position - upperArmPos).magnitude; - float innerAngle; - - if (targetShoulderDistance > arm.armLength) - { - innerAngle = 0; - } - else - { - innerAngle = Mathf.Acos(Mathf.Clamp((Mathf.Pow(arm.upperArmLength, 2) + Mathf.Pow(arm.lowerArmLength, 2) - - Mathf.Pow(targetShoulderDistance, 2)) / (2 * arm.upperArmLength * arm.lowerArmLength), -1, 1)) * Mathf.Rad2Deg; - if (left) - innerAngle = 180 - innerAngle; - else - innerAngle = 180 + innerAngle; - if (float.IsNaN(innerAngle)) - { - innerAngle = 180; - } - } - - eulerAngles.y = innerAngle; - nextLowerArmAngle = eulerAngles; - } - - //source: https://github.com/NickHardeman/ofxIKArm/blob/master/src/ofxIKArm.cpp - rotateShoulder() - { - Vector3 eulerAngles = new Vector3(); - Vector3 targetShoulderDirection = (target.position - upperArmPos).normalized; - float targetShoulderDistance = (target.position - upperArmPos).magnitude; - - eulerAngles.y = (left ? -1 : 1) * - Mathf.Acos(Mathf.Clamp((Mathf.Pow(targetShoulderDistance, 2) + Mathf.Pow(arm.upperArmLength, 2) - - Mathf.Pow(arm.lowerArmLength, 2)) / (2 * targetShoulderDistance * arm.upperArmLength), -1, 1)) * Mathf.Rad2Deg; - if (float.IsNaN(eulerAngles.y)) - eulerAngles.y = 0; - - - Quaternion shoulderRightRotation = Quaternion.FromToRotation(armDirection, targetShoulderDirection); - setUpperArmRotation(shoulderRightRotation); - arm.upperArm.rotation = Quaternion.AngleAxis(eulerAngles.y, lowerArmRotation * Vector3.up) * arm.upperArm.rotation; - setLowerArmLocalRotation(Quaternion.Euler(nextLowerArmAngle)); - } - - getElbowTargetAngle() - { - Vector3 localHandPosNormalized = shoulderAnker.InverseTransformPoint(handPos) / arm.armLength; - - // angle from Y - var angle = elbowSettings.yWeight * localHandPosNormalized.y + elbowSettings.offsetAngle; - - // angle from Z - /*angle += Mathf.Lerp(elbowSettings.zWeightBottom, elbowSettings.zWeightTop, Mathf.Clamp01((localHandPosNormalized.y + 1f) - elbowSettings.zBorderY)) * - (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0f));*/ - if (localHandPosNormalized.y > 0) - angle += elbowSettings.zWeightTop * (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0) * Mathf.Max(localHandPosNormalized.y, 0)); - else - angle += elbowSettings.zWeightBottom * (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0) * Mathf.Max(-localHandPosNormalized.y, 0)); - - - // angle from X - angle += elbowSettings.xWeight * Mathf.Max(localHandPosNormalized.x * (left ? 1.0 : -1.0) + elbowSettings.xDistanceStart, 0); - - if (elbowSettings.clampElbowAngle) - { - if (elbowSettings.softClampElbowAngle) - { - if (angle < elbowSettings.minAngle + elbowSettings.softClampRange) - { - float a = elbowSettings.minAngle + elbowSettings.softClampRange - angle; - angle = elbowSettings.minAngle + elbowSettings.softClampRange * (1 - Mathf.Log(1 + a) * 3); - } - } - else - { - angle = Mathf.Clamp(angle, elbowSettings.minAngle, elbowSettings.maxAngle); - } - } - - if (left) - angle *= -1; - - return angle; - } - - correctElbowRotation() - { - var s = beforePositioningSettings; - - Vector3 localTargetPos = shoulderAnker.InverseTransformPoint(target.position) / arm.armLength; - float elbowOutsideFactor = Mathf.Clamp01( - Mathf.Clamp01((s.startBelowZ - localTargetPos.z) / - Mathf.Abs(s.startBelowZ) * .5) * - Mathf.Clamp01((localTargetPos.y - s.startAboveY) / - Mathf.Abs(s.startAboveY)) * - Mathf.Clamp01(1 - localTargetPos.x * (left ? -1 : 1)) - ) * s.weight; - - Vector3 shoulderHandDirection = (upperArmPos - handPos).normalized; - Vector3 targetDir = shoulder.transform.rotation * (Vector3.up + (s.correctElbowOutside ? (armDirection + Vector3.forward * -.2) * elbowOutsideFactor : Vector3.zero)); - Vector3 cross = Vector3.Cross(shoulderHandDirection, targetDir * 1000); - - Vector3 upperArmUp = upperArmRotation * Vector3.up; - - float elbowTargetUp = Vector3.Dot(upperArmUp, targetDir); - float elbowAngle = Vector3.Angle(cross, upperArmUp) + (left ? 0 : 180); - Quaternion rotation = Quaternion.AngleAxis(elbowAngle * Mathf.Sign(elbowTargetUp), shoulderHandDirection); - arm.upperArm.rotation = rotation * arm.upperArm.rotation; - } - - /// - /// reduces calculation problems when hand is moving around shoulder XZ coordinates -> forces elbow to be outside of body - /// - correctElbowAfterPositioning() - { - var s = elbowCorrectionSettings; - Vector3 localTargetPos = shoulderAnker.InverseTransformPoint(target.position) / arm.armLength; - Vector3 shoulderHandDirection = (upperArmPos - handPos).normalized; - Vector3 elbowPos = s.localElbowPos; - - if (left) - elbowPos.x *= -1; - - Vector3 targetDir = shoulder.transform.rotation * elbowPos.normalized; - Vector3 cross = Vector3.Cross(shoulderHandDirection, targetDir); - - Vector3 upperArmUp = upperArmRotation * Vector3.up; - - - Vector3 distance = target.position - upperArmPos; - distance = distance.magnitude * shoulder.transform.InverseTransformDirection(distance / distance.magnitude); - - float weight = Mathf.Clamp01(Mathf.Clamp01((s.startBelowDistance - distance.xz().magnitude / arm.armLength) / - s.startBelowDistance) * s.weight + Mathf.Clamp01((-distance.z + .1) * 3)) * - Mathf.Clamp01((s.startBelowY - localTargetPos.y) / - s.startBelowY); - - float elbowTargetUp = Vector3.Dot(upperArmUp, targetDir); - float elbowAngle2 = Vector3.Angle(cross, upperArmUp) + (left ? 0 : 180); - Quaternion rotation = Quaternion.AngleAxis((elbowAngle2 * Mathf.Sign(elbowTargetUp)).toSignedEulerAngle() * Mathf.Clamp(weight, 0, 1), shoulderHandDirection); - arm.upperArm.rotation = rotation * arm.upperArm.rotation; - } - - rotateElbow(float angle) - { - Vector3 shoulderHandDirection = (upperArmPos - handPos).normalized; - - Quaternion rotation = Quaternion.AngleAxis(angle, shoulderHandDirection); - setUpperArmRotation(rotation * upperArmRotation); - } - - //source: https://github.com/NickHardeman/ofxIKArm/blob/master/src/ofxIKArm.cpp - positionElbow() - { - float targetElbowAngle = getElbowTargetAngle(); - rotateElbow(targetElbowAngle); - } - - - rotateElbowWithHandRight() - { - var s = handSettings; - Vector3 handUpVec = target.rotation * Vector3.up; - float forwardAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.right, target.rotation * Vector3.right, - lowerArmRotation * Vector3.up, lowerArmRotation * Vector3.forward); - - // todo reduce influence if hand local forward rotation is high (hand tilted inside) - Quaternion handForwardRotation = Quaternion.AngleAxis(-forwardAngle, lowerArmRotation * Vector3.forward); - handUpVec = handForwardRotation * handUpVec; - - float elbowTargetAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.up, handUpVec, - lowerArmRotation * Vector3.forward, lowerArmRotation * armDirection); - - float deltaElbow = (elbowTargetAngle + (left ? -s.handDeltaOffset : s.handDeltaOffset)) / 180; - - deltaElbow = Mathf.Sign(deltaElbow) * Mathf.Pow(Mathf.Abs(deltaElbow), s.handDeltaPow) * 180 * s.handDeltaFactor; - interpolatedDeltaElbow = - Mathf.LerpAngle(interpolatedDeltaElbow, deltaElbow, Time.deltaTime / s.rotateElbowWithHandDelay); - rotateElbow(interpolatedDeltaElbow); - } - - rotateElbowWithHandFoward() - { - var s = handSettings; - Vector3 handRightVec = target.rotation * armDirection; - - float elbowTargetAngleForward = VectorHelpers.getAngleBetween(lowerArmRotation * armDirection, handRightVec, - lowerArmRotation * Vector3.up, lowerArmRotation * Vector3.forward); - - float deltaElbowForward = (elbowTargetAngleForward + (left ? -s.handDeltaForwardOffset : s.handDeltaForwardOffset)) / 180; - - if (Mathf.Abs(deltaElbowForward) < s.handDeltaForwardDeadzone) - deltaElbowForward = 0; - else - { - deltaElbowForward = (deltaElbowForward - Mathf.Sign(deltaElbowForward) * s.handDeltaForwardDeadzone) / (1 - s.handDeltaForwardDeadzone); - } - - deltaElbowForward = Mathf.Sign(deltaElbowForward) * Mathf.Pow(Mathf.Abs(deltaElbowForward), s.handDeltaForwardPow) * 180; - interpolatedDeltaElbowForward = Mathf.LerpAngle(interpolatedDeltaElbowForward, deltaElbowForward, Time.deltaTime / s.rotateElbowWithHandDelay); - - float signedInterpolated = interpolatedDeltaElbowForward.toSignedEulerAngle(); - rotateElbow(signedInterpolated * s.handDeltaForwardFactor); - } - - rotateHand() - { - if (handSettings.useWristRotation) - { - Vector3 handUpVec = target.rotation * Vector3.up; - float forwardAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.right, target.rotation * Vector3.right, - lowerArmRotation * Vector3.up, lowerArmRotation * Vector3.forward); - - // todo reduce influence if hand local forward rotation is high (hand tilted inside) - Quaternion handForwardRotation = Quaternion.AngleAxis(-forwardAngle, lowerArmRotation * Vector3.forward); - handUpVec = handForwardRotation * handUpVec; - - float elbowTargetAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.up, handUpVec, - lowerArmRotation * Vector3.forward, lowerArmRotation * armDirection); - - elbowTargetAngle = Mathf.Clamp(elbowTargetAngle, -90, 90); - if (arm.wrist1 != null) - setWrist1Rotation(Quaternion.AngleAxis(elbowTargetAngle * .3, lowerArmRotation * armDirection) * lowerArmRotation); - if (arm.wrist2 != null) - setWrist2Rotation(Quaternion.AngleAxis(elbowTargetAngle * .8, lowerArmRotation * armDirection) * lowerArmRotation); - } - setHandRotation(target.rotation); - } - - removeShoulderRightRotation(Vector3 direction) { - return Quaternion.AngleAxis(-shoulderPoser.shoulderRightRotation, shoulder.transform.right) * direction; - } - - get armDirection() { - return left ? Vector3.left : Vector3.right; - } - get upperArmPos() { - return arm.upperArm.position; - } - get lowerArmPos() { - return arm.lowerArm.position; - } - get handPos() { - return arm.hand.position; - } - get shoulderAnker() { - return left ? shoulder.leftShoulderAnchor : shoulder.rightShoulderAnchor; - } - - get upperArmRotation() { - return arm.upperArm.rotation * Quaternion.Inverse(upperArmStartRotation); - } - get lowerArmRotation() { - return arm.lowerArm.rotation * Quaternion.Inverse(lowerArmStartRotation); - } - get handRotation() { - return arm.hand.rotation * Quaternion.Inverse(handStartRotation); - } - - setUpperArmRotation(Quaternion rotation) { - return arm.upperArm.rotation = rotation * upperArmStartRotation; - } - setLowerArmRotation(Quaternion rotation) { - return arm.lowerArm.rotation = rotation * lowerArmStartRotation; - } - setLowerArmLocalRotation(Quaternion rotation) { - return arm.lowerArm.rotation = upperArmRotation * rotation * lowerArmStartRotation; - } - setWrist1Rotation(Quaternion rotation) { - return arm.wrist1.rotation = rotation * wristStartRotation; - } - setWrist2Rotation(Quaternion rotation) { - return arm.wrist2.rotation = rotation * wristStartRotation; - } - setWristLocalRotation(Quaternion rotation) { - return arm.wrist1.rotation = arm.lowerArm.rotation * rotation * wristStartRotation; - } - setHandRotation(Quaternion rotation) { - return arm.hand.rotation = arm.hand.rotation = rotation * handStartRotation; - } - } - +import ArmTransforms from './ArmTransforms.js'; +import ShoulderTransforms from './ShoulderTransforms.js'; +import ShoulderPoser from './ShoulderPoser.js'; + +class ArmIKElbowSettings +{ + constructor() { + this.calcElbowAngle = true; + this.clampElbowAngle = true; + this.softClampElbowAngle = true; + this.maxAngle = 175; + this.minAngle = 13; + this.softClampRange = 10; + this.offsetAngle = 135; + this.yWeight = -60; + this.zWeightTop = 260; + this.zWeightBottom = -100; + this.zBorderY = -.25; + this.zDistanceStart = .6; + this.xWeight = -50; + this.xDistanceStart = .1; + } +} + +class BeforePositioningSettings +{ + constructor() { + this.correctElbowOutside = true; + this.weight = -0.5; + this.startBelowZ = .4; + this.startAboveY = 0.1; + } +} + +class ElbowCorrectionSettings +{ + constructor() { + this.useFixedElbowWhenNearShoulder = true; + this.startBelowDistance = .5; + this.startBelowY = 0.1; + this.weight = 2; + this.localElbowPos = new Vector3(0.3, -1, -2); + } +} + +class HandSettings +{ + constructor() { + this.useWristRotation = true; + this.rotateElbowWithHandRight = true; + this.rotateElbowWithHandForward = true; + this.handDeltaPow = 1.5; + this.handDeltaFactor = -.3; + this.handDeltaOffset = 45; + // todo fix rotateElbowWithHandForward with factor != 1 -> horrible jumps. good value would be between [0.4, 0.6] + this.handDeltaForwardPow = 2; + this.handDeltaForwardFactor = 1; + this.handDeltaForwardOffset = 0; + this.handDeltaForwardDeadzone = .3; + this.rotateElbowWithHandDelay = .08; + } +} + + class VRArmIK + { + constructor() { + this.arm = new ArmTransforms(); // XXX these need to be this'd below + this.shoulder = ShoulderTransforms(); + this.shoulderPoser = new ShoulderPoser(); + this.target = new Transform(); + this.left = true; + + this.elbowSettings = new ArmIKElbowSettings(); + this.beforePositioningSettings = new BeforePositioningSettings(); + this.elbowCorrectionSettings = ElbowCorrectionSettings(); + this.handSettings = new HandSettings(); + + this.nextLowerArmAngle = new Vector3(); + + this.upperArmStartRotation = new Quaternion(); + this.lowerArmStartRotation = new Quaternion(); + this.wristStartRotation = new Quaternion(); + this.handStartRotation = new Quaternion(); + + this.interpolatedDeltaElbow = 0; + this.interpolatedDeltaElbowForward = 0; + } + + Awake() + { + upperArmStartRotation = arm.upperArm.rotation; + lowerArmStartRotation = arm.lowerArm.rotation; + wristStartRotation = Quaternion.identity; + if (arm.wrist1 != null) + wristStartRotation = arm.wrist1.rotation; + handStartRotation = arm.hand.rotation; + } + + OnEnable() + { + setUpperArmRotation(Quaternion.identity); + setLowerArmRotation(Quaternion.identity); + setHandRotation(Quaternion.identity); + } + + LateUpdate() + { + updateUpperArmPosition(); + calcElbowInnerAngle(); + rotateShoulder(); + correctElbowRotation(); + if (elbowSettings.calcElbowAngle) + { + positionElbow(); + if (elbowCorrectionSettings.useFixedElbowWhenNearShoulder) + correctElbowAfterPositioning(); + if (handSettings.rotateElbowWithHandRight) + rotateElbowWithHandRight(); + if (handSettings.rotateElbowWithHandForward) + rotateElbowWithHandFoward(); + rotateHand(); + } + } + + updateArmAndTurnElbowUp() + { + updateUpperArmPosition(); + calcElbowInnerAngle(); + rotateShoulder(); + correctElbowRotation(); + } + + updateUpperArmPosition() + { + //arm.upperArm.position = shoulderAnker.transform.position; + } + + calcElbowInnerAngle() + { + Vector3 eulerAngles = new Vector3(); + float targetShoulderDistance = (target.position - upperArmPos).magnitude; + float innerAngle; + + if (targetShoulderDistance > arm.armLength) + { + innerAngle = 0; + } + else + { + innerAngle = Mathf.Acos(Mathf.Clamp((Mathf.Pow(arm.upperArmLength, 2) + Mathf.Pow(arm.lowerArmLength, 2) - + Mathf.Pow(targetShoulderDistance, 2)) / (2 * arm.upperArmLength * arm.lowerArmLength), -1, 1)) * Mathf.Rad2Deg; + if (left) + innerAngle = 180 - innerAngle; + else + innerAngle = 180 + innerAngle; + if (float.IsNaN(innerAngle)) + { + innerAngle = 180; + } + } + + eulerAngles.y = innerAngle; + nextLowerArmAngle = eulerAngles; + } + + //source: https://github.com/NickHardeman/ofxIKArm/blob/master/src/ofxIKArm.cpp + rotateShoulder() + { + Vector3 eulerAngles = new Vector3(); + Vector3 targetShoulderDirection = (target.position - upperArmPos).normalized; + float targetShoulderDistance = (target.position - upperArmPos).magnitude; + + eulerAngles.y = (left ? -1 : 1) * + Mathf.Acos(Mathf.Clamp((Mathf.Pow(targetShoulderDistance, 2) + Mathf.Pow(arm.upperArmLength, 2) - + Mathf.Pow(arm.lowerArmLength, 2)) / (2 * targetShoulderDistance * arm.upperArmLength), -1, 1)) * Mathf.Rad2Deg; + if (float.IsNaN(eulerAngles.y)) + eulerAngles.y = 0; + + + Quaternion shoulderRightRotation = Quaternion.FromToRotation(armDirection, targetShoulderDirection); + setUpperArmRotation(shoulderRightRotation); + arm.upperArm.rotation = Quaternion.AngleAxis(eulerAngles.y, lowerArmRotation * Vector3.up) * arm.upperArm.rotation; + setLowerArmLocalRotation(Quaternion.Euler(nextLowerArmAngle)); + } + + getElbowTargetAngle() + { + Vector3 localHandPosNormalized = shoulderAnker.InverseTransformPoint(handPos) / arm.armLength; + + // angle from Y + var angle = elbowSettings.yWeight * localHandPosNormalized.y + elbowSettings.offsetAngle; + + // angle from Z + /*angle += Mathf.Lerp(elbowSettings.zWeightBottom, elbowSettings.zWeightTop, Mathf.Clamp01((localHandPosNormalized.y + 1f) - elbowSettings.zBorderY)) * + (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0f));*/ + if (localHandPosNormalized.y > 0) + angle += elbowSettings.zWeightTop * (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0) * Mathf.Max(localHandPosNormalized.y, 0)); + else + angle += elbowSettings.zWeightBottom * (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0) * Mathf.Max(-localHandPosNormalized.y, 0)); + + + // angle from X + angle += elbowSettings.xWeight * Mathf.Max(localHandPosNormalized.x * (left ? 1.0 : -1.0) + elbowSettings.xDistanceStart, 0); + + if (elbowSettings.clampElbowAngle) + { + if (elbowSettings.softClampElbowAngle) + { + if (angle < elbowSettings.minAngle + elbowSettings.softClampRange) + { + float a = elbowSettings.minAngle + elbowSettings.softClampRange - angle; + angle = elbowSettings.minAngle + elbowSettings.softClampRange * (1 - Mathf.Log(1 + a) * 3); + } + } + else + { + angle = Mathf.Clamp(angle, elbowSettings.minAngle, elbowSettings.maxAngle); + } + } + + if (left) + angle *= -1; + + return angle; + } + + correctElbowRotation() + { + var s = beforePositioningSettings; + + Vector3 localTargetPos = shoulderAnker.InverseTransformPoint(target.position) / arm.armLength; + float elbowOutsideFactor = Mathf.Clamp01( + Mathf.Clamp01((s.startBelowZ - localTargetPos.z) / + Mathf.Abs(s.startBelowZ) * .5) * + Mathf.Clamp01((localTargetPos.y - s.startAboveY) / + Mathf.Abs(s.startAboveY)) * + Mathf.Clamp01(1 - localTargetPos.x * (left ? -1 : 1)) + ) * s.weight; + + Vector3 shoulderHandDirection = (upperArmPos - handPos).normalized; + Vector3 targetDir = shoulder.transform.rotation * (Vector3.up + (s.correctElbowOutside ? (armDirection + Vector3.forward * -.2) * elbowOutsideFactor : Vector3.zero)); + Vector3 cross = Vector3.Cross(shoulderHandDirection, targetDir * 1000); + + Vector3 upperArmUp = upperArmRotation * Vector3.up; + + float elbowTargetUp = Vector3.Dot(upperArmUp, targetDir); + float elbowAngle = Vector3.Angle(cross, upperArmUp) + (left ? 0 : 180); + Quaternion rotation = Quaternion.AngleAxis(elbowAngle * Mathf.Sign(elbowTargetUp), shoulderHandDirection); + arm.upperArm.rotation = rotation * arm.upperArm.rotation; + } + + /// + /// reduces calculation problems when hand is moving around shoulder XZ coordinates -> forces elbow to be outside of body + /// + correctElbowAfterPositioning() + { + var s = elbowCorrectionSettings; + Vector3 localTargetPos = shoulderAnker.InverseTransformPoint(target.position) / arm.armLength; + Vector3 shoulderHandDirection = (upperArmPos - handPos).normalized; + Vector3 elbowPos = s.localElbowPos; + + if (left) + elbowPos.x *= -1; + + Vector3 targetDir = shoulder.transform.rotation * elbowPos.normalized; + Vector3 cross = Vector3.Cross(shoulderHandDirection, targetDir); + + Vector3 upperArmUp = upperArmRotation * Vector3.up; + + + Vector3 distance = target.position - upperArmPos; + distance = distance.magnitude * shoulder.transform.InverseTransformDirection(distance / distance.magnitude); + + float weight = Mathf.Clamp01(Mathf.Clamp01((s.startBelowDistance - distance.xz().magnitude / arm.armLength) / + s.startBelowDistance) * s.weight + Mathf.Clamp01((-distance.z + .1) * 3)) * + Mathf.Clamp01((s.startBelowY - localTargetPos.y) / + s.startBelowY); + + float elbowTargetUp = Vector3.Dot(upperArmUp, targetDir); + float elbowAngle2 = Vector3.Angle(cross, upperArmUp) + (left ? 0 : 180); + Quaternion rotation = Quaternion.AngleAxis((elbowAngle2 * Mathf.Sign(elbowTargetUp)).toSignedEulerAngle() * Mathf.Clamp(weight, 0, 1), shoulderHandDirection); + arm.upperArm.rotation = rotation * arm.upperArm.rotation; + } + + rotateElbow(float angle) + { + Vector3 shoulderHandDirection = (upperArmPos - handPos).normalized; + + Quaternion rotation = Quaternion.AngleAxis(angle, shoulderHandDirection); + setUpperArmRotation(rotation * upperArmRotation); + } + + //source: https://github.com/NickHardeman/ofxIKArm/blob/master/src/ofxIKArm.cpp + positionElbow() + { + float targetElbowAngle = getElbowTargetAngle(); + rotateElbow(targetElbowAngle); + } + + + rotateElbowWithHandRight() + { + var s = handSettings; + Vector3 handUpVec = target.rotation * Vector3.up; + float forwardAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.right, target.rotation * Vector3.right, + lowerArmRotation * Vector3.up, lowerArmRotation * Vector3.forward); + + // todo reduce influence if hand local forward rotation is high (hand tilted inside) + Quaternion handForwardRotation = Quaternion.AngleAxis(-forwardAngle, lowerArmRotation * Vector3.forward); + handUpVec = handForwardRotation * handUpVec; + + float elbowTargetAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.up, handUpVec, + lowerArmRotation * Vector3.forward, lowerArmRotation * armDirection); + + float deltaElbow = (elbowTargetAngle + (left ? -s.handDeltaOffset : s.handDeltaOffset)) / 180; + + deltaElbow = Mathf.Sign(deltaElbow) * Mathf.Pow(Mathf.Abs(deltaElbow), s.handDeltaPow) * 180 * s.handDeltaFactor; + interpolatedDeltaElbow = + Mathf.LerpAngle(interpolatedDeltaElbow, deltaElbow, Time.deltaTime / s.rotateElbowWithHandDelay); + rotateElbow(interpolatedDeltaElbow); + } + + rotateElbowWithHandFoward() + { + var s = handSettings; + Vector3 handRightVec = target.rotation * armDirection; + + float elbowTargetAngleForward = VectorHelpers.getAngleBetween(lowerArmRotation * armDirection, handRightVec, + lowerArmRotation * Vector3.up, lowerArmRotation * Vector3.forward); + + float deltaElbowForward = (elbowTargetAngleForward + (left ? -s.handDeltaForwardOffset : s.handDeltaForwardOffset)) / 180; + + if (Mathf.Abs(deltaElbowForward) < s.handDeltaForwardDeadzone) + deltaElbowForward = 0; + else + { + deltaElbowForward = (deltaElbowForward - Mathf.Sign(deltaElbowForward) * s.handDeltaForwardDeadzone) / (1 - s.handDeltaForwardDeadzone); + } + + deltaElbowForward = Mathf.Sign(deltaElbowForward) * Mathf.Pow(Mathf.Abs(deltaElbowForward), s.handDeltaForwardPow) * 180; + interpolatedDeltaElbowForward = Mathf.LerpAngle(interpolatedDeltaElbowForward, deltaElbowForward, Time.deltaTime / s.rotateElbowWithHandDelay); + + float signedInterpolated = interpolatedDeltaElbowForward.toSignedEulerAngle(); + rotateElbow(signedInterpolated * s.handDeltaForwardFactor); + } + + rotateHand() + { + if (handSettings.useWristRotation) + { + Vector3 handUpVec = target.rotation * Vector3.up; + float forwardAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.right, target.rotation * Vector3.right, + lowerArmRotation * Vector3.up, lowerArmRotation * Vector3.forward); + + // todo reduce influence if hand local forward rotation is high (hand tilted inside) + Quaternion handForwardRotation = Quaternion.AngleAxis(-forwardAngle, lowerArmRotation * Vector3.forward); + handUpVec = handForwardRotation * handUpVec; + + float elbowTargetAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.up, handUpVec, + lowerArmRotation * Vector3.forward, lowerArmRotation * armDirection); + + elbowTargetAngle = Mathf.Clamp(elbowTargetAngle, -90, 90); + if (arm.wrist1 != null) + setWrist1Rotation(Quaternion.AngleAxis(elbowTargetAngle * .3, lowerArmRotation * armDirection) * lowerArmRotation); + if (arm.wrist2 != null) + setWrist2Rotation(Quaternion.AngleAxis(elbowTargetAngle * .8, lowerArmRotation * armDirection) * lowerArmRotation); + } + setHandRotation(target.rotation); + } + + removeShoulderRightRotation(Vector3 direction) { + return Quaternion.AngleAxis(-shoulderPoser.shoulderRightRotation, shoulder.transform.right) * direction; + } + + get armDirection() { + return left ? Vector3.left : Vector3.right; + } + get upperArmPos() { + return arm.upperArm.position; + } + get lowerArmPos() { + return arm.lowerArm.position; + } + get handPos() { + return arm.hand.position; + } + get shoulderAnker() { + return left ? shoulder.leftShoulderAnchor : shoulder.rightShoulderAnchor; + } + + get upperArmRotation() { + return arm.upperArm.rotation * Quaternion.Inverse(upperArmStartRotation); + } + get lowerArmRotation() { + return arm.lowerArm.rotation * Quaternion.Inverse(lowerArmStartRotation); + } + get handRotation() { + return arm.hand.rotation * Quaternion.Inverse(handStartRotation); + } + + setUpperArmRotation(Quaternion rotation) { + return arm.upperArm.rotation = rotation * upperArmStartRotation; + } + setLowerArmRotation(Quaternion rotation) { + return arm.lowerArm.rotation = rotation * lowerArmStartRotation; + } + setLowerArmLocalRotation(Quaternion rotation) { + return arm.lowerArm.rotation = upperArmRotation * rotation * lowerArmStartRotation; + } + setWrist1Rotation(Quaternion rotation) { + return arm.wrist1.rotation = rotation * wristStartRotation; + } + setWrist2Rotation(Quaternion rotation) { + return arm.wrist2.rotation = rotation * wristStartRotation; + } + setWristLocalRotation(Quaternion rotation) { + return arm.wrist1.rotation = arm.lowerArm.rotation * rotation * wristStartRotation; + } + setHandRotation(Quaternion rotation) { + return arm.hand.rotation = arm.hand.rotation = rotation * handStartRotation; + } + } + export default VRArmIK; \ No newline at end of file diff --git a/vrarmik/VRTrackingReferences.js b/vrarmik/VRTrackingReferences.js index c703d31..dcadcf3 100644 --- a/vrarmik/VRTrackingReferences.js +++ b/vrarmik/VRTrackingReferences.js @@ -1,11 +1,11 @@ -using UnityEngine; - -namespace VRArmIK -{ - - public class VRTrackingReferences : MonoBehaviour - { - public Transform leftController, rightController, hmd; - public Transform leftHand, rightHand, head; - } +using UnityEngine; + +namespace VRArmIK +{ + + public class VRTrackingReferences : MonoBehaviour + { + public Transform leftController, rightController, hmd; + public Transform leftHand, rightHand, head; + } } \ No newline at end of file From 250e8fbb6f7e49485cfc7817f5e6fa23301a0afd Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 13:00:44 -0400 Subject: [PATCH 026/562] Remove dead LocalVrTrackingInput.js --- vrarmik/LocalVrTrackingInput.js | 21 --------------------- 1 file changed, 21 deletions(-) delete mode 100644 vrarmik/LocalVrTrackingInput.js diff --git a/vrarmik/LocalVrTrackingInput.js b/vrarmik/LocalVrTrackingInput.js deleted file mode 100644 index c663c91..0000000 --- a/vrarmik/LocalVrTrackingInput.js +++ /dev/null @@ -1,21 +0,0 @@ -using System.Collections; -using System.Collections.Generic; -using UnityEngine; -using UnityEngine.VR; - -namespace VRArmIK -{ - public class LocalVrTrackingInput : MonoBehaviour - { - public UnityEngine.XR.XRNode node; - - void Update() - { - if (UnityEngine.XR.InputTracking.GetLocalPosition(node).magnitude < 0.000001f) - return; - - transform.localPosition = UnityEngine.XR.InputTracking.GetLocalPosition(node); - transform.localRotation = UnityEngine.XR.InputTracking.GetLocalRotation(node); - } - } -} \ No newline at end of file From e708b95fd84af243cc2aa47878188f6aa95db107 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 13:10:02 -0400 Subject: [PATCH 027/562] More porting work --- vrarmik/AvatarVRTrackingReferences.js | 30 ++++++------ vrarmik/PoseManager.js | 62 +++++++++++------------- vrarmik/ShoulderPoser.js | 1 + vrarmik/StaticOffsetTransform.js | 68 +++++++++++++-------------- vrarmik/VRTrackingReferences.js | 18 +++---- 5 files changed, 88 insertions(+), 91 deletions(-) diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index 35c0ba2..b167b5e 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -1,24 +1,25 @@ -using UnityEngine; -namespace VRArmIK -{ - public class AvatarVRTrackingReferences : MonoBehaviour + class AvatarVRTrackingReferences { - public StaticOffsetTransform head, hmd, leftHand, rightHand; + constructor() { + this.head = new StaticOffsetTransform(); + this.hmd = new StaticOffsetTransform(); + this.leftHand = new StaticOffsetTransform(); + this.rightHand = new StaticOffsetTransform(); + } - void Start() + Start() { initTransforms(); } - [ContextMenu("init transforms")] - public void initTransforms() + initTransforms() { createTransforms(); connectTransforms(); } - void setStaticOffsetSettings(StaticOffsetTransform s) + setStaticOffsetSettings(StaticOffsetTransform s) { s.referenceLocalPosition = false; s.referenceLocalRotation = false; @@ -30,7 +31,7 @@ namespace VRArmIK } - void createTransform(ref StaticOffsetTransform t, string name) + createTransform(ref StaticOffsetTransform t, string name) { if (t == null) { @@ -40,7 +41,7 @@ namespace VRArmIK } } - void createHandTransform(ref Transform t, string name, Transform parent) + createHandTransform(ref Transform t, string name, Transform parent) { if (t == null) { @@ -50,7 +51,7 @@ namespace VRArmIK } } - void createTransforms() + createTransforms() { createTransform(ref head, nameof(head)); createTransform(ref leftHand, nameof(leftHand)); @@ -58,7 +59,7 @@ namespace VRArmIK createTransform(ref hmd, nameof(hmd)); } - void connectTransforms() + connectTransforms() { StaticOffsetTransform sot = this.GetOrAddComponent(); if (sot.reference == null) @@ -76,4 +77,5 @@ namespace VRArmIK : PoseManager.Instance.vrTransforms.rightHand; } } -} \ No newline at end of file + +export default AvatarVRTrackingReferences; \ No newline at end of file diff --git a/vrarmik/PoseManager.js b/vrarmik/PoseManager.js index 754a768..c8c4f64 100644 --- a/vrarmik/PoseManager.js +++ b/vrarmik/PoseManager.js @@ -1,30 +1,24 @@ -using UnityEngine; -using UnityEngine.XR; +import VRTrackingReferences from './VRTrackingReferences.js'; -namespace VRArmIK -{ - [ExecuteInEditMode] - public class PoseManager : MonoBehaviour +class PoseManager { - public static PoseManager Instance = null; - public VRTrackingReferences vrTransforms; + constructor() { + this.vrTransforms = new VRTrackingReferences(); + this.OnCalibrateListener = null; - public delegate void OnCalibrateListener(); + // Oculus uses a different reference position -> 0 is the reference head position if the user is standing in the middle of the room. + // In OpenVR, the 0 position is the ground position and the user is then at (0, playerHeightHmd, 0) if he is in the middle of the room, so I need to correct this for shoulder calculation + this.vrSystemOffsetHeight = 0.0; - public event OnCalibrateListener onCalibrate; + this.referencePlayerHeightHmd = 1.7; + this.referencePlayerWidthWrist = 1.39; + this.playerHeightHmd = 1.70; + this.playerWidthWrist = 1.39; + this.playerWidthShoulders = 0.31; + this.loadPlayerSizeOnAwake = false; + } - // Oculus uses a different reference position -> 0 is the reference head position if the user is standing in the middle of the room. - // In OpenVR, the 0 position is the ground position and the user is then at (0, playerHeightHmd, 0) if he is in the middle of the room, so I need to correct this for shoulder calculation - public float vrSystemOffsetHeight = 0.0f; - - public const float referencePlayerHeightHmd = 1.7f; - public const float referencePlayerWidthWrist = 1.39f; - public float playerHeightHmd = 1.70f; - public float playerWidthWrist = 1.39f; - public float playerWidthShoulders = 0.31f; - public bool loadPlayerSizeOnAwake = false; - - void OnEnable() + OnEnable() { if (Instance == null) { @@ -36,7 +30,7 @@ namespace VRArmIK } } - void Awake() + Awake() { if (loadPlayerSizeOnAwake) { @@ -46,36 +40,34 @@ namespace VRArmIK vrSystemOffsetHeight = string.IsNullOrEmpty(device) || device == "OpenVR" ? 0 : playerHeightHmd; } - void Start() + Start() { onCalibrate += OnCalibrate; } - [ContextMenu("calibrate")] - void OnCalibrate() + OnCalibrate() { playerHeightHmd = Camera.main.transform.position.y; } - void loadPlayerWidthShoulders() + loadPlayerWidthShoulders() { - playerWidthShoulders = PlayerPrefs.GetFloat("VRArmIK_PlayerWidthShoulders", 0.31f); + playerWidthShoulders = PlayerPrefs.GetFloat("VRArmIK_PlayerWidthShoulders", 0.31); } - public void savePlayerWidthShoulders(float width) + savePlayerWidthShoulders(float width) { PlayerPrefs.SetFloat("VRArmIK_PlayerWidthShoulders", width); } - [ContextMenu("setArmLength")] - public void calibrateIK() + calibrateIK() { playerWidthWrist = (vrTransforms.leftHand.position - vrTransforms.rightHand.position).magnitude; playerHeightHmd = vrTransforms.hmd.position.y; savePlayerSize(playerHeightHmd, playerWidthWrist); } - public void savePlayerSize(float heightHmd, float widthWrist) + savePlayerSize(float heightHmd, float widthWrist) { PlayerPrefs.SetFloat("VRArmIK_PlayerHeightHmd", heightHmd); PlayerPrefs.SetFloat("VRArmIK_PlayerWidthWrist", widthWrist); @@ -83,10 +75,12 @@ namespace VRArmIK onCalibrate?.Invoke(); } - public void loadPlayerSize() + loadPlayerSize() { playerHeightHmd = PlayerPrefs.GetFloat("VRArmIK_PlayerHeightHmd", referencePlayerHeightHmd); playerWidthWrist = PlayerPrefs.GetFloat("VRArmIK_PlayerWidthWrist", referencePlayerWidthWrist); } } -} \ No newline at end of file + PoseManager.Instance = null; + +export default PoseManager; \ No newline at end of file diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index 532b71a..e9f7079 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -1,4 +1,5 @@ import ShoulderTransforms from './ShoulderTransforms.js'; +import VRTrackingReferences from './VRTrackingReferences.js'; class ShoulderPoser { diff --git a/vrarmik/StaticOffsetTransform.js b/vrarmik/StaticOffsetTransform.js index c8bc1dd..b6a1b7c 100644 --- a/vrarmik/StaticOffsetTransform.js +++ b/vrarmik/StaticOffsetTransform.js @@ -1,34 +1,33 @@ -using UnityEngine; - -namespace VRArmIK -{ - public class StaticOffsetTransform : MonoBehaviour +const EulerOrder = { + XYZ: 'XYZ', + XZY: 'XZY', + YXZ: 'YXZ', + YZX: 'YZX', + ZXY: 'ZXY', + ZYX: 'ZYX', +}; + +class StaticOffsetTransform { - public enum EulerOrder - { - XYZ, - XZY, - YXZ, - YZX, - ZXY, - ZYX - }; - - public Transform reference = null; - public Vector3 offsetPosition; - public Vector3 offsetRotation; - public Vector3 orientationalOffset; - public Vector3 referenceRotationMultiplicator = Vector3.one; - - public EulerOrder axisOrder; - - public bool referenceLocalPosition = false, referenceLocalRotation = false; - public bool applyLocalPosition = false, applyLocalRotation = false; - public bool applyPosition = true, applyRotation = true; - public bool applyForwardOffsetAfterRotationOffset = false; - - - public static Vector3 switchAxis(Vector3 r, EulerOrder order) + constructor() { + this.reference = null; + this.offsetPosition = new Vector3(); + this.offsetRotation = new Vector3(); + this.orientationalOffset = new Vector3(); + this.referenceRotationMultiplicator = Vector3.one; + + this.axisOrder = new EulerOrder(); + + this.referenceLocalPosition = false; + this.referenceLocalRotation = false; + this.applyLocalPosition = false; + this.applyLocalRotation = false; + this.applyPosition = true; + this.applyRotation = true; + this.applyForwardOffsetAfterRotationOffset = false; + } + + switchAxis(Vector3 r, EulerOrder order) { switch (order) { @@ -50,17 +49,17 @@ namespace VRArmIK } } - void Awake() + Awake() { updatePosition(); } - void Update() + Update() { updatePosition(); } - void updatePosition() + updatePosition() { if (reference == null) return; @@ -113,4 +112,5 @@ namespace VRArmIK } } } -} \ No newline at end of file + +export default StaticOffsetTransform; \ No newline at end of file diff --git a/vrarmik/VRTrackingReferences.js b/vrarmik/VRTrackingReferences.js index dcadcf3..861c2e2 100644 --- a/vrarmik/VRTrackingReferences.js +++ b/vrarmik/VRTrackingReferences.js @@ -1,11 +1,11 @@ -using UnityEngine; - -namespace VRArmIK -{ - - public class VRTrackingReferences : MonoBehaviour + class VRTrackingReferences { - public Transform leftController, rightController, hmd; - public Transform leftHand, rightHand, head; + this.leftController = new Transform(); + this.rightController = new Transform(); + this.hmd = new Transform(); + this.leftHand = new Transform(); + this.rightHand = new Transform(); + this.head = new Transform(); } -} \ No newline at end of file + +export default VRTrackingReferences; \ No newline at end of file From cc507abfc9cee94c0fbcfda8b0c6067eea5edd63 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 13:21:18 -0400 Subject: [PATCH 028/562] Remove dead utils files --- vrarmik/Utils/Editor.meta | 10 - .../Editor/LabelOverrideInspector.cs.meta | 13 - .../Utils/Editor/LabelOverrideInspector.js | 13 - .../Utils/Editor/ReadOnlyInspector.cs.meta | 13 - vrarmik/Utils/Editor/ReadOnlyInspector.js | 39 --- vrarmik/Utils/Extensions.cs.meta | 13 - vrarmik/Utils/Extensions.js | 301 ------------------ vrarmik/Utils/InspectorAttributes.cs.meta | 13 - vrarmik/Utils/InspectorAttributes.js | 14 - vrarmik/Utils/VectorHelpers.cs.meta | 13 - 10 files changed, 442 deletions(-) delete mode 100644 vrarmik/Utils/Editor.meta delete mode 100644 vrarmik/Utils/Editor/LabelOverrideInspector.cs.meta delete mode 100644 vrarmik/Utils/Editor/LabelOverrideInspector.js delete mode 100644 vrarmik/Utils/Editor/ReadOnlyInspector.cs.meta delete mode 100644 vrarmik/Utils/Editor/ReadOnlyInspector.js delete mode 100644 vrarmik/Utils/Extensions.cs.meta delete mode 100644 vrarmik/Utils/Extensions.js delete mode 100644 vrarmik/Utils/InspectorAttributes.cs.meta delete mode 100644 vrarmik/Utils/InspectorAttributes.js delete mode 100644 vrarmik/Utils/VectorHelpers.cs.meta diff --git a/vrarmik/Utils/Editor.meta b/vrarmik/Utils/Editor.meta deleted file mode 100644 index 99d9d7a..0000000 --- a/vrarmik/Utils/Editor.meta +++ /dev/null @@ -1,10 +0,0 @@ -fileFormatVersion: 2 -guid: ff05c1415d8afee45b3616972c5e311b -folderAsset: yes -timeCreated: 1528126517 -licenseType: Free -DefaultImporter: - externalObjects: {} - userData: - assetBundleName: - assetBundleVariant: diff --git a/vrarmik/Utils/Editor/LabelOverrideInspector.cs.meta b/vrarmik/Utils/Editor/LabelOverrideInspector.cs.meta deleted file mode 100644 index 564a570..0000000 --- a/vrarmik/Utils/Editor/LabelOverrideInspector.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: 3a301e6db22cc5747b54c20deee95547 -timeCreated: 1528126534 -licenseType: Free -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/vrarmik/Utils/Editor/LabelOverrideInspector.js b/vrarmik/Utils/Editor/LabelOverrideInspector.js deleted file mode 100644 index a808af9..0000000 --- a/vrarmik/Utils/Editor/LabelOverrideInspector.js +++ /dev/null @@ -1,13 +0,0 @@ -using UnityEngine; -using UnityEditor; - -[CustomPropertyDrawer(typeof(LabelOverride))] -public class ThisPropertyDrawer : PropertyDrawer -{ - public override void OnGUI(Rect position, SerializedProperty property, GUIContent label) - { - var propertyAttribute = attribute as LabelOverride; - label.text = propertyAttribute?.label; - EditorGUI.PropertyField(position, property, label); - } -} \ No newline at end of file diff --git a/vrarmik/Utils/Editor/ReadOnlyInspector.cs.meta b/vrarmik/Utils/Editor/ReadOnlyInspector.cs.meta deleted file mode 100644 index 5944573..0000000 --- a/vrarmik/Utils/Editor/ReadOnlyInspector.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: 995d6e718b3990448939a03e8c25bbf2 -timeCreated: 1528126534 -licenseType: Free -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/vrarmik/Utils/Editor/ReadOnlyInspector.js b/vrarmik/Utils/Editor/ReadOnlyInspector.js deleted file mode 100644 index 3443cbb..0000000 --- a/vrarmik/Utils/Editor/ReadOnlyInspector.js +++ /dev/null @@ -1,39 +0,0 @@ -using System.ComponentModel; -using UnityEditor; -using UnityEngine; - -[CustomPropertyDrawer(typeof(DisplayOnlyAttribute))] -public class DisplayOnlyInspector : PropertyDrawer -{ - public override void OnGUI(Rect position, SerializedProperty prop, GUIContent label) - { - string valueStr; - - switch (prop.propertyType) - { - case SerializedPropertyType.Integer: - valueStr = prop.intValue.ToString(); - break; - case SerializedPropertyType.Boolean: - valueStr = prop.boolValue.ToString(); - break; - case SerializedPropertyType.Float: - valueStr = prop.floatValue.ToString("0.00000"); - break; - case SerializedPropertyType.String: - valueStr = prop.stringValue; - break; - case SerializedPropertyType.Vector2: - valueStr = prop.vector2Value.ToString(); - break; - case SerializedPropertyType.Vector3: - valueStr = prop.vector3Value.ToString(); - break; - default: - valueStr = "(not supported)"; - break; - } - - EditorGUI.LabelField(position, label.text, valueStr); - } -} \ No newline at end of file diff --git a/vrarmik/Utils/Extensions.cs.meta b/vrarmik/Utils/Extensions.cs.meta deleted file mode 100644 index 8db612c..0000000 --- a/vrarmik/Utils/Extensions.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: dd40c275b2b3c5042bee83817b9d5277 -timeCreated: 1528126366 -licenseType: Free -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/vrarmik/Utils/Extensions.js b/vrarmik/Utils/Extensions.js deleted file mode 100644 index 08b84b1..0000000 --- a/vrarmik/Utils/Extensions.js +++ /dev/null @@ -1,301 +0,0 @@ -using System; -using System.Collections; -using System.Collections.Generic; -using System.Linq; -using UnityEngine; -using UnityEngine.UI; -using Random = UnityEngine.Random; - - -namespace VRArmIK -{ - - public static class LinqExtensions - { - public static void ForEach(this IEnumerable source, Action action) - { - foreach (var item in source) - action(item); - } - } - - public static class UtilsExtensions - { - public static string F(this string self, params object[] objects) - { - return string.Format(self, objects); - } - } - - public static class TransformExtensions - { - public static void SetParentForReal(this Transform self, Transform parent) - { - self.SetParent(parent); - self.localScale = Vector3.one; - } - } - - public static class MonoBehaviourExtensions - { - public static T GetOrAddComponent(this Component self) where T : Component - { - T component = self.GetComponent(); - return component != null ? component : self.gameObject.AddComponent(); - } - - public static T GetOrAddComponentInChildren(this MonoBehaviour self) where T : MonoBehaviour - { - T component = self.GetComponentInChildren(); - return component != null ? component : self.gameObject.AddComponent(); - } - } - - - public static class LayerMaskExtensions - { - public static LayerMask Create(params string[] layerNames) - { - return NamesToMask(layerNames); - } - - public static LayerMask Create(params int[] layerNumbers) - { - return LayerNumbersToMask(layerNumbers); - } - - public static LayerMask NamesToMask(params string[] layerNames) - { - LayerMask ret = (LayerMask)0; - foreach (var name in layerNames) - { - ret |= (1 << LayerMask.NameToLayer(name)); - } - - return ret; - } - - public static LayerMask LayerNumbersToMask(params int[] layerNumbers) - { - LayerMask ret = (LayerMask)0; - foreach (var layer in layerNumbers) - { - ret |= (1 << layer); - } - return ret; - } - - public static LayerMask Inverse(this LayerMask original) - { - return ~original; - } - - public static LayerMask AddToMask(this LayerMask original, params string[] layerNames) - { - return original | NamesToMask(layerNames); - } - - public static LayerMask RemoveFromMask(this LayerMask original, params string[] layerNames) - { - return original & ~NamesToMask(layerNames); - } - - public static string[] MaskToNames(this LayerMask original) - { - var output = new List(); - - for (int i = 0; i < 32; ++i) - { - int shifted = 1 << i; - if ((original & shifted) == shifted) - { - string layerName = LayerMask.LayerToName(i); - if (!string.IsNullOrEmpty(layerName)) - { - output.Add(layerName); - } - } - } - return output.ToArray(); - } - - public static string MaskToString(this LayerMask original) - { - return MaskToString(original, ", "); - } - - public static string MaskToString(this LayerMask original, string delimiter) - { - return string.Join(delimiter, MaskToNames(original)); - } - } - - public static class VectorExtensions - { - public static Vector3 toVector3(this Vector2 self) - { - return new Vector3(self.x, self.y); - } - - public static Vector2 xy(this Vector3 self) - { - return new Vector2(self.x, self.y); - } - - public static Vector2 xz(this Vector3 self) - { - return new Vector2(self.x, self.z); - } - - public static Vector2 yz(this Vector3 self) - { - return new Vector2(self.y, self.y); - } - } - - public static class ListExtensions - { - public static T random(this List self) => self[(int)(self.Count * Random.value)]; - public static T next(this List self, int currentIndex, bool loop = false) - { - currentIndex++; - if (!loop && currentIndex >= self.Count) - { - throw new IndexOutOfRangeException(); - } - - currentIndex %= self.Count; - return self[currentIndex]; - } - - public static T next(this List self, T current, bool loop = false) - { - return self.next(self.IndexOf(current), loop); - } - } - - - public static class RandomExtensions - { - public static Vector2 insideUnitCircle(this System.Random self) - { - float radius = (float)self.NextDouble(); - float angle = (float)self.NextDouble() * Mathf.PI * 2; - - return new Vector2(Mathf.Cos(angle) * radius, Mathf.Sin(angle) * radius); - } - - public static float range(this System.Random self, float min, float max) - { - return (float)(self.NextDouble() * (max - min) + min); - } - - public static Vector3 insideUnitCube(this System.Random self) - { - return new Vector3(self.range(-1f, 1f), self.range(-1f, 1f), self.range(-1f, 1f)); - } - } - - public static class EnumExtenstions - { - public class Enum where T : struct, IConvertible - { - public static int Count - { - get - { - if (!typeof(T).IsEnum) - throw new ArgumentException("T must be an enumerated type"); - - return Enum.GetNames(typeof(T)).Length; - } - } - } - - public static T Next(this T src) where T : struct - { - if (!typeof(T).IsEnum) throw new ArgumentException($"Argumnent {typeof(T).FullName} is not an Enum"); - - T[] Arr = (T[])Enum.GetValues(src.GetType()); - int j = Array.IndexOf(Arr, src) + 1; - return Arr.Length == j ? Arr[0] : Arr[j]; - } - - public static bool IsLast(this T src) where T : struct - { - if (!typeof(T).IsEnum) throw new ArgumentException($"Argumnent {typeof(T).FullName} is not an Enum"); - - T[] Arr = (T[])Enum.GetValues(src.GetType()); - int j = Array.IndexOf(Arr, src); - return Arr.Length == j + 1; - } - } - - public static class UIExtensions - { - public static void setNormalButtonColor(this Button self, Color color) - { - - var buttonColors = self.colors; - buttonColors.normalColor = color; - self.colors = buttonColors; - } - - public static IEnumerator fadeIn(this CanvasGroup self, float duration) - { - float timePassed = 0.0f; - - while (timePassed < duration) - { - self.alpha = timePassed / duration; - timePassed += Time.unscaledDeltaTime; - yield return null; - } - - self.alpha = 1f; - } - - public static IEnumerator fadeOut(this CanvasGroup self, float duration) - { - float timePassed = 0.0f; - - while (timePassed < duration) - { - self.alpha = 1f - timePassed / duration; - timePassed += Time.unscaledDeltaTime; - yield return null; - } - - self.alpha = 0f; - } - } - - public static class IENumerableExtensions - { - public static T random(this IEnumerable self) - { - if (self.Count() == 0) - { - throw new IndexOutOfRangeException(); - } - return self.ElementAt(Random.Range(0, self.Count())); - } - } - - public static class FloatExtensions - { - public static float toSignedEulerAngle(this float self) - { - float result = self.toPositiveEulerAngle(); - if (result > 180f) - result = result - 360f; - return result; - } - - public static float toPositiveEulerAngle(this float self) - { - float result = (self % 360f + 360f) % 360f; - return result; - } - } -} \ No newline at end of file diff --git a/vrarmik/Utils/InspectorAttributes.cs.meta b/vrarmik/Utils/InspectorAttributes.cs.meta deleted file mode 100644 index 3e7f3ba..0000000 --- a/vrarmik/Utils/InspectorAttributes.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: a729eda56804e2043ac3fc58960a13bb -timeCreated: 1528126528 -licenseType: Free -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: diff --git a/vrarmik/Utils/InspectorAttributes.js b/vrarmik/Utils/InspectorAttributes.js deleted file mode 100644 index 235f6f2..0000000 --- a/vrarmik/Utils/InspectorAttributes.js +++ /dev/null @@ -1,14 +0,0 @@ -using UnityEngine; - -public class DisplayOnlyAttribute : PropertyAttribute -{ -} -public class LabelOverride : PropertyAttribute -{ - public string label; - public LabelOverride(string label) - { - this.label = label; - } - -} \ No newline at end of file diff --git a/vrarmik/Utils/VectorHelpers.cs.meta b/vrarmik/Utils/VectorHelpers.cs.meta deleted file mode 100644 index 2c65cb4..0000000 --- a/vrarmik/Utils/VectorHelpers.cs.meta +++ /dev/null @@ -1,13 +0,0 @@ -fileFormatVersion: 2 -guid: 56770d9500874ac4d8980df3b398f7ee -timeCreated: 1528126637 -licenseType: Free -MonoImporter: - externalObjects: {} - serializedVersion: 2 - defaultReferences: [] - executionOrder: 0 - icon: {instanceID: 0} - userData: - assetBundleName: - assetBundleVariant: From 44c612b3d51f08fd0149c5d47268842b7959f41e Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 13:23:01 -0400 Subject: [PATCH 029/562] Port vrarmik VectorHelpers --- vrarmik/Utils/VectorHelpers.js | 18 ++++++++---------- vrarmik/VRArmIK.js | 1 + 2 files changed, 9 insertions(+), 10 deletions(-) diff --git a/vrarmik/Utils/VectorHelpers.js b/vrarmik/Utils/VectorHelpers.js index aed02c1..dfb63a9 100644 --- a/vrarmik/Utils/VectorHelpers.js +++ b/vrarmik/Utils/VectorHelpers.js @@ -1,19 +1,17 @@ -using UnityEngine; - -public static class VectorHelpers -{ - public static float axisAngle(Vector3 v, Vector3 forward, Vector3 axis) +const VectorHelpers = { + axisAngle(Vector3 v, Vector3 forward, Vector3 axis) { Vector3 right = Vector3.Cross(axis, forward); forward = Vector3.Cross(right, axis); return Mathf.Atan2(Vector3.Dot(v, right), Vector3.Dot(v, forward)) * Mathf.Rad2Deg; - } - - public static float getAngleBetween(Vector3 a, Vector3 b, Vector3 forward, Vector3 axis) + }, + getAngleBetween(Vector3 a, Vector3 b, Vector3 forward, Vector3 axis) { float angleA = axisAngle(a, forward, axis); float angleB = axisAngle(b, forward, axis); return Mathf.DeltaAngle(angleA, angleB); - } -} + }, +}; + +export default VectorHelpers; \ No newline at end of file diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index 14ac05f..1ded3e1 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -1,6 +1,7 @@ import ArmTransforms from './ArmTransforms.js'; import ShoulderTransforms from './ShoulderTransforms.js'; import ShoulderPoser from './ShoulderPoser.js'; +import VectorHelpers from 'Utils/VectorHelpers.js'; class ArmIKElbowSettings { From b7c4b17d2c7eb5a36c774cde241650c9885b3e41 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 14:47:07 -0400 Subject: [PATCH 030/562] Small vrarmik cleanup --- vrarmik/VRArmIK.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index 1ded3e1..dde97a0 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -66,7 +66,7 @@ class HandSettings { constructor() { this.arm = new ArmTransforms(); // XXX these need to be this'd below - this.shoulder = ShoulderTransforms(); + this.shoulder = new ShoulderTransforms(); this.shoulderPoser = new ShoulderPoser(); this.target = new Transform(); this.left = true; @@ -422,4 +422,4 @@ class HandSettings } } -export default VRArmIK; \ No newline at end of file +export default VRArmIK; From 07b8baf3300c09c788a71a2922709313150bea7d Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 20:45:47 -0400 Subject: [PATCH 031/562] Add initial Transform.js --- vrarmik/Transform.js | 9 +++++++++ 1 file changed, 9 insertions(+) create mode 100644 vrarmik/Transform.js diff --git a/vrarmik/Transform.js b/vrarmik/Transform.js new file mode 100644 index 0000000..5b74539 --- /dev/null +++ b/vrarmik/Transform.js @@ -0,0 +1,9 @@ +class Transform { + constructor() { + this.position = new Vector3(); + this.rotation = new Quaternion(); + this.scale = new Vector3(1, 1, 1); + } +} + +export default Transform; From 0e5a5eb1266a11dc358c71e06a25dfc8d5860df1 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 22:49:08 -0400 Subject: [PATCH 032/562] Remove dead AvatarPlatformOffset --- vrarmik/AvatarPlatformOffset.js | 19 ------------------- vrarmik/{Transform.js => Unity.js} | 0 2 files changed, 19 deletions(-) delete mode 100644 vrarmik/AvatarPlatformOffset.js rename vrarmik/{Transform.js => Unity.js} (100%) diff --git a/vrarmik/AvatarPlatformOffset.js b/vrarmik/AvatarPlatformOffset.js deleted file mode 100644 index dee5972..0000000 --- a/vrarmik/AvatarPlatformOffset.js +++ /dev/null @@ -1,19 +0,0 @@ -using System.Collections; -using System.Collections.Generic; -using UnityEngine; -using VRArmIK; - -public class AvatarPlatformOffset : MonoBehaviour -{ - public bool correctVrPlatformOffsetOnStart = true; - - void Start() - { - if (correctVrPlatformOffsetOnStart) - { - var p = transform.position; - p.y -= PoseManager.Instance.vrSystemOffsetHeight; - transform.position = p; - } - } -} diff --git a/vrarmik/Transform.js b/vrarmik/Unity.js similarity index 100% rename from vrarmik/Transform.js rename to vrarmik/Unity.js From 459d446b286cefc305f98ac08374e7fdf9b56c2b Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 22:51:11 -0400 Subject: [PATCH 033/562] Major vrarmik porting work --- vrarmik/ArmTransforms.js | 61 +++--- vrarmik/AvatarVRTrackingReferences.js | 43 ++-- vrarmik/PoseManager.js | 41 ++-- vrarmik/ShoulderPoser.js | 209 +++++++++---------- vrarmik/StaticOffsetTransform.js | 46 ++--- vrarmik/Unity.js | 24 ++- vrarmik/VRArmIK.js | 283 +++++++++++++------------- vrarmik/VRTrackingReferences.js | 20 +- 8 files changed, 381 insertions(+), 346 deletions(-) diff --git a/vrarmik/ArmTransforms.js b/vrarmik/ArmTransforms.js index 31ffada..5895098 100644 --- a/vrarmik/ArmTransforms.js +++ b/vrarmik/ArmTransforms.js @@ -1,3 +1,6 @@ +import Transform from './Transform.js'; +import PoseManager from './PoseManager.js'; + class ArmTransforms { constructor() { @@ -13,80 +16,80 @@ class ArmTransforms } get upperArmLength() { - return distance(upperArm, lowerArm); + return distance(this.upperArm, this.lowerArm); } get lowerArmLength() { - return distance(lowerArm, hand); + return distance(this.lowerArm, this.hand); } get armLength() { - return upperArmLength + lowerArmLength; + return this.upperArmLength + this.lowerArmLength; } - distance(Transform a, Transform b) { + distance(a, b) { return (a.position - b.position).magnitude; } Start() { - PoseManager.Instance.onCalibrate += updateArmLengths; - updateArmLengths(); + PoseManager.Instance.onCalibrate += this.updateArmLengths; + this.updateArmLengths(); } updateArmLengths() { - var shoulderWidth = (upperArm.position - lowerArm.position).magnitude; - var _armLength = (PoseManager.Instance.playerWidthWrist - shoulderWidth) / 2; - setArmLength(_armLength); + const shoulderWidth = (this.upperArm.position - this.lowerArm.position).magnitude; + const _armLength = (PoseManager.Instance.playerWidthWrist - shoulderWidth) / 2; + this.setArmLength(_armLength); } - setUpperArmLength(float length) + setUpperArmLength(length) { if (armLengthByScale) { - float oldLowerArmLength = distance(lowerArm, hand); + const oldLowerArmLength = distance(this.lowerArm, this.hand); - Vector3 newScale = upperArm.localScale - Vector3.Scale(upperArm.localScale, scaleAxis).magnitude * scaleAxis; - float scaleFactor = Vector3.Scale(upperArm.localScale, scaleAxis).magnitude / upperArmLength * length; - newScale += scaleAxis * scaleFactor; - upperArm.localScale = newScale; + let newScale = this.upperArm.localScale - Vector3.Scale(this.upperArm.localScale, this.scaleAxis).magnitude * this.scaleAxis; + const scaleFactor = Vector3.Scale(this.upperArm.localScale, this.scaleAxis).magnitude / upperArmLength * length; + newScale += this.scaleAxis * scaleFactor; + this.upperArm.localScale = newScale; - setLowerArmLength(oldLowerArmLength); + this.setLowerArmLength(oldLowerArmLength); } else { - Vector3 pos = lowerArm.localPosition; + const pos = this.lowerArm.localPosition; pos.x = Mathf.Sign(pos.x) * length; - lowerArm.localPosition = pos; + this.lowerArm.localPosition = pos; } } - setLowerArmLength(float length) + setLowerArmLength(length) { - if (armLengthByScale) + if (this.armLengthByScale) { } else { - Vector3 pos = hand.localPosition; + const pos = this.hand.localPosition; pos.x = Mathf.Sign(pos.x) * length; - hand.localPosition = pos; + this.hand.localPosition = pos; } } - setArmLength(float length) + setArmLength(length) { - float upperArmFactor = .48; + const upperArmFactor = .48; if (armLengthByScale) { - upperArm.localScale = upperArm.localScale / armLength * length; - hand.localScale = Vector3.one / (1 - (1 - scaleHandFactor) * (1 - upperArm.localScale.x)); + this.upperArm.localScale = this.upperArm.localScale / this.armLength * length; + this.hand.localScale = Vector3.one / (1 - (1 - this.scaleHandFactor) * (1 - this.upperArm.localScale.x)); } else { - setUpperArmLength(length * upperArmFactor); - setLowerArmLength(length * (1 - upperArmFactor)); + this.setUpperArmLength(length * upperArmFactor); + this.setLowerArmLength(length * (1 - upperArmFactor)); } } } -export default ArmTransforms; \ No newline at end of file +export default ArmTransforms; diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index b167b5e..1cdba5d 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -1,3 +1,4 @@ +import StaticOffsetTransform from './StaticOffsetTransform.js'; class AvatarVRTrackingReferences { @@ -10,16 +11,16 @@ Start() { - initTransforms(); + this.initTransforms(); } initTransforms() { - createTransforms(); - connectTransforms(); + this.createTransforms(); + this.connectTransforms(); } - setStaticOffsetSettings(StaticOffsetTransform s) + setStaticOffsetSettings(s) { s.referenceLocalPosition = false; s.referenceLocalRotation = false; @@ -31,19 +32,19 @@ } - createTransform(ref StaticOffsetTransform t, string name) + createTransform(t, name) { - if (t == null) + if (t === null) { t = new GameObject(name).AddComponent(); t.transform.parent = transform; - setStaticOffsetSettings(t); + this.setStaticOffsetSettings(t); } } - createHandTransform(ref Transform t, string name, Transform parent) + createHandTransform(t, name, parent) { - if (t == null) + if (t === null) { t = new GameObject(name).transform; t.transform.localPosition = Vector3.zero; @@ -53,29 +54,29 @@ createTransforms() { - createTransform(ref head, nameof(head)); - createTransform(ref leftHand, nameof(leftHand)); - createTransform(ref rightHand, nameof(rightHand)); - createTransform(ref hmd, nameof(hmd)); + this.createTransform(head, nameof(head)); + this.createTransform(leftHand, nameof(leftHand)); + this.createTransform(rightHand, nameof(rightHand)); + this.createTransform(hmd, nameof(hmd)); } connectTransforms() { - StaticOffsetTransform sot = this.GetOrAddComponent(); + /*StaticOffsetTransform */const sot = this.GetOrAddComponent/**/(); if (sot.reference == null) { sot.reference = transform.parent; } - head.reference = head.reference != null ? head.reference : PoseManager.Instance.vrTransforms.head; - hmd.reference = hmd.reference != null ? hmd.reference : PoseManager.Instance.vrTransforms.hmd; - leftHand.reference = leftHand.reference != null - ? leftHand.reference + this.head.reference = this.head.reference !== null ? this.head.reference : PoseManager.Instance.vrTransforms.head; + this.hmd.reference = this.hmd.reference !== null ? this.hmd.reference : PoseManager.Instance.vrTransforms.hmd; + this.leftHand.reference = leftHand.reference !== null + ? this.leftHand.reference : PoseManager.Instance.vrTransforms.leftHand; - rightHand.reference = rightHand.reference != null - ? rightHand.reference + this.rightHand.reference = this.rightHand.reference !== null + ? this.rightHand.reference : PoseManager.Instance.vrTransforms.rightHand; } } -export default AvatarVRTrackingReferences; \ No newline at end of file +export default AvatarVRTrackingReferences; diff --git a/vrarmik/PoseManager.js b/vrarmik/PoseManager.js index c8c4f64..8a91bfb 100644 --- a/vrarmik/PoseManager.js +++ b/vrarmik/PoseManager.js @@ -1,4 +1,5 @@ import VRTrackingReferences from './VRTrackingReferences.js'; +import {XRSettings} from './Unity.js'; class PoseManager { @@ -20,11 +21,11 @@ class PoseManager OnEnable() { - if (Instance == null) + if (PoseManager.Instance === null) { - Instance = this; + PoseManager.Instance = this; } - else if (Instance != null) + else if (PoseManager.Instance !== null) { Debug.LogError("Multiple Instances of PoseManager in Scene"); } @@ -32,12 +33,12 @@ class PoseManager Awake() { - if (loadPlayerSizeOnAwake) + if (this.loadPlayerSizeOnAwake) { - loadPlayerSize(); + this.loadPlayerSize(); } - var device = XRSettings.loadedDeviceName; - vrSystemOffsetHeight = string.IsNullOrEmpty(device) || device == "OpenVR" ? 0 : playerHeightHmd; + const device = XRSettings.loadedDeviceName; + this.vrSystemOffsetHeight = string.IsNullOrEmpty(device) || device == "OpenVR" ? 0 : this.playerHeightHmd; } Start() @@ -47,40 +48,40 @@ class PoseManager OnCalibrate() { - playerHeightHmd = Camera.main.transform.position.y; + this.playerHeightHmd = Camera.main.transform.position.y; } loadPlayerWidthShoulders() { - playerWidthShoulders = PlayerPrefs.GetFloat("VRArmIK_PlayerWidthShoulders", 0.31); + this.playerWidthShoulders = PlayerPrefs.GetFloat("VRArmIK_PlayerWidthShoulders", 0.31); } - savePlayerWidthShoulders(float width) + savePlayerWidthShoulders(width) { PlayerPrefs.SetFloat("VRArmIK_PlayerWidthShoulders", width); } calibrateIK() { - playerWidthWrist = (vrTransforms.leftHand.position - vrTransforms.rightHand.position).magnitude; - playerHeightHmd = vrTransforms.hmd.position.y; - savePlayerSize(playerHeightHmd, playerWidthWrist); + this.playerWidthWrist = (this.vrTransforms.leftHand.position - this.vrTransforms.rightHand.position).magnitude; + this.playerHeightHmd = this.vrTransforms.hmd.position.y; + this.savePlayerSize(this.playerHeightHmd, this.playerWidthWrist); } savePlayerSize(float heightHmd, float widthWrist) { - PlayerPrefs.SetFloat("VRArmIK_PlayerHeightHmd", heightHmd); - PlayerPrefs.SetFloat("VRArmIK_PlayerWidthWrist", widthWrist); - loadPlayerSize(); - onCalibrate?.Invoke(); + PlayerPrefs.SetFloat("VRArmIK_PlayerHeightHmd", this.heightHmd); + PlayerPrefs.SetFloat("VRArmIK_PlayerWidthWrist", this.widthWrist); + this.loadPlayerSize(); + this.onCalibrate && this.onCalibrate.Invoke(); } loadPlayerSize() { - playerHeightHmd = PlayerPrefs.GetFloat("VRArmIK_PlayerHeightHmd", referencePlayerHeightHmd); - playerWidthWrist = PlayerPrefs.GetFloat("VRArmIK_PlayerWidthWrist", referencePlayerWidthWrist); + this.playerHeightHmd = PlayerPrefs.GetFloat("VRArmIK_PlayerHeightHmd", this.referencePlayerHeightHmd); + this.playerWidthWrist = PlayerPrefs.GetFloat("VRArmIK_PlayerWidthWrist", this.referencePlayerWidthWrist); } } PoseManager.Instance = null; -export default PoseManager; \ No newline at end of file +export default PoseManager; diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index e9f7079..6098137 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -1,5 +1,6 @@ import ShoulderTransforms from './ShoulderTransforms.js'; import VRTrackingReferences from './VRTrackingReferences.js'; +import PoseManager from './PoseManager.js'; class ShoulderPoser { @@ -51,81 +52,81 @@ class ShoulderPoser Start() { - if (vrTrackingReferences == null) - vrTrackingReferences = PoseManager.Instance.vrTransforms; + if (this.vrTrackingReferences === null) + this.vrTrackingReferences = PoseManager.Instance.vrTransforms; - leftShoulderAnkerStartLocalPosition = shoulder.transform.InverseTransformPoint(shoulder.leftShoulderAnchor.position); - rightShoulderAnkerStartLocalPosition = - shoulder.transform.InverseTransformPoint(shoulder.rightShoulderAnchor.position); + this.leftShoulderAnkerStartLocalPosition = this.shoulder.transform.InverseTransformPoint(this.shoulder.leftShoulderAnchor.position); + this.rightShoulderAnkerStartLocalPosition = + this.shoulder.transform.InverseTransformPoint(this.shoulder.rightShoulderAnchor.position); } onCalibrate() { - shoulder.leftArm.setArmLength((avatarTrackingReferences.leftHand.transform.position - shoulder.leftShoulderAnchor.position) + this.shoulder.leftArm.setArmLength((avatarTrackingReferences.leftHand.transform.position - this.shoulder.leftShoulderAnchor.position) .magnitude); - shoulder.rightArm.setArmLength((avatarTrackingReferences.rightHand.transform.position - shoulder.rightShoulderAnchor.position) + this.shoulder.rightArm.setArmLength((avatarTrackingReferences.rightHand.transform.position - this.shoulder.rightShoulderAnchor.position) .magnitude); } Update() { - shoulder.transform.rotation = Quaternion.identity; - positionShoulder(); - rotateShoulderUp(); - rotateShoulderRight(); + this.shoulder.transform.rotation = Quaternion.identity; + this.positionShoulder(); + this.rotateShoulderUp(); + this.rotateShoulderRight(); - if (enableDistinctShoulderRotation) + if (this.enableDistinctShoulderRotation) { - rotateLeftShoulder(); - rotateRightShoulder(); + this.rotateLeftShoulder(); + this.rotateRightShoulder(); } - if (enableShoulderDislocation) + if (this.enableShoulderDislocation) { - clampShoulderHandDistance(); + this.clampShoulderHandDistance(); } else { - shoulder.leftArm.transform.localPosition = Vector3.zero; - shoulder.rightArm.transform.localPosition = Vector3.zero; + this.shoulder.leftArm.transform.localPosition = Vector3.zero; + this.shoulder.rightArm.transform.localPosition = Vector3.zero; } - Debug.DrawRay(shoulder.transform.position, shoulder.transform.forward); + Debug.DrawRay(this.shoulder.transform.position, this.shoulder.transform.forward); } rotateLeftShoulder() { - rotateShoulderUp(shoulder.leftShoulder, shoulder.leftArm, avatarTrackingReferences.leftHand.transform, - leftShoulderAnkerStartLocalPosition, 1); + this.rotateShoulderUp(shoulder.leftShoulder, this.shoulder.leftArm, this.avatarTrackingReferences.leftHand.transform, + this.leftShoulderAnkerStartLocalPosition, 1); } rotateRightShoulder() { - rotateShoulderUp(shoulder.rightShoulder, shoulder.rightArm, avatarTrackingReferences.rightHand.transform, - rightShoulderAnkerStartLocalPosition, -1); + this.rotateShoulderUp(shoulder.rightShoulder, this.shoulder.rightArm, this.avatarTrackingReferences.rightHand.transform, + this.rightShoulderAnkerStartLocalPosition, -1); } - rotateShoulderUp(Transform shoulderSide, ArmTransforms arm, Transform targetHand, - Vector3 initialShoulderLocalPos, float angleSign) + rotateShoulderUp(shoulderSide, arm, targetHand, + initialShoulderLocalPos, angleSign) { - Vector3 initialShoulderPos = shoulder.transform.TransformPoint(initialShoulderLocalPos); - Vector3 handShoulderOffset = targetHand.position - initialShoulderPos; - float armLength = arm.armLength; + const initialShoulderPos = this.shoulder.transform.TransformPoint(initialShoulderLocalPos); + const handShoulderOffset = targetHand.position - this.initialShoulderPos; + const armLength = arm.armLength; - Vector3 targetAngle = Vector3.zero; + const targetAngle = Vector3.zero; - float forwardDistanceRatio = Vector3.Dot(handShoulderOffset, shoulder.transform.forward) / armLength; - float upwardDistanceRatio = Vector3.Dot(handShoulderOffset, shoulder.transform.up) / armLength; + const forwardDistanceRatio = Vector3.Dot(handShoulderOffset, this.shoulder.transform.forward) / armLength; + const upwardDistanceRatio = Vector3.Dot(handShoulderOffset, this.shoulder.transform.up) / armLength; if (forwardDistanceRatio > 0) { - targetAngle.y = Mathf.Clamp((forwardDistanceRatio - 0.5) * distinctShoulderRotationMultiplier, 0, - distinctShoulderRotationLimitForward); + targetAngle.y = Mathf.Clamp((forwardDistanceRatio - 0.5) * this.distinctShoulderRotationMultiplier, 0, + this.distinctShoulderRotationLimitForward); } else { - targetAngle.y = Mathf.Clamp(-(forwardDistanceRatio + 0.08) * distinctShoulderRotationMultiplier * 10, - -distinctShoulderRotationLimitBackward, 0); + targetAngle.y = Mathf.Clamp(-(forwardDistanceRatio + 0.08) * this.distinctShoulderRotationMultiplier * 10, + -this.distinctShoulderRotationLimitBackward, 0); } targetAngle.z = Mathf.Clamp(-(upwardDistanceRatio - 0.5) * distinctShoulderRotationMultiplier, @@ -137,153 +138,155 @@ class ShoulderPoser positionShoulder() { - Vector3 headNeckOffset = avatarTrackingReferences.hmd.transform.rotation * headNeckDirectionVector; - Vector3 targetPosition = avatarTrackingReferences.head.transform.position + headNeckOffset * headNeckDistance; - shoulder.transform.localPosition = - shoulder.transform.parent.InverseTransformPoint(targetPosition) + neckShoulderDistance; + const headNeckOffset = this.avatarTrackingReferences.hmd.transform.rotation * this.headNeckDirectionVector; + const targetPosition = this.avatarTrackingReferences.head.transform.position + headNeckOffset * this.headNeckDistance; + this.shoulder.transform.localPosition = + this.shoulder.transform.parent.InverseTransformPoint(targetPosition) + this.neckShoulderDistance; } rotateShoulderUp() { - float angle = getCombinedDirectionAngleUp(); + const angle = getCombinedDirectionAngleUp(); - Vector3 targetRotation = new Vector3(0, angle, 0); + const targetRotation = new Vector3(0, angle, 0); - if (autoDetectHandsBehindHead) + if (this.autoDetectHandsBehindHead) { - detectHandsBehindHead(ref targetRotation); + this.detectHandsBehindHead(targetRotation); } - if (clampRotationToHead) + if (this.clampRotationToHead) { - clampHeadRotationDeltaUp(ref targetRotation); + this.clampHeadRotationDeltaUp(targetRotation); } - shoulder.transform.eulerAngles = targetRotation; + shouldthis.er.transform.eulerAngles = targetRotation; } rotateShoulderRight() { - float heightDiff = vrTrackingReferences.hmd.transform.position.y - PoseManager.Instance.vrSystemOffsetHeight; - float relativeHeightDiff = -heightDiff / PoseManager.Instance.playerHeightHmd; + const heightDiff = this.vrTrackingReferences.hmd.transform.position.y - PoseManager.Instance.vrSystemOffsetHeight; + const relativeHeightDiff = -heightDiff / PoseManager.Instance.playerHeightHmd; - float headRightRotation = VectorHelpers.getAngleBetween(shoulder.transform.forward, - avatarTrackingReferences.hmd.transform.forward, - Vector3.up, shoulder.transform.right) + rightRotationHeadRotationOffset; - float heightFactor = Mathf.Clamp(relativeHeightDiff - rightRotationStartHeight, 0, 1); - shoulderRightRotation = heightFactor * rightRotationHeightFactor; - shoulderRightRotation += Mathf.Clamp(headRightRotation * rightRotationHeadRotationFactor * heightFactor, 0, 50); + const headRightRotation = VectorHelpers.getAngleBetween(this.shoulder.transform.forward, + this.avatarTrackingReferences.hmd.transform.forward, + Vector3.up, this.shoulder.transform.right) + this.rightRotationHeadRotationOffset; + const heightFactor = Mathf.Clamp(relativeHeightDiff - this.rightRotationStartHeight, 0, 1); + this.shoulderRightRotation = heightFactor * this.rightRotationHeightFactor; + this.shoulderRightRotation += Mathf.Clamp(headRightRotation * this.rightRotationHeadRotationFactor * heightFactor, 0, 50); - shoulderRightRotation = Mathf.Clamp(shoulderRightRotation, 0, 50); + this.shoulderRightRotation = Mathf.Clamp(this.shoulderRightRotation, 0, 50); - Quaternion deltaRot = Quaternion.AngleAxis(shoulderRightRotation, shoulder.transform.right); + const deltaRot = Quaternion.AngleAxis(this.shoulderRightRotation, this.shoulder.transform.right); - shoulder.transform.rotation = deltaRot * shoulder.transform.rotation; - positionShoulderRelative(); + this.shoulder.transform.rotation = deltaRot * this.shoulder.transform.rotation; + this.positionShoulderRelative(); } positionShoulderRelative() { - Quaternion deltaRot = Quaternion.AngleAxis(shoulderRightRotation, shoulder.transform.right); - Vector3 shoulderHeadDiff = shoulder.transform.position - avatarTrackingReferences.head.transform.position; - shoulder.transform.position = deltaRot * shoulderHeadDiff + avatarTrackingReferences.head.transform.position; + const deltaRot = Quaternion.AngleAxis(this.shoulderRightRotation, this.shoulder.transform.right); + const shoulderHeadDiff = this.shoulder.transform.position - this.avatarTrackingReferences.head.transform.position; + this.shoulder.transform.position = deltaRot * shoulderHeadDiff + this.avatarTrackingReferences.head.transform.position; } getCombinedDirectionAngleUp() { - Transform leftHand = avatarTrackingReferences.leftHand.transform, rightHand = avatarTrackingReferences.rightHand.transform; + const leftHand = this.avatarTrackingReferences.leftHand.transform; + const rightHand = this.avatarTrackingReferences.rightHand.transform; - Vector3 distanceLeftHand = leftHand.position - shoulder.transform.position, - distanceRightHand = rightHand.position - shoulder.transform.position; + const distanceLeftHand = leftHand.position - this.shoulder.transform.position; + const distanceRightHand = rightHand.position - this.shoulder.transform.position; - if (ignoreYPos) + if (this.ignoreYPos) { - distanceLeftHand.y = 0; - distanceRightHand.y = 0; + this.distanceLeftHand.y = 0; + this.distanceRightHand.y = 0; } - Vector3 directionLeftHand = distanceLeftHand.normalized, - directionRightHand = distanceRightHand.normalized; + const directionLeftHand = distanceLeftHand.normalized; + const directionRightHand = distanceRightHand.normalized; - Vector3 combinedDirection = directionLeftHand + directionRightHand; + const combinedDirection = directionLeftHand + directionRightHand; return Mathf.Atan2(combinedDirection.x, combinedDirection.z) * 180 / Mathf.PI; } - detectHandsBehindHead(ref Vector3 targetRotation) + detectHandsBehindHead(targetRotation) { - float delta = Mathf.Abs(targetRotation.y - lastAngle.y + 360) % 360; - if (delta > 150 && delta < 210 && lastAngle.magnitude > 0.000001 && !clampingHeadRotation) + const delta = Mathf.Abs(targetRotation.y - this.lastAngle.y + 360) % 360; + if (delta > 150 && delta < 210 && this.lastAngle.magnitude > 0.000001 && !clampingHeadRotation) { - handsBehindHead = !handsBehindHead; + this.handsBehindHead = !this.handsBehindHead; } - lastAngle = targetRotation; + this.lastAngle = targetRotation; - if (handsBehindHead) + if (this.handsBehindHead) { targetRotation.y += 180; } } - clampHeadRotationDeltaUp(ref Vector3 targetRotation) + clampHeadRotationDeltaUp(targetRotation) { - float headUpRotation = (avatarTrackingReferences.head.transform.eulerAngles.y + 360) % 360; - float targetUpRotation = (targetRotation.y + 360) % 360; + const headUpRotation = (this.avatarTrackingReferences.head.transform.eulerAngles.y + 360) % 360; + const targetUpRotation = (targetRotation.y + 360) % 360; - float delta = headUpRotation - targetUpRotation; + const delta = headUpRotation - targetUpRotation; - if (delta > maxDeltaHeadRotation && delta < 180 || delta < -180 && delta >= -360 + maxDeltaHeadRotation) + if (delta > this.maxDeltaHeadRotation && delta < 180 || delta < -180 && delta >= -360 + this.maxDeltaHeadRotation) { - targetRotation.y = headUpRotation - maxDeltaHeadRotation; - clampingHeadRotation = true; + targetRotation.y = headUpRotation - this.maxDeltaHeadRotation; + this.clampingHeadRotation = true; } - else if (delta < -maxDeltaHeadRotation && delta > -180 || delta > 180 && delta < 360 - maxDeltaHeadRotation) + else if (delta < -this.maxDeltaHeadRotation && delta > -180 || delta > 180 && delta < 360 - this.maxDeltaHeadRotation) { - targetRotation.y = headUpRotation + maxDeltaHeadRotation; - clampingHeadRotation = true; + targetRotation.y = headUpRotation + this.maxDeltaHeadRotation; + this.clampingHeadRotation = true; } else { - clampingHeadRotation = false; + this.clampingHeadRotation = false; } } clampShoulderHandDistance() { - Vector3 leftHandVector = avatarTrackingReferences.leftHand.transform.position - shoulder.leftShoulderAnchor.position; - Vector3 rightHandVector = avatarTrackingReferences.rightHand.transform.position - shoulder.rightShoulderAnchor.position; - float leftShoulderHandDistance = leftHandVector.magnitude, rightShoulderHandDistance = rightHandVector.magnitude; - shoulderDislocated = false; + const leftHandVector = this.avatarTrackingReferences.leftHand.transform.position - this.shoulder.leftShoulderAnchor.position; + const rightHandVector = this.avatarTrackingReferences.rightHand.transform.position - this.shoulder.rightShoulderAnchor.position; + const leftShoulderHandDistance = leftHandVector.magnitude; + const rightShoulderHandDistance = rightHandVector.magnitude; + this.shoulderDislocated = false; - float startBeforeFactor = (1 - startShoulderDislocationBefore); + const startBeforeFactor = (1 - this.startShoulderDislocationBefore); - if (leftShoulderHandDistance > shoulder.leftArm.armLength * startBeforeFactor) + if (leftShoulderHandDistance > this.shoulder.leftArm.armLength * startBeforeFactor) { - shoulderDislocated = true; - shoulder.leftArm.transform.position = shoulder.leftShoulderAnchor.position + + this.shoulderDislocated = true; + this.shoulder.leftArm.transform.position = this.shoulder.leftShoulderAnchor.position + leftHandVector.normalized * - (leftShoulderHandDistance - shoulder.leftArm.armLength * startBeforeFactor); + (leftShoulderHandDistance - this.shoulder.leftArm.armLength * startBeforeFactor); } else { - shoulder.leftArm.transform.localPosition = Vector3.zero; + this.shoulder.leftArm.transform.localPosition = Vector3.zero; } - if (rightShoulderHandDistance > shoulder.rightArm.armLength * startBeforeFactor) + if (rightShoulderHandDistance > this.shoulder.rightArm.armLength * this.startBeforeFactor) { - shoulderDislocated = true; - shoulder.rightArm.transform.position = shoulder.rightShoulderAnchor.position + + this.shoulderDislocated = true; + this.shoulder.rightArm.transform.position = this.shoulder.rightShoulderAnchor.position + rightHandVector.normalized * (rightShoulderHandDistance - - shoulder.rightArm.armLength * startBeforeFactor); + this.shoulder.rightArm.armLength * startBeforeFactor); } else { - shoulder.rightArm.transform.localPosition = Vector3.zero; + this.shoulder.rightArm.transform.localPosition = Vector3.zero; } } } -export default ShoulderPoser; \ No newline at end of file +export default ShoulderPoser; diff --git a/vrarmik/StaticOffsetTransform.js b/vrarmik/StaticOffsetTransform.js index b6a1b7c..c715032 100644 --- a/vrarmik/StaticOffsetTransform.js +++ b/vrarmik/StaticOffsetTransform.js @@ -27,7 +27,7 @@ class StaticOffsetTransform this.applyForwardOffsetAfterRotationOffset = false; } - switchAxis(Vector3 r, EulerOrder order) + switchAxis(r, order) { switch (order) { @@ -51,66 +51,66 @@ class StaticOffsetTransform Awake() { - updatePosition(); + this.updatePosition(); } Update() { - updatePosition(); + this.updatePosition(); } updatePosition() { - if (reference == null) + if (this.reference === null) return; - Vector3 rot = switchAxis(referenceLocalRotation ? reference.localEulerAngles : reference.eulerAngles, axisOrder) + - offsetRotation; + const rot = this.switchAxis(this.referenceLocalRotation ? reference.localEulerAngles : reference.eulerAngles, this.axisOrder) + + this.offsetRotation; rot.Scale(referenceRotationMultiplicator); - Vector3 pos = referenceLocalPosition ? reference.localPosition : reference.position; + const pos = this.referenceLocalPosition ? this.reference.localPosition : this.reference.position; - if (applyForwardOffsetAfterRotationOffset) + if (this.applyForwardOffsetAfterRotationOffset) { - pos += Quaternion.Euler(rot) * Vector3.right * orientationalOffset.x; - pos += Quaternion.Euler(rot) * Vector3.up * orientationalOffset.y; - pos += Quaternion.Euler(rot) * Vector3.forward * orientationalOffset.z; + pos += Quaternion.Euler(rot) * Vector3.right * this.orientationalOffset.x; + pos += Quaternion.Euler(rot) * Vector3.up * othis.rientationalOffset.y; + pos += Quaternion.Euler(rot) * Vector3.forward * this.orientationalOffset.z; } else { - pos += reference.right * orientationalOffset.x; - pos += reference.up * orientationalOffset.y; - pos += reference.forward * orientationalOffset.z; + pos += this.reference.right * this.orientationalOffset.x; + pos += this.reference.up * this.orientationalOffset.y; + pos += this.reference.forward * this.orientationalOffset.z; } pos += offsetPosition; - if (applyPosition) + if (this.applyPosition) { - if (applyLocalPosition) + if (this.applyLocalPosition) { - transform.localPosition = pos; + this.transform.localPosition = pos; } else { - transform.position = pos; + this.transform.position = pos; } } - if (applyRotation) + if (this.applyRotation) { - if (applyLocalRotation) + if (this.applyLocalRotation) { - transform.localEulerAngles = rot; + this.transform.localEulerAngles = rot; } else { - transform.eulerAngles = rot; + this.transform.eulerAngles = rot; } } } } -export default StaticOffsetTransform; \ No newline at end of file +export default StaticOffsetTransform; diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index 5b74539..ec78eb1 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -6,4 +6,26 @@ class Transform { } } -export default Transform; +const PlayerPrefs = { + data: {}, + GetFloat(k, d) { + let v = this.data[k]; + if (v === undefined) { + v = d; + } + return v; + }, + SetFloat(k, v) { + this.data[k] = v; + }, +}; + +const XRSettings = { + loadedDeviceName: 'OpenVR', +}; + +export { + Transform, + PlayerPrefs, + XRSettings, +}; diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index dde97a0..713981b 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -1,3 +1,4 @@ +import Transform from './Transform.js'; import ArmTransforms from './ArmTransforms.js'; import ShoulderTransforms from './ShoulderTransforms.js'; import ShoulderPoser from './ShoulderPoser.js'; @@ -89,46 +90,46 @@ class HandSettings Awake() { - upperArmStartRotation = arm.upperArm.rotation; - lowerArmStartRotation = arm.lowerArm.rotation; - wristStartRotation = Quaternion.identity; - if (arm.wrist1 != null) - wristStartRotation = arm.wrist1.rotation; - handStartRotation = arm.hand.rotation; + this.upperArmStartRotation = this.arm.upperArm.rotation; + this.lowerArmStartRotation = this.arm.lowerArm.rotation; + this.wristStartRotation = Quaternion.identity; + if (this.arm.wrist1 != null) + wthis.ristStartRotation = arm.wrist1.rotation; + this.handStartRotation = arm.hand.rotation; } OnEnable() { - setUpperArmRotation(Quaternion.identity); - setLowerArmRotation(Quaternion.identity); - setHandRotation(Quaternion.identity); + this.setUpperArmRotation(Quaternion.identity); + this.setLowerArmRotation(Quaternion.identity); + this.setHandRotation(Quaternion.identity); } LateUpdate() { - updateUpperArmPosition(); - calcElbowInnerAngle(); - rotateShoulder(); - correctElbowRotation(); - if (elbowSettings.calcElbowAngle) + this.updateUpperArmPosition(); + this.calcElbowInnerAngle(); + this.rotateShoulder(); + this.correctElbowRotation(); + if (this.elbowSettings.calcElbowAngle) { - positionElbow(); - if (elbowCorrectionSettings.useFixedElbowWhenNearShoulder) - correctElbowAfterPositioning(); - if (handSettings.rotateElbowWithHandRight) - rotateElbowWithHandRight(); - if (handSettings.rotateElbowWithHandForward) - rotateElbowWithHandFoward(); - rotateHand(); + this.positionElbow(); + if (this.elbowCorrectionSettings.useFixedElbowWhenNearShoulder) + this.correctElbowAfterPositioning(); + if (this.handSettings.rotateElbowWithHandRight) + this.rotateElbowWithHandRight(); + if (this.handSettings.rotateElbowWithHandForward) + this.rotateElbowWithHandFoward(); + this.rotateHand(); } } updateArmAndTurnElbowUp() { - updateUpperArmPosition(); - calcElbowInnerAngle(); - rotateShoulder(); - correctElbowRotation(); + this.updateUpperArmPosition(); + this.calcElbowInnerAngle(); + this.rotateShoulder(); + this.correctElbowRotation(); } updateUpperArmPosition() @@ -138,19 +139,19 @@ class HandSettings calcElbowInnerAngle() { - Vector3 eulerAngles = new Vector3(); - float targetShoulderDistance = (target.position - upperArmPos).magnitude; - float innerAngle; + const eulerAngles = new Vector3(); + const targetShoulderDistance = (this.target.position - this.upperArmPos).magnitude; + let innerAngle; - if (targetShoulderDistance > arm.armLength) + if (targetShoulderDistance > this.arm.armLength) { innerAngle = 0; } else { - innerAngle = Mathf.Acos(Mathf.Clamp((Mathf.Pow(arm.upperArmLength, 2) + Mathf.Pow(arm.lowerArmLength, 2) - - Mathf.Pow(targetShoulderDistance, 2)) / (2 * arm.upperArmLength * arm.lowerArmLength), -1, 1)) * Mathf.Rad2Deg; - if (left) + innerAngle = Mathf.Acos(Mathf.Clamp((Mathf.Pow(this.arm.upperArmLength, 2) + Mathf.Pow(this.arm.lowerArmLength, 2) - + Mathf.Pow(targetShoulderDistance, 2)) / (2 * this.arm.upperArmLength * this.arm.lowerArmLength), -1, 1)) * Mathf.Rad2Deg; + if (this.left) innerAngle = 180 - innerAngle; else innerAngle = 180 + innerAngle; @@ -161,65 +162,65 @@ class HandSettings } eulerAngles.y = innerAngle; - nextLowerArmAngle = eulerAngles; + this.nextLowerArmAngle = eulerAngles; } //source: https://github.com/NickHardeman/ofxIKArm/blob/master/src/ofxIKArm.cpp rotateShoulder() { - Vector3 eulerAngles = new Vector3(); - Vector3 targetShoulderDirection = (target.position - upperArmPos).normalized; - float targetShoulderDistance = (target.position - upperArmPos).magnitude; + const eulerAngles = new Vector3(); + const targetShoulderDirection = (target.position - upperArmPos).normalized; + const targetShoulderDistance = (target.position - upperArmPos).magnitude; - eulerAngles.y = (left ? -1 : 1) * - Mathf.Acos(Mathf.Clamp((Mathf.Pow(targetShoulderDistance, 2) + Mathf.Pow(arm.upperArmLength, 2) - - Mathf.Pow(arm.lowerArmLength, 2)) / (2 * targetShoulderDistance * arm.upperArmLength), -1, 1)) * Mathf.Rad2Deg; + eulerAngles.y = (this.left ? -1 : 1) * + Mathf.Acos(Mathf.Clamp((Mathf.Pow(targetShoulderDistance, 2) + Mathf.Pow(this.arm.upperArmLength, 2) - + Mathf.Pow(this.arm.lowerArmLength, 2)) / (2 * targetShoulderDistance * this.arm.upperArmLength), -1, 1)) * Mathf.Rad2Deg; if (float.IsNaN(eulerAngles.y)) eulerAngles.y = 0; - Quaternion shoulderRightRotation = Quaternion.FromToRotation(armDirection, targetShoulderDirection); - setUpperArmRotation(shoulderRightRotation); - arm.upperArm.rotation = Quaternion.AngleAxis(eulerAngles.y, lowerArmRotation * Vector3.up) * arm.upperArm.rotation; - setLowerArmLocalRotation(Quaternion.Euler(nextLowerArmAngle)); + const shoulderRightRotation = Quaternion.FromToRotation(this.armDirection, targetShoulderDirection); + this.setUpperArmRotation(shoulderRightRotation); + this.arm.upperArm.rotation = Quaternion.AngleAxis(eulerAngles.y, lowerArmRotation * Vector3.up) * this.arm.upperArm.rotation; + this.setLowerArmLocalRotation(Quaternion.Euler(this.nextLowerArmAngle)); } getElbowTargetAngle() { - Vector3 localHandPosNormalized = shoulderAnker.InverseTransformPoint(handPos) / arm.armLength; + const localHandPosNormalized = this.shoulderAnker.InverseTransformPoint(this.handPos) / this.arm.armLength; // angle from Y - var angle = elbowSettings.yWeight * localHandPosNormalized.y + elbowSettings.offsetAngle; + let angle = this.elbowSettings.yWeight * localHandPosNormalized.y + this.elbowSettings.offsetAngle; // angle from Z /*angle += Mathf.Lerp(elbowSettings.zWeightBottom, elbowSettings.zWeightTop, Mathf.Clamp01((localHandPosNormalized.y + 1f) - elbowSettings.zBorderY)) * (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0f));*/ if (localHandPosNormalized.y > 0) - angle += elbowSettings.zWeightTop * (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0) * Mathf.Max(localHandPosNormalized.y, 0)); + angle += this.elbowSettings.zWeightTop * (Mathf.Max(this.elbowSettings.zDistanceStart - localHandPosNormalized.z, 0) * Mathf.Max(localHandPosNormalized.y, 0)); else - angle += elbowSettings.zWeightBottom * (Mathf.Max(elbowSettings.zDistanceStart - localHandPosNormalized.z, 0) * Mathf.Max(-localHandPosNormalized.y, 0)); + angle += this.elbowSettings.zWeightBottom * (Mathf.Max(this.elbowSettings.zDistanceStart - localHandPosNormalized.z, 0) * Mathf.Max(-localHandPosNormalized.y, 0)); // angle from X - angle += elbowSettings.xWeight * Mathf.Max(localHandPosNormalized.x * (left ? 1.0 : -1.0) + elbowSettings.xDistanceStart, 0); + angle += this.elbowSettings.xWeight * Mathf.Max(localHandPosNormalized.x * (this.left ? 1.0 : -1.0) + this.elbowSettings.xDistanceStart, 0); - if (elbowSettings.clampElbowAngle) + if (this.elbowSettings.clampElbowAngle) { - if (elbowSettings.softClampElbowAngle) + if (this.elbowSettings.softClampElbowAngle) { - if (angle < elbowSettings.minAngle + elbowSettings.softClampRange) + if (angle < this.elbowSettings.minAngle + this.elbowSettings.softClampRange) { - float a = elbowSettings.minAngle + elbowSettings.softClampRange - angle; - angle = elbowSettings.minAngle + elbowSettings.softClampRange * (1 - Mathf.Log(1 + a) * 3); + const a = this.elbowSettings.minAngle + this.elbowSettings.softClampRange - angle; + angle = this.elbowSettings.minAngle + this.elbowSettings.softClampRange * (1 - Mathf.Log(1 + a) * 3); } } else { - angle = Mathf.Clamp(angle, elbowSettings.minAngle, elbowSettings.maxAngle); + angle = Mathf.Clamp(angle, this.elbowSettings.minAngle, this.elbowSettings.maxAngle); } } - if (left) + if (this.left) angle *= -1; return angle; @@ -227,27 +228,27 @@ class HandSettings correctElbowRotation() { - var s = beforePositioningSettings; + const s = beforePositioningSettings; - Vector3 localTargetPos = shoulderAnker.InverseTransformPoint(target.position) / arm.armLength; - float elbowOutsideFactor = Mathf.Clamp01( + const localTargetPos = this.shoulderAnker.InverseTransformPoint(target.position) / arm.armLength; + const elbowOutsideFactor = Mathf.Clamp01( Mathf.Clamp01((s.startBelowZ - localTargetPos.z) / Mathf.Abs(s.startBelowZ) * .5) * Mathf.Clamp01((localTargetPos.y - s.startAboveY) / Mathf.Abs(s.startAboveY)) * - Mathf.Clamp01(1 - localTargetPos.x * (left ? -1 : 1)) + Mathf.Clamp01(1 - localTargetPos.x * (this.left ? -1 : 1)) ) * s.weight; - Vector3 shoulderHandDirection = (upperArmPos - handPos).normalized; - Vector3 targetDir = shoulder.transform.rotation * (Vector3.up + (s.correctElbowOutside ? (armDirection + Vector3.forward * -.2) * elbowOutsideFactor : Vector3.zero)); - Vector3 cross = Vector3.Cross(shoulderHandDirection, targetDir * 1000); + const shoulderHandDirection = (this.upperArmPos - this.handPos).normalized; + const targetDir = this.shoulder.transform.rotation * (Vector3.up + (s.correctElbowOutside ? (this.armDirection + Vector3.forward * -.2) * elbowOutsideFactor : Vector3.zero)); + const cross = Vector3.Cross(shoulderHandDirection, targetDir * 1000); - Vector3 upperArmUp = upperArmRotation * Vector3.up; + const upperArmUp = upperArmRotation * Vector3.up; - float elbowTargetUp = Vector3.Dot(upperArmUp, targetDir); - float elbowAngle = Vector3.Angle(cross, upperArmUp) + (left ? 0 : 180); - Quaternion rotation = Quaternion.AngleAxis(elbowAngle * Mathf.Sign(elbowTargetUp), shoulderHandDirection); - arm.upperArm.rotation = rotation * arm.upperArm.rotation; + const elbowTargetUp = Vector3.Dot(upperArmUp, targetDir); + const elbowAngle = Vector3.Angle(cross, upperArmUp) + (this.left ? 0 : 180); + const rotation = Quaternion.AngleAxis(elbowAngle * Mathf.Sign(elbowTargetUp), shoulderHandDirection); + this.arm.upperArm.rotation = rotation * this.arm.upperArm.rotation; } /// @@ -255,81 +256,81 @@ class HandSettings /// correctElbowAfterPositioning() { - var s = elbowCorrectionSettings; - Vector3 localTargetPos = shoulderAnker.InverseTransformPoint(target.position) / arm.armLength; - Vector3 shoulderHandDirection = (upperArmPos - handPos).normalized; - Vector3 elbowPos = s.localElbowPos; + const s = this.elbowCorrectionSettings; + const localTargetPos = this.shoulderAnker.InverseTransformPoint(this.target.position) / this.arm.armLength; + const shoulderHandDirection = (this.upperArmPos - this.handPos).normalized; + const elbowPos = s.localElbowPos; - if (left) + if (this.left) elbowPos.x *= -1; - Vector3 targetDir = shoulder.transform.rotation * elbowPos.normalized; - Vector3 cross = Vector3.Cross(shoulderHandDirection, targetDir); + const targetDir = this.shoulder.transform.rotation * elbowPos.normalized; + const cross = Vector3.Cross(shoulderHandDirection, targetDir); - Vector3 upperArmUp = upperArmRotation * Vector3.up; + const upperArmUp = this.upperArmRotation * Vector3.up; - Vector3 distance = target.position - upperArmPos; - distance = distance.magnitude * shoulder.transform.InverseTransformDirection(distance / distance.magnitude); + let distance = this.target.position - this.upperArmPos; + distance = distance.magnitude * this.shoulder.transform.InverseTransformDirection(distance / distance.magnitude); - float weight = Mathf.Clamp01(Mathf.Clamp01((s.startBelowDistance - distance.xz().magnitude / arm.armLength) / + const weight = Mathf.Clamp01(Mathf.Clamp01((s.startBelowDistance - distance.xz().magnitude / arm.armLength) / s.startBelowDistance) * s.weight + Mathf.Clamp01((-distance.z + .1) * 3)) * Mathf.Clamp01((s.startBelowY - localTargetPos.y) / s.startBelowY); - float elbowTargetUp = Vector3.Dot(upperArmUp, targetDir); - float elbowAngle2 = Vector3.Angle(cross, upperArmUp) + (left ? 0 : 180); - Quaternion rotation = Quaternion.AngleAxis((elbowAngle2 * Mathf.Sign(elbowTargetUp)).toSignedEulerAngle() * Mathf.Clamp(weight, 0, 1), shoulderHandDirection); - arm.upperArm.rotation = rotation * arm.upperArm.rotation; + const elbowTargetUp = Vector3.Dot(upperArmUp, targetDir); + const elbowAngle2 = Vector3.Angle(cross, upperArmUp) + (this.left ? 0 : 180); + const rotation = Quaternion.AngleAxis((elbowAngle2 * Mathf.Sign(elbowTargetUp)).toSignedEulerAngle() * Mathf.Clamp(weight, 0, 1), shoulderHandDirection); + this.arm.upperArm.rotation = rotation * this.arm.upperArm.rotation; } - rotateElbow(float angle) + rotateElbow(angle) { - Vector3 shoulderHandDirection = (upperArmPos - handPos).normalized; + const shoulderHandDirection = (this.upperArmPos - this.handPos).normalized; - Quaternion rotation = Quaternion.AngleAxis(angle, shoulderHandDirection); - setUpperArmRotation(rotation * upperArmRotation); + const rotation = Quaternion.AngleAxis(angle, shoulderHandDirection); + this.setUpperArmRotation(rotation * this.upperArmRotation); } //source: https://github.com/NickHardeman/ofxIKArm/blob/master/src/ofxIKArm.cpp positionElbow() { - float targetElbowAngle = getElbowTargetAngle(); - rotateElbow(targetElbowAngle); + const targetElbowAngle = this.getElbowTargetAngle(); + this.rotateElbow(targetElbowAngle); } rotateElbowWithHandRight() { - var s = handSettings; - Vector3 handUpVec = target.rotation * Vector3.up; - float forwardAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.right, target.rotation * Vector3.right, + const s = this.handSettings; + let handUpVec = this.target.rotation * Vector3.up; + const forwardAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.right, this.target.rotation * Vector3.right, lowerArmRotation * Vector3.up, lowerArmRotation * Vector3.forward); // todo reduce influence if hand local forward rotation is high (hand tilted inside) - Quaternion handForwardRotation = Quaternion.AngleAxis(-forwardAngle, lowerArmRotation * Vector3.forward); + const handForwardRotation = Quaternion.AngleAxis(-forwardAngle, lowerArmRotation * Vector3.forward); handUpVec = handForwardRotation * handUpVec; - float elbowTargetAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.up, handUpVec, - lowerArmRotation * Vector3.forward, lowerArmRotation * armDirection); + const elbowTargetAngle = VectorHelpers.getAngleBetween(this.lowerArmRotation * Vector3.up, handUpVec, + this.lowerArmRotation * Vector3.forward, this.lowerArmRotation * this.armDirection); - float deltaElbow = (elbowTargetAngle + (left ? -s.handDeltaOffset : s.handDeltaOffset)) / 180; + let deltaElbow = (elbowTargetAngle + (this.left ? -s.handDeltaOffset : s.handDeltaOffset)) / 180; deltaElbow = Mathf.Sign(deltaElbow) * Mathf.Pow(Mathf.Abs(deltaElbow), s.handDeltaPow) * 180 * s.handDeltaFactor; - interpolatedDeltaElbow = - Mathf.LerpAngle(interpolatedDeltaElbow, deltaElbow, Time.deltaTime / s.rotateElbowWithHandDelay); - rotateElbow(interpolatedDeltaElbow); + this.interpolatedDeltaElbow = + Mathf.LerpAngle(this.interpolatedDeltaElbow, deltaElbow, Time.deltaTime / s.rotateElbowWithHandDelay); + this.rotateElbow(this.interpolatedDeltaElbow); } rotateElbowWithHandFoward() { - var s = handSettings; - Vector3 handRightVec = target.rotation * armDirection; + const s = this.handSettings; + const handRightVec = this.target.rotation * this.armDirection; - float elbowTargetAngleForward = VectorHelpers.getAngleBetween(lowerArmRotation * armDirection, handRightVec, - lowerArmRotation * Vector3.up, lowerArmRotation * Vector3.forward); + const elbowTargetAngleForward = VectorHelpers.getAngleBetween(this.lowerArmRotation * armDirection, handRightVec, + this.lowerArmRotation * Vector3.up, this.lowerArmRotation * Vector3.forward); - float deltaElbowForward = (elbowTargetAngleForward + (left ? -s.handDeltaForwardOffset : s.handDeltaForwardOffset)) / 180; + let deltaElbowForward = (elbowTargetAngleForward + (this.left ? -s.handDeltaForwardOffset : s.handDeltaForwardOffset)) / 180; if (Mathf.Abs(deltaElbowForward) < s.handDeltaForwardDeadzone) deltaElbowForward = 0; @@ -339,86 +340,86 @@ class HandSettings } deltaElbowForward = Mathf.Sign(deltaElbowForward) * Mathf.Pow(Mathf.Abs(deltaElbowForward), s.handDeltaForwardPow) * 180; - interpolatedDeltaElbowForward = Mathf.LerpAngle(interpolatedDeltaElbowForward, deltaElbowForward, Time.deltaTime / s.rotateElbowWithHandDelay); + this.interpolatedDeltaElbowForward = Mathf.LerpAngle(this.interpolatedDeltaElbowForward, deltaElbowForward, Time.deltaTime / s.rotateElbowWithHandDelay); - float signedInterpolated = interpolatedDeltaElbowForward.toSignedEulerAngle(); - rotateElbow(signedInterpolated * s.handDeltaForwardFactor); + const signedInterpolated = this.interpolatedDeltaElbowForward.toSignedEulerAngle(); + this.rotateElbow(signedInterpolated * s.handDeltaForwardFactor); } rotateHand() { - if (handSettings.useWristRotation) + if (this.handSettings.useWristRotation) { - Vector3 handUpVec = target.rotation * Vector3.up; - float forwardAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.right, target.rotation * Vector3.right, - lowerArmRotation * Vector3.up, lowerArmRotation * Vector3.forward); + let handUpVec = this.target.rotation * Vector3.up; + const forwardAngle = VectorHelpers.getAngleBetween(this.lowerArmRotation * Vector3.right, target.rotation * Vector3.right, + this.lowerArmRotation * Vector3.up, this.lowerArmRotation * Vector3.forward); // todo reduce influence if hand local forward rotation is high (hand tilted inside) - Quaternion handForwardRotation = Quaternion.AngleAxis(-forwardAngle, lowerArmRotation * Vector3.forward); + const handForwardRotation = Quaternion.AngleAxis(-forwardAngle, this.lowerArmRotation * Vector3.forward); handUpVec = handForwardRotation * handUpVec; - float elbowTargetAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.up, handUpVec, - lowerArmRotation * Vector3.forward, lowerArmRotation * armDirection); + let elbowTargetAngle = VectorHelpers.getAngleBetween(this.lowerArmRotation * Vector3.up, handUpVec, + this.lowerArmRotation * Vector3.forward, this.lowerArmRotation * this.armDirection); elbowTargetAngle = Mathf.Clamp(elbowTargetAngle, -90, 90); if (arm.wrist1 != null) - setWrist1Rotation(Quaternion.AngleAxis(elbowTargetAngle * .3, lowerArmRotation * armDirection) * lowerArmRotation); + this.setWrist1Rotation(Quaternion.AngleAxis(elbowTargetAngle * .3, this.lowerArmRotation * this.armDirection) * this.lowerArmRotation); if (arm.wrist2 != null) - setWrist2Rotation(Quaternion.AngleAxis(elbowTargetAngle * .8, lowerArmRotation * armDirection) * lowerArmRotation); + this.setWrist2Rotation(Quaternion.AngleAxis(elbowTargetAngle * .8, this.lowerArmRotation * this.armDirection) * this.lowerArmRotation); } - setHandRotation(target.rotation); + this.setHandRotation(target.rotation); } - removeShoulderRightRotation(Vector3 direction) { - return Quaternion.AngleAxis(-shoulderPoser.shoulderRightRotation, shoulder.transform.right) * direction; + removeShoulderRightRotation(direction) { + return Quaternion.AngleAxis(-this.shoulderPoser.shoulderRightRotation, this.shoulder.transform.right) * this.direction; } get armDirection() { - return left ? Vector3.left : Vector3.right; + return this.left ? Vector3.left : Vector3.right; } get upperArmPos() { - return arm.upperArm.position; + return this.arm.upperArm.position; } get lowerArmPos() { - return arm.lowerArm.position; + return this.arm.lowerArm.position; } get handPos() { - return arm.hand.position; + return this.arm.hand.position; } get shoulderAnker() { - return left ? shoulder.leftShoulderAnchor : shoulder.rightShoulderAnchor; + return this.left ? this.shoulder.leftShoulderAnchor : this.shoulder.rightShoulderAnchor; } get upperArmRotation() { - return arm.upperArm.rotation * Quaternion.Inverse(upperArmStartRotation); + return this.arm.upperArm.rotation * Quaternion.Inverse(this.upperArmStartRotation); } get lowerArmRotation() { - return arm.lowerArm.rotation * Quaternion.Inverse(lowerArmStartRotation); + return this.arm.lowerArm.rotation * Quaternion.Inverse(this.lowerArmStartRotation); } get handRotation() { - return arm.hand.rotation * Quaternion.Inverse(handStartRotation); + return this.arm.hand.rotation * Quaternion.Inverse(this.handStartRotation); } - setUpperArmRotation(Quaternion rotation) { - return arm.upperArm.rotation = rotation * upperArmStartRotation; + setUpperArmRotation(rotation) { + return this.arm.upperArm.rotation = rotation * this.upperArmStartRotation; } - setLowerArmRotation(Quaternion rotation) { - return arm.lowerArm.rotation = rotation * lowerArmStartRotation; + setLowerArmRotation(rotation) { + return this.arm.lowerArm.rotation = rotation * this.lowerArmStartRotation; } - setLowerArmLocalRotation(Quaternion rotation) { - return arm.lowerArm.rotation = upperArmRotation * rotation * lowerArmStartRotation; + setLowerArmLocalRotation(rotation) { + return this.arm.lowerArm.rotation = this.upperArmRotation * rotation * this.lowerArmStartRotation; } - setWrist1Rotation(Quaternion rotation) { - return arm.wrist1.rotation = rotation * wristStartRotation; + setWrist1Rotation(rotation) { + return this.arm.wrist1.rotation = rotation * this.wristStartRotation; } - setWrist2Rotation(Quaternion rotation) { - return arm.wrist2.rotation = rotation * wristStartRotation; + setWrist2Rotation(rotation) { + return this.arm.wrist2.rotation = rotation * this.wristStartRotation; } - setWristLocalRotation(Quaternion rotation) { - return arm.wrist1.rotation = arm.lowerArm.rotation * rotation * wristStartRotation; + setWristLocalRotation(rotation) { + return this.arm.wrist1.rotation = this.arm.lowerArm.rotation * rotation * this.wristStartRotation; } - setHandRotation(Quaternion rotation) { - return arm.hand.rotation = arm.hand.rotation = rotation * handStartRotation; + setHandRotation(rotation) { + return this.arm.hand.rotation = this.arm.hand.rotation = rotation * this.handStartRotation; } } diff --git a/vrarmik/VRTrackingReferences.js b/vrarmik/VRTrackingReferences.js index 861c2e2..3b0b0ac 100644 --- a/vrarmik/VRTrackingReferences.js +++ b/vrarmik/VRTrackingReferences.js @@ -1,11 +1,15 @@ - class VRTrackingReferences +import Transform from './Transform.js'; + + class VRTrackingReferences { - this.leftController = new Transform(); - this.rightController = new Transform(); - this.hmd = new Transform(); - this.leftHand = new Transform(); - this.rightHand = new Transform(); - this.head = new Transform(); + constructor() { + this.leftController = new Transform(); + this.rightController = new Transform(); + this.hmd = new Transform(); + this.leftHand = new Transform(); + this.rightHand = new Transform(); + this.head = new Transform(); + } } -export default VRTrackingReferences; \ No newline at end of file +export default VRTrackingReferences; From 90b13d7e061b14352f1ef86404c5a3dd0a3d8f59 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 23:30:37 -0400 Subject: [PATCH 034/562] Bugfix euler order --- vrarmik/StaticOffsetTransform.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/vrarmik/StaticOffsetTransform.js b/vrarmik/StaticOffsetTransform.js index c715032..5dae7d1 100644 --- a/vrarmik/StaticOffsetTransform.js +++ b/vrarmik/StaticOffsetTransform.js @@ -16,7 +16,7 @@ class StaticOffsetTransform this.orientationalOffset = new Vector3(); this.referenceRotationMultiplicator = Vector3.one; - this.axisOrder = new EulerOrder(); + this.axisOrder = EulerOrder.XYZ; this.referenceLocalPosition = false; this.referenceLocalRotation = false; From 62d10cc9e1c8ec8d1e316e8d61b59ba8b02d55bc Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 23:31:58 -0400 Subject: [PATCH 035/562] More unity compatibility debugging --- vrarmik/StaticOffsetTransform.js | 2 ++ vrarmik/Unity.js | 57 ++++++++++++++++++++++++++++++++ 2 files changed, 59 insertions(+) diff --git a/vrarmik/StaticOffsetTransform.js b/vrarmik/StaticOffsetTransform.js index 5dae7d1..16d2eeb 100644 --- a/vrarmik/StaticOffsetTransform.js +++ b/vrarmik/StaticOffsetTransform.js @@ -1,3 +1,5 @@ +import {Vector3} from './Unity.js'; + const EulerOrder = { XYZ: 'XYZ', XZY: 'XZY', diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index ec78eb1..4f58df8 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -1,9 +1,66 @@ +const DEG2RAD = Math.PI/180; +const RAD2DEG = 180/Math.PI; + +class Vector3 extends THREE.Vector3 { + static get zero() { + return new Vector3(0, 0, 0); + } + static get one() { + return new Vector3(1, 1, 1); + } + static get right() { + return new Vector3(1, 0, 0); + } + static get up() { + return new Vector3(0, 1, 0); + } + static get forward() { + return new Vector3(0, 0, 1); + } + + get normalized() { + return this.clone().normalize(); + } + Dot(v) { + return this.dot(v); + } + Cross(v) { + return this.clone().cross(v); + } + Angle(v) { + return this.angleTo(v) * RAD2DEG; + } +} + +class Quaternion extends THREE.Quaternion() { + static get identity() { + return new Quaternion(0, 0, 0, 1); + } + static AngleAxis(angle, axis) { + return new Quaternion().setFromAxisAngle(axis, angle * DEG2RAD); + } + static FromToRotation(a, b) { + return new Quaternion().setFromUnitVectors(a, b); + } + static Euler(v) { + return new Quaternion().setFromEuler(new Euler(v.x, v.y, v.z, 'ZXY')); + } + + Inverse() { + return this.clone().inverse(); + } +} + class Transform { constructor() { this.position = new Vector3(); this.rotation = new Quaternion(); this.scale = new Vector3(1, 1, 1); } + get eulerAngles() { + const e = new THREE.Euler().setFromQuaternion(this.rotation, 'ZXY'); + return new Vector3(e.x, e.y, e.z); + } } const PlayerPrefs = { From 0a5a1ebdfcbd6f524c25a62e4c68f42cf99ba107 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 23:47:33 -0400 Subject: [PATCH 036/562] Bugfix unity exposures --- vrarmik/Unity.js | 2 ++ 1 file changed, 2 insertions(+) diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index 4f58df8..469e43e 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -82,6 +82,8 @@ const XRSettings = { }; export { + Vector3, + Quaternion, Transform, PlayerPrefs, XRSettings, From c75ebdf1c373819ccbf005bd7d3f33d232ef2e50 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 23:47:50 -0400 Subject: [PATCH 037/562] Add Unity math library --- vrarmik/Unity.js | 60 ++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 60 insertions(+) diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index 469e43e..ff84444 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -63,6 +63,65 @@ class Transform { } } +const Mathf = { + Deg2Rad: DEG2RAD, + Rad2Deg: RAD2DEG, + PI: Math.PI, + Clamp(v, min, max) { + return Math.min(Math.max(v, min), max); + }, + Clamp01(v) { + return Mathf.Clamp(v, 0, 1); + }, + Min(a, b) { + return Math.min(a, b); + }, + Max(a, b) { + return Math.max(a, b); + }, + Abs(v) { + return Math.abs(v); + }, + Lerp(a, b, t) { + return a*(1-v) + b*v; + }, + LerpAngle(a, b, t) { + const num = Mathf.Repeat(b - a, 360); + if (num > 180) { + num -= 360; + } + return a + num * Mathf.Clamp01(t); + }, + Floor(v) { + return Math.floor(v); + }, + Ceil(v) { + return Math.ceil(v); + }, + Repeat(t, length) { + return t - Mathf.Floor(t / length) * length; + }, + DeltaAngle(current, target) { + let num = Mathf.Repeat(target - current, 360); + if (num > 180) { + num -= 360; + } + return num; + }, + Acos(v) { + return Math.acos(v); + }, + Atan2(a, b) { + return Math.atan2(a, b); + }, + Sign(v) { + return v >= 0 ? 1 : -1; + }, + Pow(a, b) { + return Math.pow(a, b); + }, +}; + const PlayerPrefs = { data: {}, GetFloat(k, d) { @@ -85,6 +144,7 @@ export { Vector3, Quaternion, Transform, + Mathf, PlayerPrefs, XRSettings, }; From 45a4391579c86788014690757ea2186e5b6a6321 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 23:49:24 -0400 Subject: [PATCH 038/562] Add missing Vector3.Scale --- vrarmik/Unity.js | 3 +++ 1 file changed, 3 insertions(+) diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index ff84444..79dca26 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -21,6 +21,9 @@ class Vector3 extends THREE.Vector3 { get normalized() { return this.clone().normalize(); } + Scale(a, b) { + return new Vector3(a.x + b.x, a.y + b.y, a.z + b.z); + } Dot(v) { return this.dot(v); } From 760b63bc6ec387920e9dd88cb97c91a1e09876c4 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 23:51:14 -0400 Subject: [PATCH 039/562] Port VectorHelpers --- vrarmik/Utils/VectorHelpers.js | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/vrarmik/Utils/VectorHelpers.js b/vrarmik/Utils/VectorHelpers.js index dfb63a9..d321590 100644 --- a/vrarmik/Utils/VectorHelpers.js +++ b/vrarmik/Utils/VectorHelpers.js @@ -1,17 +1,19 @@ +import {Vector3, Mathf} from '../Unity.js'; + const VectorHelpers = { - axisAngle(Vector3 v, Vector3 forward, Vector3 axis) + axisAngle(v, forward, axis) { - Vector3 right = Vector3.Cross(axis, forward); + const right = Vector3.Cross(axis, forward); forward = Vector3.Cross(right, axis); return Mathf.Atan2(Vector3.Dot(v, right), Vector3.Dot(v, forward)) * Mathf.Rad2Deg; }, - getAngleBetween(Vector3 a, Vector3 b, Vector3 forward, Vector3 axis) + getAngleBetween(a, b, forward, axis) { - float angleA = axisAngle(a, forward, axis); - float angleB = axisAngle(b, forward, axis); + const angleA = this.axisAngle(a, forward, axis); + const angleB = this.axisAngle(b, forward, axis); return Mathf.DeltaAngle(angleA, angleB); }, }; -export default VectorHelpers; \ No newline at end of file +export default VectorHelpers; From e23529b7150c1cd8e0d659532061c97d19a21338 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 23:57:23 -0400 Subject: [PATCH 040/562] More unity functions work --- vrarmik/ArmTransforms.js | 2 +- vrarmik/AvatarVRTrackingReferences.js | 2 ++ vrarmik/ShoulderPoser.js | 1 + vrarmik/ShoulderTransforms.js | 3 ++- vrarmik/StaticOffsetTransform.js | 2 +- vrarmik/Unity.js | 3 +++ vrarmik/VRArmIK.js | 2 +- 7 files changed, 11 insertions(+), 4 deletions(-) diff --git a/vrarmik/ArmTransforms.js b/vrarmik/ArmTransforms.js index 5895098..e3fbebc 100644 --- a/vrarmik/ArmTransforms.js +++ b/vrarmik/ArmTransforms.js @@ -1,4 +1,4 @@ -import Transform from './Transform.js'; +import {Vector3, Transform} from './Unity.js'; import PoseManager from './PoseManager.js'; class ArmTransforms diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index 1cdba5d..fee3ef6 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -1,3 +1,5 @@ +import {Vector3} from './Unity.js'; + import StaticOffsetTransform from './StaticOffsetTransform.js'; class AvatarVRTrackingReferences diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index 6098137..a3a6cdf 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -1,3 +1,4 @@ +import {Vector3, Quaternion} from './Unity.js'; import ShoulderTransforms from './ShoulderTransforms.js'; import VRTrackingReferences from './VRTrackingReferences.js'; import PoseManager from './PoseManager.js'; diff --git a/vrarmik/ShoulderTransforms.js b/vrarmik/ShoulderTransforms.js index d9b4be0..bdaa9fe 100644 --- a/vrarmik/ShoulderTransforms.js +++ b/vrarmik/ShoulderTransforms.js @@ -1,3 +1,4 @@ +import {Vector3, Transform} from './Unity.js'; import ArmTransforms from './ArmTransforms.js'; class ShoulderTransforms @@ -53,4 +54,4 @@ class ShoulderTransforms } } -export default ShoulderTransforms; \ No newline at end of file +export default ShoulderTransforms; diff --git a/vrarmik/StaticOffsetTransform.js b/vrarmik/StaticOffsetTransform.js index 16d2eeb..369a414 100644 --- a/vrarmik/StaticOffsetTransform.js +++ b/vrarmik/StaticOffsetTransform.js @@ -1,4 +1,4 @@ -import {Vector3} from './Unity.js'; +import {Vector3, Quaternion} from './Unity.js'; const EulerOrder = { XYZ: 'XYZ', diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index 79dca26..a641680 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -18,6 +18,9 @@ class Vector3 extends THREE.Vector3 { return new Vector3(0, 0, 1); } + get magnitude() { + return this.length(); + } get normalized() { return this.clone().normalize(); } diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index 713981b..b267557 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -1,4 +1,4 @@ -import Transform from './Transform.js'; +import {Vector3, Quaternion, Transform} from './Unity.js'; import ArmTransforms from './ArmTransforms.js'; import ShoulderTransforms from './ShoulderTransforms.js'; import ShoulderPoser from './ShoulderPoser.js'; From a3679ed3716406e1638913e4109fc7b4a61f2343 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 11 Oct 2019 23:59:44 -0400 Subject: [PATCH 041/562] Add more unity transform methods --- vrarmik/Unity.js | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index a641680..4e61753 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -67,6 +67,14 @@ class Transform { const e = new THREE.Euler().setFromQuaternion(this.rotation, 'ZXY'); return new Vector3(e.x, e.y, e.z); } + TransformPoint(v) { + return v.applyMatrix4(new THREE.Matrix4().compose(this.position, this.rotation, this.scale)); + } + InverseTransformPoint(v) { + const m = new THREE.Matrix4().compose(this.position, this.rotation, this.scale); + m.getInverse(m); + return v.applyMatrix4(m); + } } const Mathf = { From bbe6d865035043fca85ebc18c706c444ad9a1247 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sat, 12 Oct 2019 00:00:29 -0400 Subject: [PATCH 042/562] More unity debugging --- vrarmik/VRTrackingReferences.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/vrarmik/VRTrackingReferences.js b/vrarmik/VRTrackingReferences.js index 3b0b0ac..6f57b90 100644 --- a/vrarmik/VRTrackingReferences.js +++ b/vrarmik/VRTrackingReferences.js @@ -1,4 +1,4 @@ -import Transform from './Transform.js'; +import {Transform} from './Unity.js'; class VRTrackingReferences { From 8c32c66c32c231b468024dfe34785eeb8bc8611b Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sat, 12 Oct 2019 00:04:06 -0400 Subject: [PATCH 043/562] Bugfix isNaN calls --- vrarmik/VRArmIK.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index b267557..57bb636 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -155,7 +155,7 @@ class HandSettings innerAngle = 180 - innerAngle; else innerAngle = 180 + innerAngle; - if (float.IsNaN(innerAngle)) + if (isNaN(innerAngle)) { innerAngle = 180; } @@ -175,7 +175,7 @@ class HandSettings eulerAngles.y = (this.left ? -1 : 1) * Mathf.Acos(Mathf.Clamp((Mathf.Pow(targetShoulderDistance, 2) + Mathf.Pow(this.arm.upperArmLength, 2) - Mathf.Pow(this.arm.lowerArmLength, 2)) / (2 * targetShoulderDistance * this.arm.upperArmLength), -1, 1)) * Mathf.Rad2Deg; - if (float.IsNaN(eulerAngles.y)) + if (isNaN(eulerAngles.y)) eulerAngles.y = 0; From 405bc17b38c05343a207a890eca386a2e9783d63 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sat, 12 Oct 2019 00:04:14 -0400 Subject: [PATCH 044/562] Bugfix float references --- vrarmik/PoseManager.js | 2 +- vrarmik/ShoulderTransforms.js | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/vrarmik/PoseManager.js b/vrarmik/PoseManager.js index 8a91bfb..0a41f34 100644 --- a/vrarmik/PoseManager.js +++ b/vrarmik/PoseManager.js @@ -68,7 +68,7 @@ class PoseManager this.savePlayerSize(this.playerHeightHmd, this.playerWidthWrist); } - savePlayerSize(float heightHmd, float widthWrist) + savePlayerSize(heightHmd, widthWrist) { PlayerPrefs.SetFloat("VRArmIK_PlayerHeightHmd", this.heightHmd); PlayerPrefs.SetFloat("VRArmIK_PlayerWidthWrist", this.widthWrist); diff --git a/vrarmik/ShoulderTransforms.js b/vrarmik/ShoulderTransforms.js index bdaa9fe..dfe8e75 100644 --- a/vrarmik/ShoulderTransforms.js +++ b/vrarmik/ShoulderTransforms.js @@ -41,7 +41,7 @@ class ShoulderTransforms setShoulderWidth(PoseManager.Instance.playerWidthShoulders); } - setShoulderWidth(float width) + setShoulderWidth(width) { Vector3 localScale = new Vector3(width * .5, .05, .05); Vector3 localPosition = new Vector3(width * .25, 0, 0); From 9eb580d892818c51978e4656f2907757069f0051 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 01:38:20 -0400 Subject: [PATCH 045/562] Comment out dead method --- vrarmik/AvatarVRTrackingReferences.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index fee3ef6..2d63344 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -44,7 +44,7 @@ import StaticOffsetTransform from './StaticOffsetTransform.js'; } } - createHandTransform(t, name, parent) + /* createHandTransform(t, name, parent) { if (t === null) { @@ -52,7 +52,7 @@ import StaticOffsetTransform from './StaticOffsetTransform.js'; t.transform.localPosition = Vector3.zero; t.transform.parent = parent; } - } + } */ createTransforms() { From 32ae3603ccd2e85109c78d56d78ddc4bf200e4a8 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 01:39:06 -0400 Subject: [PATCH 046/562] More dead code cleanup --- vrarmik/AvatarVRTrackingReferences.js | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index 2d63344..3895c2e 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -64,11 +64,11 @@ import StaticOffsetTransform from './StaticOffsetTransform.js'; connectTransforms() { - /*StaticOffsetTransform */const sot = this.GetOrAddComponent/**/(); - if (sot.reference == null) + /* const sot = this.GetOrAddComponent(StaticOffsetTransform); + if (sot.reference === null) { sot.reference = transform.parent; - } + } */ this.head.reference = this.head.reference !== null ? this.head.reference : PoseManager.Instance.vrTransforms.head; this.hmd.reference = this.hmd.reference !== null ? this.hmd.reference : PoseManager.Instance.vrTransforms.hmd; From 87e41d1fdb7e36a17fac6c0f6731987e62c309d9 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 01:39:29 -0400 Subject: [PATCH 047/562] Syntax cleanup --- vrarmik/ShoulderTransforms.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/vrarmik/ShoulderTransforms.js b/vrarmik/ShoulderTransforms.js index dfe8e75..7ca5c25 100644 --- a/vrarmik/ShoulderTransforms.js +++ b/vrarmik/ShoulderTransforms.js @@ -43,8 +43,8 @@ class ShoulderTransforms setShoulderWidth(width) { - Vector3 localScale = new Vector3(width * .5, .05, .05); - Vector3 localPosition = new Vector3(width * .25, 0, 0); + const localScale = new Vector3(width * .5, .05, .05); + const localPosition = new Vector3(width * .25, 0, 0); leftShoulderRenderer.localScale = localScale; leftShoulderRenderer.localPosition = -localPosition; From 86357305f405bf7e48f2ded13124368f36408570 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 01:40:11 -0400 Subject: [PATCH 048/562] Major Unity transform work --- vrarmik/Unity.js | 172 +++++++++++++++++++++++++++++++++++++++++++++-- 1 file changed, 167 insertions(+), 5 deletions(-) diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index 4e61753..9e21f2b 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -2,6 +2,46 @@ const DEG2RAD = Math.PI/180; const RAD2DEG = 180/Math.PI; class Vector3 extends THREE.Vector3 { + set(x, y, z) { + super.set(x, y, z); + this.onchange && this.onchange(); + } + copy(p) { + super.copy(p); + this.onchange && this.onchange(); + } + + bindOnchange(onchange) { + let x = this.x, y = this.y, z = this.z; + Object.defineProperty(this, 'x', { + get() { + return x; + }, + set(newX) { + x = newX; + onchange(); + }, + }); + Object.defineProperty(this, 'y', { + get() { + return y; + }, + set(newY) { + y = newY; + onchange(); + }, + }); + Object.defineProperty(this, 'z', { + get() { + return z; + }, + set(newZ) { + z = newZ; + onchange(); + }, + }); + } + static get zero() { return new Vector3(0, 0, 0); } @@ -17,7 +57,6 @@ class Vector3 extends THREE.Vector3 { static get forward() { return new Vector3(0, 0, 1); } - get magnitude() { return this.length(); } @@ -38,7 +77,56 @@ class Vector3 extends THREE.Vector3 { } } -class Quaternion extends THREE.Quaternion() { +class Quaternion extends THREE.Quaternion { + set(x, y, z, w) { + super.set(x, y, z, w); + this.onchange && this.onchange(); + } + copy(p) { + super.copy(p); + this.onchange && this.onchange(); + } + + bindOnchange(onchange) { + let x = this.x, y = this.y, z = this.z, w = this.w; + Object.defineProperty(this, 'x', { + get() { + return x; + }, + set(newX) { + x = newX; + onchange(); + }, + }); + Object.defineProperty(this, 'y', { + get() { + return y; + }, + set(newY) { + y = newY; + onchange(); + }, + }); + Object.defineProperty(this, 'z', { + get() { + return z; + }, + set(newZ) { + z = newZ; + onchange(); + }, + }); + Object.defineProperty(this, 'w', { + get() { + return w; + }, + set(newW) { + w = newW; + onchange(); + }, + }); + } + static get identity() { return new Quaternion(0, 0, 0, 1); } @@ -59,10 +147,84 @@ class Quaternion extends THREE.Quaternion() { class Transform { constructor() { - this.position = new Vector3(); - this.rotation = new Quaternion(); - this.scale = new Vector3(1, 1, 1); + this._position = new Vector3(); + this._rotation = new Quaternion(); + this._scale = new Vector3(1, 1, 1); + + this._localPosition = new Vector3(); + const localChange = this.localChange.bind(this); + this._localPosition.bindOnchange(localChange); + this._localRotation = new Quaternion(); + this._localRotation.bindOnchange(localChange); + this._localScale = new Vector3(1, 1, 1); + this._localScale.bindOnchange(localChange); + + this._parent = null; + + this._matrix = new THREE,Matrix4(); + this._matrixWorld = new THREE,Matrix4(); + } + + get position() { + this.updateMatrixWorld(); + return this._position; + } + set position(position) { + this._position.copy(position); + } + get rotation() { + this.updateMatrixWorld(); + return this._rotation; } + set rotation(rotation) { + this._rotation.copy(rotation); + } + get scale() { + this.updateMatrixWorld(); + return this._scale; + } + set scale(scale) { + this._scale.copy(scale); + } + + get localPosition() { + return this._localPosition; + } + set localPosition(localPosition) { + this._localPosition.copy(localPosition); + } + get localRotation() { + return this._localRotation; + } + set localRotation(localRotation) { + this._localRotation.copy(localRotation); + } + get localScale() { + return this._localScale; + } + set localScale(localScale) { + this._localScale.copy(localScale); + } + + updateMatrixWorld() { + if (this.matrixWorldNeedsUpdate) { + this._matrix.compose(this._position, this._rotation, this._scale); + this._matrixWorld.copy(this._matrix); + + if (t.parent) { + t.parent.updateMatrixWorld(); + this._matrixWorld.premultiply(t.parent._matrixWorld); + } + + this._matrixWorld.decompose(this._position, this._rotation, this._scale); + + this.matrixWorldNeedsUpdate = false; + } + } + localChange() { + this.matrixWorldNeedsUpdate = true; + } + get eulerAngles() { const e = new THREE.Euler().setFromQuaternion(this.rotation, 'ZXY'); return new Vector3(e.x, e.y, e.z); From b217cefe7ee00ab7cacdefe72532b937c1d8847b Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 01:40:57 -0400 Subject: [PATCH 049/562] Add Unity MonoBehavior --- vrarmik/Unity.js | 23 +++++++++++++++++++++++ 1 file changed, 23 insertions(+) diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index 9e21f2b..6436ac0 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -239,6 +239,28 @@ class Transform { } } +class MonoBehavior { + constructor(transform = new Transform()) { + this.transform = transform; + this.components = new Map(); + } + + GetComponent(Constructor) { + let component = this.components.get(Constructor); + if (component === undefined) { + component = new Constructor(this.transform); + this.components.set(Constructor, component); + } + return component; + } + GetOrAddComponent(Constructor) { + return this.GetComponent(Constructor); + } + GetComponentInChildren(Constructor) { + return this.GetComponent(Constructor); + } +} + const Mathf = { Deg2Rad: DEG2RAD, Rad2Deg: RAD2DEG, @@ -320,6 +342,7 @@ export { Vector3, Quaternion, Transform, + MonoBehavior, Mathf, PlayerPrefs, XRSettings, From 866b48ac1c50f09a6a00eb7ce3d7c1aa3777f7eb Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 01:41:14 -0400 Subject: [PATCH 050/562] Add Unity GameObject --- vrarmik/Unity.js | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index 6436ac0..a33948b 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -239,6 +239,17 @@ class Transform { } } +class GameObject { + constructor(name) { + this.name = name; + this.transform = new Transform(); + } + + AddComponent(Constructor) { + return new Constructor(this.transform); + } +} + class MonoBehavior { constructor(transform = new Transform()) { this.transform = transform; @@ -342,6 +353,7 @@ export { Vector3, Quaternion, Transform, + GameObject, MonoBehavior, Mathf, PlayerPrefs, From cb54547fe848cf88535c1d3310a4e826b39a1afd Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 01:42:17 -0400 Subject: [PATCH 051/562] Major unity emulation debugging --- vrarmik/ArmTransforms.js | 8 ++++--- vrarmik/AvatarVRTrackingReferences.js | 30 +++++++++++++----------- vrarmik/ShoulderTransforms.js | 33 +++++++++++++++------------ vrarmik/StaticOffsetTransform.js | 8 ++++--- vrarmik/VRArmIK.js | 14 +++++++----- 5 files changed, 54 insertions(+), 39 deletions(-) diff --git a/vrarmik/ArmTransforms.js b/vrarmik/ArmTransforms.js index e3fbebc..03ae8c9 100644 --- a/vrarmik/ArmTransforms.js +++ b/vrarmik/ArmTransforms.js @@ -1,9 +1,11 @@ -import {Vector3, Transform} from './Unity.js'; +import {Vector3, Transform, MonoBehavior} from './Unity.js'; import PoseManager from './PoseManager.js'; -class ArmTransforms +class ArmTransforms extends MonoBehavior { - constructor() { + constructor(transform) { + super(transform); + this.upperArm = new Transform(); this.lowerArm = new Transform(); this.wrist1 = new Transform(); diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index 3895c2e..6092960 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -1,14 +1,16 @@ -import {Vector3} from './Unity.js'; +import {Vector3, GameObject, MonoBehaviour} from './Unity.js'; import StaticOffsetTransform from './StaticOffsetTransform.js'; - class AvatarVRTrackingReferences + class AvatarVRTrackingReferences extends MonoBehaviour { - constructor() { - this.head = new StaticOffsetTransform(); - this.hmd = new StaticOffsetTransform(); - this.leftHand = new StaticOffsetTransform(); - this.rightHand = new StaticOffsetTransform(); + constructor(transform) { + super(transform); + + this.head = null; + this.hmd = null; + this.leftHand = null; + this.rightHand = null; } Start() @@ -34,13 +36,15 @@ import StaticOffsetTransform from './StaticOffsetTransform.js'; } - createTransform(t, name) + createTransform(k) { + let t = this[k]; if (t === null) { - t = new GameObject(name).AddComponent(); + t = new GameObject(name).AddComponent(StaticOffsetTransform); t.transform.parent = transform; this.setStaticOffsetSettings(t); + this[k] = t; } } @@ -56,10 +60,10 @@ import StaticOffsetTransform from './StaticOffsetTransform.js'; createTransforms() { - this.createTransform(head, nameof(head)); - this.createTransform(leftHand, nameof(leftHand)); - this.createTransform(rightHand, nameof(rightHand)); - this.createTransform(hmd, nameof(hmd)); + this.createTransform('head'); + this.createTransform('leftHand'); + this.createTransform('rightHand'); + this.createTransform('hmd'); } connectTransforms() diff --git a/vrarmik/ShoulderTransforms.js b/vrarmik/ShoulderTransforms.js index 7ca5c25..3400808 100644 --- a/vrarmik/ShoulderTransforms.js +++ b/vrarmik/ShoulderTransforms.js @@ -1,44 +1,49 @@ -import {Vector3, Transform} from './Unity.js'; +import {Vector3, Transform, MonoBehavior} from './Unity.js'; import ArmTransforms from './ArmTransforms.js'; +import ShoulderPoser from './ShoulderPoser.js'; -class ShoulderTransforms +function Instantiate() { + // XXX +} + +class ShoulderTransforms extends MonoBehavior { - constructor() { + constructor(transform) { + constructor(transform); + this.leftShoulder = new Transform(); this.rightShoulder = new Transform(); this.leftShoulderRenderer = new Transform(); this.rightShoulderRenderer = new Transform(); this.leftShoulderAnchor = Transform(); this.rightShoulderAnchor = Transform(); - this.leftArmDummy = new ArmTransforms(); - this.rightArmDummy = new ArmTransforms(); - this.leftArm = new ArmTransforms(); - this.rightArm = new ArmTransforms(); + this.leftArm = null; + this.rightArm = null; } Awake() { if (leftArm == null) { - leftArm = Instantiate(leftArmDummy, leftShoulderAnchor.position, leftShoulderAnchor.rotation, leftShoulderAnchor); - var armIk = leftArm.GetComponentInChildren(); + this.leftArm = new ArmTransforms(); + const armIk = leftArm.GetComponentInChildren(VRArmIK); armIk.shoulder = this; - armIk.shoulderPoser = GetComponent(); + armIk.shoulderPoser = this.GetComponent(ShoulderPoser); armIk.target = armIk.shoulderPoser.avatarTrackingReferences.leftHand.transform; } if (rightArm == null) { - rightArm = Instantiate(rightArmDummy, leftShoulderAnchor.position, rightShoulderAnchor.rotation, rightShoulderAnchor); - var armIk = rightArm.GetComponentInChildren(); + this.rightArm = new ArmTransforms(); + const armIk = rightArm.GetComponentInChildren(VRArmIK); armIk.shoulder = this; - armIk.shoulderPoser = GetComponent(); + armIk.shoulderPoser = this.GetComponent(ShoulderPoser); armIk.target = armIk.shoulderPoser.avatarTrackingReferences.rightHand.transform; } } Start() { - setShoulderWidth(PoseManager.Instance.playerWidthShoulders); + this.setShoulderWidth(PoseManager.Instance.playerWidthShoulders); } setShoulderWidth(width) diff --git a/vrarmik/StaticOffsetTransform.js b/vrarmik/StaticOffsetTransform.js index 369a414..a20e692 100644 --- a/vrarmik/StaticOffsetTransform.js +++ b/vrarmik/StaticOffsetTransform.js @@ -1,4 +1,4 @@ -import {Vector3, Quaternion} from './Unity.js'; +import {Vector3, Quaternion, MonoBehavior} from './Unity.js'; const EulerOrder = { XYZ: 'XYZ', @@ -9,9 +9,11 @@ const EulerOrder = { ZYX: 'ZYX', }; -class StaticOffsetTransform +class StaticOffsetTransform extends MonoBehavior { - constructor() { + constructor(transform) { + super(transform); + this.reference = null; this.offsetPosition = new Vector3(); this.offsetRotation = new Vector3(); diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index 57bb636..239b247 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -1,8 +1,8 @@ -import {Vector3, Quaternion, Transform} from './Unity.js'; +import {Vector3, Quaternion, Transform, MonoBehavior} from './Unity.js'; import ArmTransforms from './ArmTransforms.js'; import ShoulderTransforms from './ShoulderTransforms.js'; import ShoulderPoser from './ShoulderPoser.js'; -import VectorHelpers from 'Utils/VectorHelpers.js'; +import VectorHelpers from './Utils/VectorHelpers.js'; class ArmIKElbowSettings { @@ -63,10 +63,12 @@ class HandSettings } } - class VRArmIK + class VRArmIK extends MonoBehavior { - constructor() { - this.arm = new ArmTransforms(); // XXX these need to be this'd below + constructor(transform) { + super(transform); + + this.arm = new ArmTransforms(); this.shoulder = new ShoulderTransforms(); this.shoulderPoser = new ShoulderPoser(); this.target = new Transform(); @@ -93,7 +95,7 @@ class HandSettings this.upperArmStartRotation = this.arm.upperArm.rotation; this.lowerArmStartRotation = this.arm.lowerArm.rotation; this.wristStartRotation = Quaternion.identity; - if (this.arm.wrist1 != null) + if (this.arm.wrist1 !== null) wthis.ristStartRotation = arm.wrist1.rotation; this.handStartRotation = arm.hand.rotation; } From 1b45f7509fdf1c9b8c29ac3e8f061c9c488ad3c1 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 01:44:09 -0400 Subject: [PATCH 052/562] Small vrarmik debugging --- vrarmik/ShoulderPoser.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index a3a6cdf..2a377db 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -92,7 +92,7 @@ class ShoulderPoser this.shoulder.rightArm.transform.localPosition = Vector3.zero; } - Debug.DrawRay(this.shoulder.transform.position, this.shoulder.transform.forward); + // Debug.DrawRay(this.shoulder.transform.position, this.shoulder.transform.forward); } rotateLeftShoulder() From 77db25e1262011b1c184ad15424bca320ef8b75c Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 02:02:11 -0400 Subject: [PATCH 053/562] More vrarmik debugging --- vrarmik/AvatarVRTrackingReferences.js | 4 ++-- vrarmik/ShoulderPoser.js | 11 +++++++---- vrarmik/ShoulderTransforms.js | 6 +++--- vrarmik/Unity.js | 4 ++-- 4 files changed, 14 insertions(+), 11 deletions(-) diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index 6092960..cbf635b 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -1,8 +1,8 @@ -import {Vector3, GameObject, MonoBehaviour} from './Unity.js'; +import {Vector3, GameObject, MonoBehavior} from './Unity.js'; import StaticOffsetTransform from './StaticOffsetTransform.js'; - class AvatarVRTrackingReferences extends MonoBehaviour + class AvatarVRTrackingReferences extends MonoBehavior { constructor(transform) { super(transform); diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index 2a377db..01f9036 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -1,14 +1,17 @@ -import {Vector3, Quaternion} from './Unity.js'; +import {Vector3, Quaternion, MonoBehavior} from './Unity.js'; import ShoulderTransforms from './ShoulderTransforms.js'; import VRTrackingReferences from './VRTrackingReferences.js'; +import AvatarVRTrackingReferences from './AvatarVRTrackingReferences.js'; import PoseManager from './PoseManager.js'; -class ShoulderPoser +class ShoulderPoser extends MonoBehavior { - constructor() { + constructor(transform) { + super(transform); + this.shoulder = new ShoulderTransforms(); this.vrTrackingReferences = new VRTrackingReferences(); - this.avatarTrackingReferences = AvatarVRTrackingReferences(); + this.avatarTrackingReferences = new AvatarVRTrackingReferences(); this.headNeckDistance = 0.03; this.neckShoulderDistance = new Vector3(0, -.1, -0.02); diff --git a/vrarmik/ShoulderTransforms.js b/vrarmik/ShoulderTransforms.js index 3400808..b82a134 100644 --- a/vrarmik/ShoulderTransforms.js +++ b/vrarmik/ShoulderTransforms.js @@ -9,14 +9,14 @@ function Instantiate() { class ShoulderTransforms extends MonoBehavior { constructor(transform) { - constructor(transform); + super(transform); this.leftShoulder = new Transform(); this.rightShoulder = new Transform(); this.leftShoulderRenderer = new Transform(); this.rightShoulderRenderer = new Transform(); - this.leftShoulderAnchor = Transform(); - this.rightShoulderAnchor = Transform(); + this.leftShoulderAnchor = new Transform(); + this.rightShoulderAnchor = new Transform(); this.leftArm = null; this.rightArm = null; } diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index a33948b..0a5fef6 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -161,8 +161,8 @@ class Transform { this._parent = null; - this._matrix = new THREE,Matrix4(); - this._matrixWorld = new THREE,Matrix4(); + this._matrix = new THREE.Matrix4(); + this._matrixWorld = new THREE.Matrix4(); } get position() { From b6589e0c8dd2a9d40470ed479c8b95678181bc7b Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 15:33:02 -0400 Subject: [PATCH 054/562] Bugfix wrist start location --- vrarmik/VRArmIK.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index 239b247..29ef9c8 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -96,7 +96,7 @@ class HandSettings this.lowerArmStartRotation = this.arm.lowerArm.rotation; this.wristStartRotation = Quaternion.identity; if (this.arm.wrist1 !== null) - wthis.ristStartRotation = arm.wrist1.rotation; + this.wristStartRotation = arm.wrist1.rotation; this.handStartRotation = arm.hand.rotation; } From 87c41c43a530bc3da79cf3bbdbf05b014cb6ab4d Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 22:45:42 -0400 Subject: [PATCH 055/562] Re-connect missing StaticOffsetTransform component --- vrarmik/AvatarVRTrackingReferences.js | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index cbf635b..d1f64dc 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -68,11 +68,11 @@ import StaticOffsetTransform from './StaticOffsetTransform.js'; connectTransforms() { - /* const sot = this.GetOrAddComponent(StaticOffsetTransform); + const sot = this.GetOrAddComponent(StaticOffsetTransform); if (sot.reference === null) { - sot.reference = transform.parent; - } */ + sot.reference = this.transform.parent; + } this.head.reference = this.head.reference !== null ? this.head.reference : PoseManager.Instance.vrTransforms.head; this.hmd.reference = this.hmd.reference !== null ? this.hmd.reference : PoseManager.Instance.vrTransforms.hmd; From 4545e507708ac0dea22395f746791ae90d04635d Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 22:47:23 -0400 Subject: [PATCH 056/562] Comment out calibration code --- vrarmik/ArmTransforms.js | 2 +- vrarmik/PoseManager.js | 8 ++++---- vrarmik/ShoulderPoser.js | 4 ++-- 3 files changed, 7 insertions(+), 7 deletions(-) diff --git a/vrarmik/ArmTransforms.js b/vrarmik/ArmTransforms.js index 03ae8c9..b03bb2b 100644 --- a/vrarmik/ArmTransforms.js +++ b/vrarmik/ArmTransforms.js @@ -33,7 +33,7 @@ class ArmTransforms extends MonoBehavior Start() { - PoseManager.Instance.onCalibrate += this.updateArmLengths; + // PoseManager.Instance.onCalibrate += this.updateArmLengths; this.updateArmLengths(); } diff --git a/vrarmik/PoseManager.js b/vrarmik/PoseManager.js index 0a41f34..9ea4c43 100644 --- a/vrarmik/PoseManager.js +++ b/vrarmik/PoseManager.js @@ -41,7 +41,7 @@ class PoseManager this.vrSystemOffsetHeight = string.IsNullOrEmpty(device) || device == "OpenVR" ? 0 : this.playerHeightHmd; } - Start() + /* Start() { onCalibrate += OnCalibrate; } @@ -49,7 +49,7 @@ class PoseManager OnCalibrate() { this.playerHeightHmd = Camera.main.transform.position.y; - } + } */ loadPlayerWidthShoulders() { @@ -61,7 +61,7 @@ class PoseManager PlayerPrefs.SetFloat("VRArmIK_PlayerWidthShoulders", width); } - calibrateIK() + /* calibrateIK() { this.playerWidthWrist = (this.vrTransforms.leftHand.position - this.vrTransforms.rightHand.position).magnitude; this.playerHeightHmd = this.vrTransforms.hmd.position.y; @@ -74,7 +74,7 @@ class PoseManager PlayerPrefs.SetFloat("VRArmIK_PlayerWidthWrist", this.widthWrist); this.loadPlayerSize(); this.onCalibrate && this.onCalibrate.Invoke(); - } + } */ loadPlayerSize() { diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index 01f9036..3d0b0d9 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -64,13 +64,13 @@ class ShoulderPoser extends MonoBehavior this.shoulder.transform.InverseTransformPoint(this.shoulder.rightShoulderAnchor.position); } - onCalibrate() + /* onCalibrate() { this.shoulder.leftArm.setArmLength((avatarTrackingReferences.leftHand.transform.position - this.shoulder.leftShoulderAnchor.position) .magnitude); this.shoulder.rightArm.setArmLength((avatarTrackingReferences.rightHand.transform.position - this.shoulder.rightShoulderAnchor.position) .magnitude); - } + } */ Update() { From 8875c50c704a1888473d20fa5deb436af79c2dee Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 22:47:48 -0400 Subject: [PATCH 057/562] Bugfix Unity Vector3 change code --- vrarmik/Unity.js | 17 ++++++++--------- 1 file changed, 8 insertions(+), 9 deletions(-) diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index 0a5fef6..fe53d3d 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -2,15 +2,6 @@ const DEG2RAD = Math.PI/180; const RAD2DEG = 180/Math.PI; class Vector3 extends THREE.Vector3 { - set(x, y, z) { - super.set(x, y, z); - this.onchange && this.onchange(); - } - copy(p) { - super.copy(p); - this.onchange && this.onchange(); - } - bindOnchange(onchange) { let x = this.x, y = this.y, z = this.z; Object.defineProperty(this, 'x', { @@ -40,6 +31,14 @@ class Vector3 extends THREE.Vector3 { onchange(); }, }); + this.set = (_set => function set() { + _set.apply(this, arguments); + onchange(); + })(this.set); + this.copy = (_copy => function copy() { + _copy.apply(this, arguments); + onchange(); + })(this.copy); } static get zero() { From 5e27e09230b91a5697eacec467aa70b56969a983 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 22:48:20 -0400 Subject: [PATCH 058/562] Bugfix Unity Quaternion change code --- vrarmik/Unity.js | 17 ++++++++--------- 1 file changed, 8 insertions(+), 9 deletions(-) diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index fe53d3d..77ff7a5 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -77,15 +77,6 @@ class Vector3 extends THREE.Vector3 { } class Quaternion extends THREE.Quaternion { - set(x, y, z, w) { - super.set(x, y, z, w); - this.onchange && this.onchange(); - } - copy(p) { - super.copy(p); - this.onchange && this.onchange(); - } - bindOnchange(onchange) { let x = this.x, y = this.y, z = this.z, w = this.w; Object.defineProperty(this, 'x', { @@ -124,6 +115,14 @@ class Quaternion extends THREE.Quaternion { onchange(); }, }); + this.set = (_set => function set() { + _set.apply(this, arguments); + onchange(); + })(this.set); + this.copy = (_copy => function copy() { + _copy.apply(this, arguments); + onchange(); + })(this.copy); } static get identity() { From fdfb9c5fd44c285d31819a9a30b3f1a7b2b81baf Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 22:48:50 -0400 Subject: [PATCH 059/562] Ensure Unity MonoBehavior transform is always present --- vrarmik/Unity.js | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index 77ff7a5..c3b7b6b 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -249,7 +249,11 @@ class GameObject { } class MonoBehavior { - constructor(transform = new Transform()) { + constructor(transform) { + if (!transform) { + throw new Error('bad component initialization'); + } + this.transform = transform; this.components = new Map(); } From cad20d66227087511201cfad27aaaa329fe5aef9 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 22:49:41 -0400 Subject: [PATCH 060/562] Bugfix avatar vr tracking references --- vrarmik/AvatarVRTrackingReferences.js | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index d1f64dc..ceaf78d 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -1,4 +1,5 @@ import {Vector3, GameObject, MonoBehavior} from './Unity.js'; +import PoseManager from './PoseManager.js'; import StaticOffsetTransform from './StaticOffsetTransform.js'; @@ -41,8 +42,8 @@ import StaticOffsetTransform from './StaticOffsetTransform.js'; let t = this[k]; if (t === null) { - t = new GameObject(name).AddComponent(StaticOffsetTransform); - t.transform.parent = transform; + t = new GameObject(name).GetComponent(StaticOffsetTransform); + t.transform.parent = this.transform; this.setStaticOffsetSettings(t); this[k] = t; } @@ -76,7 +77,7 @@ import StaticOffsetTransform from './StaticOffsetTransform.js'; this.head.reference = this.head.reference !== null ? this.head.reference : PoseManager.Instance.vrTransforms.head; this.hmd.reference = this.hmd.reference !== null ? this.hmd.reference : PoseManager.Instance.vrTransforms.hmd; - this.leftHand.reference = leftHand.reference !== null + this.leftHand.reference = this.leftHand.reference !== null ? this.leftHand.reference : PoseManager.Instance.vrTransforms.leftHand; this.rightHand.reference = this.rightHand.reference !== null From 9c8363ecafdd2000656249325d6a079ed104b674 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 22:49:58 -0400 Subject: [PATCH 061/562] Bugfix vr tracking references --- vrarmik/VRTrackingReferences.js | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/vrarmik/VRTrackingReferences.js b/vrarmik/VRTrackingReferences.js index 6f57b90..0f9ac37 100644 --- a/vrarmik/VRTrackingReferences.js +++ b/vrarmik/VRTrackingReferences.js @@ -1,8 +1,10 @@ -import {Transform} from './Unity.js'; +import {Transform, MonoBehavior} from './Unity.js'; - class VRTrackingReferences + class VRTrackingReferences extends MonoBehavior { - constructor() { + constructor(transform) { + super(transform); + this.leftController = new Transform(); this.rightController = new Transform(); this.hmd = new Transform(); From 5d8e41b726f922cfd25a93a47be4eb2bfb96c21b Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 22:51:37 -0400 Subject: [PATCH 062/562] PoseManager debugging --- vrarmik/PoseManager.js | 16 ++++++++++------ 1 file changed, 10 insertions(+), 6 deletions(-) diff --git a/vrarmik/PoseManager.js b/vrarmik/PoseManager.js index 9ea4c43..f971788 100644 --- a/vrarmik/PoseManager.js +++ b/vrarmik/PoseManager.js @@ -1,11 +1,15 @@ import VRTrackingReferences from './VRTrackingReferences.js'; -import {XRSettings} from './Unity.js'; +import AvatarVRTrackingReferences from './AvatarVRTrackingReferences.js'; +import {GameObject, MonoBehavior, XRSettings} from './Unity.js'; -class PoseManager +class PoseManager extends MonoBehavior { - constructor() { - this.vrTransforms = new VRTrackingReferences(); - this.OnCalibrateListener = null; + constructor(transform) { + super(transform); + + this.vrTransforms = new GameObject().AddComponent(VRTrackingReferences); + this.avatarVrTransforms = new GameObject().AddComponent(AvatarVRTrackingReferences); + // this.OnCalibrateListener = null; // Oculus uses a different reference position -> 0 is the reference head position if the user is standing in the middle of the room. // In OpenVR, the 0 position is the ground position and the user is then at (0, playerHeightHmd, 0) if he is in the middle of the room, so I need to correct this for shoulder calculation @@ -38,7 +42,7 @@ class PoseManager this.loadPlayerSize(); } const device = XRSettings.loadedDeviceName; - this.vrSystemOffsetHeight = string.IsNullOrEmpty(device) || device == "OpenVR" ? 0 : this.playerHeightHmd; + this.vrSystemOffsetHeight = /*string.IsNullOrEmpty(device) || */device == "OpenVR" ? 0 : this.playerHeightHmd; } /* Start() From ea3d75d693d7fa916ca915396c2fabf7c48d156a Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 22:51:49 -0400 Subject: [PATCH 063/562] ShoulderPoser debugging --- vrarmik/ShoulderPoser.js | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index 3d0b0d9..fe8ab72 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -9,9 +9,9 @@ class ShoulderPoser extends MonoBehavior constructor(transform) { super(transform); - this.shoulder = new ShoulderTransforms(); - this.vrTrackingReferences = new VRTrackingReferences(); - this.avatarTrackingReferences = new AvatarVRTrackingReferences(); + this.shoulder = new GameObject().GetComponent(ShoulderTransforms); + this.vrTrackingReferences = null; + this.avatarTrackingReferences = null; this.headNeckDistance = 0.03; this.neckShoulderDistance = new Vector3(0, -.1, -0.02); @@ -52,12 +52,16 @@ class ShoulderPoser extends MonoBehavior this.leftShoulderAnkerStartLocalPosition = new Vector3(); this.rightShoulderAnkerStartLocalPosition = new Vector3(); + + this.Start(); } Start() { if (this.vrTrackingReferences === null) this.vrTrackingReferences = PoseManager.Instance.vrTransforms; + if (this.avatarTrackingReferences === null) + this.avatarTrackingReferences = PoseManager.Instance.avatarVrTransforms; this.leftShoulderAnkerStartLocalPosition = this.shoulder.transform.InverseTransformPoint(this.shoulder.leftShoulderAnchor.position); this.rightShoulderAnkerStartLocalPosition = From 25743e4f660c490c9d5c46ec9f7954d2c9a48c51 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 22:52:03 -0400 Subject: [PATCH 064/562] ShoulderTransforms debugging --- vrarmik/ShoulderTransforms.js | 19 ++++++++----------- 1 file changed, 8 insertions(+), 11 deletions(-) diff --git a/vrarmik/ShoulderTransforms.js b/vrarmik/ShoulderTransforms.js index b82a134..f6e22e5 100644 --- a/vrarmik/ShoulderTransforms.js +++ b/vrarmik/ShoulderTransforms.js @@ -1,10 +1,7 @@ -import {Vector3, Transform, MonoBehavior} from './Unity.js'; +import {Vector3, Transform, GameObject, MonoBehavior} from './Unity.js'; import ArmTransforms from './ArmTransforms.js'; import ShoulderPoser from './ShoulderPoser.js'; - -function Instantiate() { - // XXX -} +import VRArmIK from './VRArmIK.js'; class ShoulderTransforms extends MonoBehavior { @@ -23,18 +20,18 @@ class ShoulderTransforms extends MonoBehavior Awake() { - if (leftArm == null) + if (this.leftArm === null) { - this.leftArm = new ArmTransforms(); - const armIk = leftArm.GetComponentInChildren(VRArmIK); + this.leftArm = new GameObject().AddComponent(ArmTransforms); + const armIk = this.leftArm.GetComponentInChildren(VRArmIK); armIk.shoulder = this; armIk.shoulderPoser = this.GetComponent(ShoulderPoser); armIk.target = armIk.shoulderPoser.avatarTrackingReferences.leftHand.transform; } - if (rightArm == null) + if (rightArm === null) { - this.rightArm = new ArmTransforms(); - const armIk = rightArm.GetComponentInChildren(VRArmIK); + this.rightArm = new GameObject().AddComponent(ArmTransforms); + const armIk = this.rightArm.GetComponentInChildren(VRArmIK); armIk.shoulder = this; armIk.shoulderPoser = this.GetComponent(ShoulderPoser); armIk.target = armIk.shoulderPoser.avatarTrackingReferences.rightHand.transform; From d71035ceb73df9f72b1a93845cccd6173981ebd3 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 22:52:16 -0400 Subject: [PATCH 065/562] StaticOffsetTransform debugging --- vrarmik/StaticOffsetTransform.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/vrarmik/StaticOffsetTransform.js b/vrarmik/StaticOffsetTransform.js index a20e692..e9216e1 100644 --- a/vrarmik/StaticOffsetTransform.js +++ b/vrarmik/StaticOffsetTransform.js @@ -68,7 +68,7 @@ class StaticOffsetTransform extends MonoBehavior if (this.reference === null) return; - const rot = this.switchAxis(this.referenceLocalRotation ? reference.localEulerAngles : reference.eulerAngles, this.axisOrder) + + const rot = this.switchAxis(this.referenceLocalRotation ? this.reference.localEulerAngles : this.reference.eulerAngles, this.axisOrder) + this.offsetRotation; rot.Scale(referenceRotationMultiplicator); From 2ee4e7c02cf9b678631cadc1389ea8f8a90ad620 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 22:52:45 -0400 Subject: [PATCH 066/562] Unity GameObject/MonoBehavior debugging --- vrarmik/Unity.js | 29 ++++++++++++++++++++++++++++- 1 file changed, 28 insertions(+), 1 deletion(-) diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index c3b7b6b..07b75d5 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -237,15 +237,35 @@ class Transform { } } +const gameObjects = []; class GameObject { constructor(name) { this.name = name; this.transform = new Transform(); } - AddComponent(Constructor) { return new Constructor(this.transform); } + AddChild(child) { + child.transform.parent = this; + } + static startAll() { + for (let i = 0; i < gameObjects.length; i++) { + gameObjects[i].components.forEach(value => { + value.Awake(); + value.OnEnable(); + value.Start(); + }); + } + } + static updateAll() { + for (let i = 0; i < gameObjects.length; i++) { + gameObjects[i].components.forEach(value => { + value.Update(); + value.LateUpdate(); + }); + } + } } class MonoBehavior { @@ -272,6 +292,13 @@ class MonoBehavior { GetComponentInChildren(Constructor) { return this.GetComponent(Constructor); } + + Awake() {} + OnEnable() {} + Start() {} + + Update() {} + LateUpdate() {} } const Mathf = { From acc9af18796a4ee0602b86d0d719ddde4503b4de Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sun, 13 Oct 2019 22:53:16 -0400 Subject: [PATCH 067/562] Bugfix VRArmIK --- vrarmik/VRArmIK.js | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index 29ef9c8..ad9b3d6 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -68,15 +68,15 @@ class HandSettings constructor(transform) { super(transform); - this.arm = new ArmTransforms(); - this.shoulder = new ShoulderTransforms(); - this.shoulderPoser = new ShoulderPoser(); + this.arm = new GameObject().GetComponent(ArmTransforms); + this.shoulder = null; + this.shoulderPoser = null; this.target = new Transform(); this.left = true; this.elbowSettings = new ArmIKElbowSettings(); this.beforePositioningSettings = new BeforePositioningSettings(); - this.elbowCorrectionSettings = ElbowCorrectionSettings(); + this.elbowCorrectionSettings = new ElbowCorrectionSettings(); this.handSettings = new HandSettings(); this.nextLowerArmAngle = new Vector3(); @@ -96,8 +96,8 @@ class HandSettings this.lowerArmStartRotation = this.arm.lowerArm.rotation; this.wristStartRotation = Quaternion.identity; if (this.arm.wrist1 !== null) - this.wristStartRotation = arm.wrist1.rotation; - this.handStartRotation = arm.hand.rotation; + this.wristStartRotation = this.arm.wrist1.rotation; + this.handStartRotation = this.arm.hand.rotation; } OnEnable() From 334bbe9cf1cb562cbd14b8900c357031b2fb4056 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 03:42:49 -0400 Subject: [PATCH 068/562] Bugfix Unity includes --- vrarmik/ArmTransforms.js | 2 +- vrarmik/ShoulderPoser.js | 3 ++- vrarmik/ShoulderTransforms.js | 2 ++ vrarmik/VRArmIK.js | 2 +- 4 files changed, 6 insertions(+), 3 deletions(-) diff --git a/vrarmik/ArmTransforms.js b/vrarmik/ArmTransforms.js index b03bb2b..23694fd 100644 --- a/vrarmik/ArmTransforms.js +++ b/vrarmik/ArmTransforms.js @@ -1,4 +1,4 @@ -import {Vector3, Transform, MonoBehavior} from './Unity.js'; +import {Vector3, Transform, MonoBehavior, Mathf} from './Unity.js'; import PoseManager from './PoseManager.js'; class ArmTransforms extends MonoBehavior diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index fe8ab72..df17718 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -1,8 +1,9 @@ -import {Vector3, Quaternion, MonoBehavior} from './Unity.js'; +import {Vector3, Quaternion, GameObject, MonoBehavior, Mathf} from './Unity.js'; import ShoulderTransforms from './ShoulderTransforms.js'; import VRTrackingReferences from './VRTrackingReferences.js'; import AvatarVRTrackingReferences from './AvatarVRTrackingReferences.js'; import PoseManager from './PoseManager.js'; +import VectorHelpers from './Utils/VectorHelpers.js'; class ShoulderPoser extends MonoBehavior { diff --git a/vrarmik/ShoulderTransforms.js b/vrarmik/ShoulderTransforms.js index f6e22e5..4ccbc42 100644 --- a/vrarmik/ShoulderTransforms.js +++ b/vrarmik/ShoulderTransforms.js @@ -2,6 +2,8 @@ import {Vector3, Transform, GameObject, MonoBehavior} from './Unity.js'; import ArmTransforms from './ArmTransforms.js'; import ShoulderPoser from './ShoulderPoser.js'; import VRArmIK from './VRArmIK.js'; +import PoseManager from './PoseManager.js'; + class ShoulderTransforms extends MonoBehavior { diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index ad9b3d6..3046f04 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -1,4 +1,4 @@ -import {Vector3, Quaternion, Transform, MonoBehavior} from './Unity.js'; +import {Vector3, Quaternion, Transform, GameObject, MonoBehavior, Mathf} from './Unity.js'; import ArmTransforms from './ArmTransforms.js'; import ShoulderTransforms from './ShoulderTransforms.js'; import ShoulderPoser from './ShoulderPoser.js'; From 4f6cf007de9ee0602bc616bdd6590d732926df3b Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 03:47:25 -0400 Subject: [PATCH 069/562] Unity bugfixing --- vrarmik/ArmTransforms.js | 4 +- vrarmik/AvatarVRTrackingReferences.js | 4 +- vrarmik/PoseManager.js | 4 +- vrarmik/ShoulderPoser.js | 4 +- vrarmik/ShoulderTransforms.js | 4 +- vrarmik/StaticOffsetTransform.js | 4 +- vrarmik/Unity.js | 71 +++++++++++++++++++++------ vrarmik/VRArmIK.js | 4 +- vrarmik/VRTrackingReferences.js | 4 +- 9 files changed, 72 insertions(+), 31 deletions(-) diff --git a/vrarmik/ArmTransforms.js b/vrarmik/ArmTransforms.js index 23694fd..e44cbbc 100644 --- a/vrarmik/ArmTransforms.js +++ b/vrarmik/ArmTransforms.js @@ -3,8 +3,8 @@ import PoseManager from './PoseManager.js'; class ArmTransforms extends MonoBehavior { - constructor(transform) { - super(transform); + constructor(...args) { + super(...args); this.upperArm = new Transform(); this.lowerArm = new Transform(); diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index ceaf78d..4517cfe 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -5,8 +5,8 @@ import StaticOffsetTransform from './StaticOffsetTransform.js'; class AvatarVRTrackingReferences extends MonoBehavior { - constructor(transform) { - super(transform); + constructor(...args) { + super(...args); this.head = null; this.hmd = null; diff --git a/vrarmik/PoseManager.js b/vrarmik/PoseManager.js index f971788..d6e141b 100644 --- a/vrarmik/PoseManager.js +++ b/vrarmik/PoseManager.js @@ -4,8 +4,8 @@ import {GameObject, MonoBehavior, XRSettings} from './Unity.js'; class PoseManager extends MonoBehavior { - constructor(transform) { - super(transform); + constructor(...args) { + super(...args); this.vrTransforms = new GameObject().AddComponent(VRTrackingReferences); this.avatarVrTransforms = new GameObject().AddComponent(AvatarVRTrackingReferences); diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index df17718..93add01 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -7,8 +7,8 @@ import VectorHelpers from './Utils/VectorHelpers.js'; class ShoulderPoser extends MonoBehavior { - constructor(transform) { - super(transform); + constructor(...args) { + super(...args); this.shoulder = new GameObject().GetComponent(ShoulderTransforms); this.vrTrackingReferences = null; diff --git a/vrarmik/ShoulderTransforms.js b/vrarmik/ShoulderTransforms.js index 4ccbc42..2b5a892 100644 --- a/vrarmik/ShoulderTransforms.js +++ b/vrarmik/ShoulderTransforms.js @@ -7,8 +7,8 @@ import PoseManager from './PoseManager.js'; class ShoulderTransforms extends MonoBehavior { - constructor(transform) { - super(transform); + constructor(...args) { + super(...args); this.leftShoulder = new Transform(); this.rightShoulder = new Transform(); diff --git a/vrarmik/StaticOffsetTransform.js b/vrarmik/StaticOffsetTransform.js index e9216e1..eb59320 100644 --- a/vrarmik/StaticOffsetTransform.js +++ b/vrarmik/StaticOffsetTransform.js @@ -11,8 +11,8 @@ const EulerOrder = { class StaticOffsetTransform extends MonoBehavior { - constructor(transform) { - super(transform); + constructor(...args) { + super(...args); this.reference = null; this.offsetPosition = new Vector3(); diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index 07b75d5..40f4f95 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -165,53 +165,71 @@ class Transform { get position() { this.updateMatrixWorld(); - return this._position; + return this._position.clone(); } set position(position) { this._position.copy(position); } get rotation() { this.updateMatrixWorld(); - return this._rotation; + return this._rotation.clone(); } set rotation(rotation) { this._rotation.copy(rotation); } get scale() { this.updateMatrixWorld(); - return this._scale; + return this._scale.clone(); } set scale(scale) { this._scale.copy(scale); } get localPosition() { - return this._localPosition; + return this._localPosition.clone(); } set localPosition(localPosition) { this._localPosition.copy(localPosition); } get localRotation() { - return this._localRotation; + return this._localRotation.clone(); } set localRotation(localRotation) { this._localRotation.copy(localRotation); } get localScale() { - return this._localScale; + return this._localScale.clone(); } set localScale(localScale) { this._localScale.copy(localScale); } + get parent() { + return this._parent; + } + set parent(parent) { + this._parent = parent; + this.matrixWorldNeedsUpdate = true; + } + + get right() { + return this.TransformPoint(Vector3.right); + } + get up() { + return this.TransformPoint(Vector3.up); + } + get forward() { + return this.TransformPoint(Vector3.forward); + } + updateMatrixWorld() { if (this.matrixWorldNeedsUpdate) { this._matrix.compose(this._position, this._rotation, this._scale); this._matrixWorld.copy(this._matrix); - if (t.parent) { - t.parent.updateMatrixWorld(); - this._matrixWorld.premultiply(t.parent._matrixWorld); + if (this._parent) { + this._parent.updateMatrixWorld(); + this._matrixWorld.premultiply(this._parent._matrixWorld); } this._matrixWorld.decompose(this._position, this._rotation, this._scale); @@ -241,19 +259,38 @@ const gameObjects = []; class GameObject { constructor(name) { this.name = name; + this.transform = new Transform(); + this.components = new Map(); + this.children = []; + + gameObjects.push(this); } AddComponent(Constructor) { - return new Constructor(this.transform); + let component = this.components.get(Constructor); + if (component === undefined) { + component = new Constructor(this.transform, this.components); + this.components.set(Constructor, component); + } + return component; } AddChild(child) { - child.transform.parent = this; + this.children.push(child); + child.transform.parent = this.transform; } static startAll() { for (let i = 0; i < gameObjects.length; i++) { gameObjects[i].components.forEach(value => { value.Awake(); + }); + } + for (let i = 0; i < gameObjects.length; i++) { + gameObjects[i].components.forEach(value => { value.OnEnable(); + }); + } + for (let i = 0; i < gameObjects.length; i++) { + gameObjects[i].components.forEach(value => { value.Start(); }); } @@ -262,6 +299,10 @@ class GameObject { for (let i = 0; i < gameObjects.length; i++) { gameObjects[i].components.forEach(value => { value.Update(); + }); + } + for (let i = 0; i < gameObjects.length; i++) { + gameObjects[i].components.forEach(value => { value.LateUpdate(); }); } @@ -269,19 +310,19 @@ class GameObject { } class MonoBehavior { - constructor(transform) { - if (!transform) { + constructor(transform, components) { + if (!transform || !components) { throw new Error('bad component initialization'); } this.transform = transform; - this.components = new Map(); + this.components = components; } GetComponent(Constructor) { let component = this.components.get(Constructor); if (component === undefined) { - component = new Constructor(this.transform); + component = new Constructor(this.transform, this.components); this.components.set(Constructor, component); } return component; diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index 3046f04..029bc55 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -65,8 +65,8 @@ class HandSettings class VRArmIK extends MonoBehavior { - constructor(transform) { - super(transform); + constructor(...args) { + super(...args); this.arm = new GameObject().GetComponent(ArmTransforms); this.shoulder = null; diff --git a/vrarmik/VRTrackingReferences.js b/vrarmik/VRTrackingReferences.js index 0f9ac37..50511a5 100644 --- a/vrarmik/VRTrackingReferences.js +++ b/vrarmik/VRTrackingReferences.js @@ -2,8 +2,8 @@ import {Transform, MonoBehavior} from './Unity.js'; class VRTrackingReferences extends MonoBehavior { - constructor(transform) { - super(transform); + constructor(...args) { + super(...args); this.leftController = new Transform(); this.rightController = new Transform(); From 4d1f703ddff6974d2f1353527cdc3bf646a1cdd9 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 03:48:06 -0400 Subject: [PATCH 070/562] Unity Vector3 bugfixing --- vrarmik/Unity.js | 38 +++++++++++++++++++++++++++++++------- 1 file changed, 31 insertions(+), 7 deletions(-) diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index 40f4f95..bdf4e89 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -1,6 +1,15 @@ const DEG2RAD = Math.PI/180; const RAD2DEG = 180/Math.PI; +class Vector2 extends THREE.Vector2 { + get magnitude() { + return this.length(); + } + get normalized() { + return this.clone().normalize(); + } +} + class Vector3 extends THREE.Vector3 { bindOnchange(onchange) { let x = this.x, y = this.y, z = this.z; @@ -47,32 +56,47 @@ class Vector3 extends THREE.Vector3 { static get one() { return new Vector3(1, 1, 1); } + static get left() { + return new Vector3(-1, 0, 0); + } static get right() { return new Vector3(1, 0, 0); } static get up() { return new Vector3(0, 1, 0); } + static get down() { + return new Vector3(0, -1, 0); + } static get forward() { return new Vector3(0, 0, 1); } + static get back() { + return new Vector3(0, 0, -1); + } get magnitude() { return this.length(); } get normalized() { return this.clone().normalize(); } - Scale(a, b) { + xz() { + return new Vector2(this.x, this.z); + } + static Scale(a, b) { return new Vector3(a.x + b.x, a.y + b.y, a.z + b.z); } - Dot(v) { - return this.dot(v); + Scale(v) { + return new Vector3(this.x + v.x, this.y + v.y, this.z + v.z); + } + static Dot(a, b) { + return a.dot(b); } - Cross(v) { - return this.clone().cross(v); + static Cross(a, b) { + return a.clone().cross(b); } - Angle(v) { - return this.angleTo(v) * RAD2DEG; + static Angle(a, b) { + return a.angleTo(b) * RAD2DEG; } } From 83d217557f7759148bff39ea1b0a1f74b2af8eea Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 03:48:26 -0400 Subject: [PATCH 071/562] Unity Quaternion bugfixing --- vrarmik/Unity.js | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index bdf4e89..0fbb111 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -159,7 +159,10 @@ class Quaternion extends THREE.Quaternion { return new Quaternion().setFromUnitVectors(a, b); } static Euler(v) { - return new Quaternion().setFromEuler(new Euler(v.x, v.y, v.z, 'ZXY')); + return new Quaternion().setFromEuler(new THREE.Euler(v.x, v.y, v.z, 'ZXY')); + } + static Inverse(q) { + return q.clone().inverse(); } Inverse() { From e54819276d5264214de47ca4d7eede2a602b7799 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 03:48:35 -0400 Subject: [PATCH 072/562] Unity Transform bugfixing --- vrarmik/Unity.js | 27 +++++++++++++++++++++++---- 1 file changed, 23 insertions(+), 4 deletions(-) diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index 0fbb111..809991a 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -272,13 +272,32 @@ class Transform { const e = new THREE.Euler().setFromQuaternion(this.rotation, 'ZXY'); return new Vector3(e.x, e.y, e.z); } + set eulerAngles(v) { + this.rotation = new THREE.Quaternion().setFromEuler(new THREE.Euler(v.x, v.y, v.z, 'ZXY')); + } + get localEulerAngles() { + const e = new THREE.Euler().setFromQuaternion(this.localRotation, 'ZXY'); + return new Vector3(e.x, e.y, e.z); + } + set localEulerAngles(v) { + this.localRotation = new THREE.Quaternion().setFromEuler(new THREE.Euler(v.x, v.y, v.z, 'ZXY')); + } + TransformPoint(v) { - return v.applyMatrix4(new THREE.Matrix4().compose(this.position, this.rotation, this.scale)); + this.updateMatrixWorld(); + return v.clone().applyMatrix4(this._matrixWorld); } InverseTransformPoint(v) { - const m = new THREE.Matrix4().compose(this.position, this.rotation, this.scale); - m.getInverse(m); - return v.applyMatrix4(m); + this.updateMatrixWorld(); + return v.clone().applyMatrix4(new THREE.Matrix4().getInverse(this._matrixWorld)); + } + TransformDirection(v) { + this.updateMatrixWorld(); + return v.clone().applyMatrix4(this._matrixWorld).normalize(); + } + InverseTransformDirection(v) { + this.updateMatrixWorld(); + return v.clone().applyMatrix4(new THREE.Matrix4().getInverse(this._matrixWorld)).normalize(); } } From 3134a164088b3eb1c855f89642821c2207aac8f2 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 04:23:00 -0400 Subject: [PATCH 073/562] Typo debugging --- vrarmik/ArmTransforms.js | 4 ++-- vrarmik/AvatarVRTrackingReferences.js | 4 ++-- vrarmik/PoseManager.js | 6 ++++-- vrarmik/ShoulderTransforms.js | 13 +++++++------ vrarmik/Unity.js | 5 +++++ 5 files changed, 20 insertions(+), 12 deletions(-) diff --git a/vrarmik/ArmTransforms.js b/vrarmik/ArmTransforms.js index e44cbbc..89883ba 100644 --- a/vrarmik/ArmTransforms.js +++ b/vrarmik/ArmTransforms.js @@ -46,7 +46,7 @@ class ArmTransforms extends MonoBehavior setUpperArmLength(length) { - if (armLengthByScale) + if (this.armLengthByScale) { const oldLowerArmLength = distance(this.lowerArm, this.hand); @@ -81,7 +81,7 @@ class ArmTransforms extends MonoBehavior setArmLength(length) { const upperArmFactor = .48; - if (armLengthByScale) + if (this.armLengthByScale) { this.upperArm.localScale = this.upperArm.localScale / this.armLength * length; this.hand.localScale = Vector3.one / (1 - (1 - this.scaleHandFactor) * (1 - this.upperArm.localScale.x)); diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index 4517cfe..f1a3851 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -69,11 +69,11 @@ import StaticOffsetTransform from './StaticOffsetTransform.js'; connectTransforms() { - const sot = this.GetOrAddComponent(StaticOffsetTransform); + /* const sot = this.GetOrAddComponent(StaticOffsetTransform); if (sot.reference === null) { sot.reference = this.transform.parent; - } + } */ this.head.reference = this.head.reference !== null ? this.head.reference : PoseManager.Instance.vrTransforms.head; this.hmd.reference = this.hmd.reference !== null ? this.hmd.reference : PoseManager.Instance.vrTransforms.hmd; diff --git a/vrarmik/PoseManager.js b/vrarmik/PoseManager.js index d6e141b..ac2ab4d 100644 --- a/vrarmik/PoseManager.js +++ b/vrarmik/PoseManager.js @@ -21,9 +21,11 @@ class PoseManager extends MonoBehavior this.playerWidthWrist = 1.39; this.playerWidthShoulders = 0.31; this.loadPlayerSizeOnAwake = false; + + PoseManager.Instance = this; } - OnEnable() + /* OnEnable() { if (PoseManager.Instance === null) { @@ -33,7 +35,7 @@ class PoseManager extends MonoBehavior { Debug.LogError("Multiple Instances of PoseManager in Scene"); } - } + } */ Awake() { diff --git a/vrarmik/ShoulderTransforms.js b/vrarmik/ShoulderTransforms.js index 2b5a892..1960e19 100644 --- a/vrarmik/ShoulderTransforms.js +++ b/vrarmik/ShoulderTransforms.js @@ -11,9 +11,10 @@ class ShoulderTransforms extends MonoBehavior super(...args); this.leftShoulder = new Transform(); + this.rightShoulder = new Transform(); - this.leftShoulderRenderer = new Transform(); - this.rightShoulderRenderer = new Transform(); + /* this.leftShoulderRenderer = new Transform(); + this.rightShoulderRenderer = new Transform(); */ this.leftShoulderAnchor = new Transform(); this.rightShoulderAnchor = new Transform(); this.leftArm = null; @@ -30,7 +31,7 @@ class ShoulderTransforms extends MonoBehavior armIk.shoulderPoser = this.GetComponent(ShoulderPoser); armIk.target = armIk.shoulderPoser.avatarTrackingReferences.leftHand.transform; } - if (rightArm === null) + if (this.rightArm === null) { this.rightArm = new GameObject().AddComponent(ArmTransforms); const armIk = this.rightArm.GetComponentInChildren(VRArmIK); @@ -40,7 +41,7 @@ class ShoulderTransforms extends MonoBehavior } } - Start() + /* Start() { this.setShoulderWidth(PoseManager.Instance.playerWidthShoulders); } @@ -51,11 +52,11 @@ class ShoulderTransforms extends MonoBehavior const localPosition = new Vector3(width * .25, 0, 0); leftShoulderRenderer.localScale = localScale; - leftShoulderRenderer.localPosition = -localPosition; + leftShoulderRenderer.localPosition = localPosition.clone().multiplyScalar(-1); rightShoulderRenderer.localScale = localScale; rightShoulderRenderer.localPosition = localPosition; - } + } */ } export default ShoulderTransforms; diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index 809991a..f0374d7 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -388,6 +388,10 @@ class MonoBehavior { LateUpdate() {} } +const Time = { + deltaTime: 1/90, +}; + const Mathf = { Deg2Rad: DEG2RAD, Rad2Deg: RAD2DEG, @@ -471,6 +475,7 @@ export { Transform, GameObject, MonoBehavior, + Time, Mathf, PlayerPrefs, XRSettings, From acc24361b589b6281627bfd57a51801a3e235637 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 04:23:32 -0400 Subject: [PATCH 074/562] Add missing Time import --- vrarmik/VRArmIK.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index 029bc55..24baeed 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -1,4 +1,4 @@ -import {Vector3, Quaternion, Transform, GameObject, MonoBehavior, Mathf} from './Unity.js'; +import {Vector3, Quaternion, Transform, GameObject, MonoBehavior, Time, Mathf} from './Unity.js'; import ArmTransforms from './ArmTransforms.js'; import ShoulderTransforms from './ShoulderTransforms.js'; import ShoulderPoser from './ShoulderPoser.js'; From 17dccd07890b60d475e40dbbc1343b7e08c3a22d Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 04:24:18 -0400 Subject: [PATCH 075/562] Bugfix initialization order --- vrarmik/AvatarVRTrackingReferences.js | 2 +- vrarmik/ShoulderTransforms.js | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index f1a3851..6bb4f8e 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -14,7 +14,7 @@ import StaticOffsetTransform from './StaticOffsetTransform.js'; this.rightHand = null; } - Start() + Awake() { this.initTransforms(); } diff --git a/vrarmik/ShoulderTransforms.js b/vrarmik/ShoulderTransforms.js index 1960e19..06b5ff4 100644 --- a/vrarmik/ShoulderTransforms.js +++ b/vrarmik/ShoulderTransforms.js @@ -21,7 +21,7 @@ class ShoulderTransforms extends MonoBehavior this.rightArm = null; } - Awake() + OnEnable() { if (this.leftArm === null) { From ba6dc1ab37b174a47c767c18474d5e0365c591cf Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 04:25:54 -0400 Subject: [PATCH 076/562] Component initialization debugging --- vrarmik/AvatarVRTrackingReferences.js | 2 +- vrarmik/ShoulderPoser.js | 2 +- vrarmik/VRArmIK.js | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index 6bb4f8e..e60936c 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -42,7 +42,7 @@ import StaticOffsetTransform from './StaticOffsetTransform.js'; let t = this[k]; if (t === null) { - t = new GameObject(name).GetComponent(StaticOffsetTransform); + t = new GameObject(name).AddComponent(StaticOffsetTransform); t.transform.parent = this.transform; this.setStaticOffsetSettings(t); this[k] = t; diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index 93add01..41d5911 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -10,7 +10,7 @@ class ShoulderPoser extends MonoBehavior constructor(...args) { super(...args); - this.shoulder = new GameObject().GetComponent(ShoulderTransforms); + this.shoulder = this.GetComponent(ShoulderTransforms); this.vrTrackingReferences = null; this.avatarTrackingReferences = null; diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index 24baeed..667cf6e 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -68,7 +68,7 @@ class HandSettings constructor(...args) { super(...args); - this.arm = new GameObject().GetComponent(ArmTransforms); + this.arm = new GameObject().AddComponent(ArmTransforms); this.shoulder = null; this.shoulderPoser = null; this.target = new Transform(); From 35e15686ff9b91699d0979c5da06ce7e6d93298b Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 04:26:26 -0400 Subject: [PATCH 077/562] Add missing Unity euler angle methods --- vrarmik/VRArmIK.js | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index 667cf6e..fb661cd 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -63,6 +63,19 @@ class HandSettings } } +function toSignedEulerAngle(n) +{ + let result = toPositiveEulerAngle(n); + if (result > 180) + result = result - 360; + return result; +} +function toPositiveEulerAngle(n) +{ + const result = (n % 360 + 360) % 360; + return result; +} + class VRArmIK extends MonoBehavior { constructor(...args) { From 78230d874a21e2d0e29aca536722d16059c56ea3 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 04:27:11 -0400 Subject: [PATCH 078/562] More major Unity debugging --- vrarmik/ArmTransforms.js | 14 ++-- vrarmik/ShoulderPoser.js | 50 +++++++------- vrarmik/StaticOffsetTransform.js | 20 +++--- vrarmik/VRArmIK.js | 113 +++++++++++++++---------------- 4 files changed, 97 insertions(+), 100 deletions(-) diff --git a/vrarmik/ArmTransforms.js b/vrarmik/ArmTransforms.js index 89883ba..c7891c7 100644 --- a/vrarmik/ArmTransforms.js +++ b/vrarmik/ArmTransforms.js @@ -18,17 +18,17 @@ class ArmTransforms extends MonoBehavior } get upperArmLength() { - return distance(this.upperArm, this.lowerArm); + return this.distance(this.upperArm, this.lowerArm); } get lowerArmLength() { - return distance(this.lowerArm, this.hand); + return this.distance(this.lowerArm, this.hand); } get armLength() { return this.upperArmLength + this.lowerArmLength; } distance(a, b) { - return (a.position - b.position).magnitude; + return a.position.distanceTo(b.position); } Start() @@ -39,7 +39,7 @@ class ArmTransforms extends MonoBehavior updateArmLengths() { - const shoulderWidth = (this.upperArm.position - this.lowerArm.position).magnitude; + const shoulderWidth = new Vector3().subVectors(this.upperArm.position, this.lowerArm.position).magnitude; const _armLength = (PoseManager.Instance.playerWidthWrist - shoulderWidth) / 2; this.setArmLength(_armLength); } @@ -50,7 +50,7 @@ class ArmTransforms extends MonoBehavior { const oldLowerArmLength = distance(this.lowerArm, this.hand); - let newScale = this.upperArm.localScale - Vector3.Scale(this.upperArm.localScale, this.scaleAxis).magnitude * this.scaleAxis; + let newScale = new Vector3().subVectors(this.upperArm.localScale, this.scaleAxis.clone().multiplyScalar(Vector3.Scale(this.upperArm.localScale, this.scaleAxis).magnitude)); const scaleFactor = Vector3.Scale(this.upperArm.localScale, this.scaleAxis).magnitude / upperArmLength * length; newScale += this.scaleAxis * scaleFactor; this.upperArm.localScale = newScale; @@ -83,8 +83,8 @@ class ArmTransforms extends MonoBehavior const upperArmFactor = .48; if (this.armLengthByScale) { - this.upperArm.localScale = this.upperArm.localScale / this.armLength * length; - this.hand.localScale = Vector3.one / (1 - (1 - this.scaleHandFactor) * (1 - this.upperArm.localScale.x)); + this.upperArm.localScale = this.upperArm.localScale.clone().divideScalar(this.armLength).multiplyScalar(length); + this.hand.localScale = Vector3.one.divideScalar(1 - (1 - this.scaleHandFactor) * (1 - this.upperArm.localScale.x)); } else { diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index 41d5911..827fe20 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -105,14 +105,14 @@ class ShoulderPoser extends MonoBehavior rotateLeftShoulder() { - this.rotateShoulderUp(shoulder.leftShoulder, this.shoulder.leftArm, this.avatarTrackingReferences.leftHand.transform, + this.rotateShoulderUp(this.shoulder.leftShoulder, this.shoulder.leftArm, this.avatarTrackingReferences.leftHand.transform, this.leftShoulderAnkerStartLocalPosition, 1); } rotateRightShoulder() { - this.rotateShoulderUp(shoulder.rightShoulder, this.shoulder.rightArm, this.avatarTrackingReferences.rightHand.transform, + this.rotateShoulderUp(this.shoulder.rightShoulder, this.shoulder.rightArm, this.avatarTrackingReferences.rightHand.transform, this.rightShoulderAnkerStartLocalPosition, -1); } @@ -120,7 +120,7 @@ class ShoulderPoser extends MonoBehavior initialShoulderLocalPos, angleSign) { const initialShoulderPos = this.shoulder.transform.TransformPoint(initialShoulderLocalPos); - const handShoulderOffset = targetHand.position - this.initialShoulderPos; + const handShoulderOffset = new Vector3().subVectors(targetHand.position, this.initialShoulderPos); const armLength = arm.armLength; const targetAngle = Vector3.zero; @@ -147,15 +147,15 @@ class ShoulderPoser extends MonoBehavior positionShoulder() { - const headNeckOffset = this.avatarTrackingReferences.hmd.transform.rotation * this.headNeckDirectionVector; - const targetPosition = this.avatarTrackingReferences.head.transform.position + headNeckOffset * this.headNeckDistance; + const headNeckOffset = this.headNeckDirectionVector.clone().applyQuaternion(this.avatarTrackingReferences.hmd.transform.rotation); + const targetPosition = new Vector3().addVectors(this.avatarTrackingReferences.head.transform.position, headNeckOffset.clone().multiplyScalar(this.headNeckDistance)); this.shoulder.transform.localPosition = - this.shoulder.transform.parent.InverseTransformPoint(targetPosition) + this.neckShoulderDistance; + new Vector3().addVectors(this.shoulder.transform.parent.InverseTransformPoint(targetPosition), this.neckShoulderDistance); } rotateShoulderUp() { - const angle = getCombinedDirectionAngleUp(); + const angle = this.getCombinedDirectionAngleUp(); const targetRotation = new Vector3(0, angle, 0); @@ -169,12 +169,13 @@ class ShoulderPoser extends MonoBehavior this.clampHeadRotationDeltaUp(targetRotation); } - shouldthis.er.transform.eulerAngles = targetRotation; + this.shoulder.transform.eulerAngles = targetRotation; } rotateShoulderRight() { - const heightDiff = this.vrTrackingReferences.hmd.transform.position.y - PoseManager.Instance.vrSystemOffsetHeight; + + const heightDiff = this.vrTrackingReferences.hmd.position.y - PoseManager.Instance.vrSystemOffsetHeight; const relativeHeightDiff = -heightDiff / PoseManager.Instance.playerHeightHmd; const headRightRotation = VectorHelpers.getAngleBetween(this.shoulder.transform.forward, @@ -189,15 +190,15 @@ class ShoulderPoser extends MonoBehavior const deltaRot = Quaternion.AngleAxis(this.shoulderRightRotation, this.shoulder.transform.right); - this.shoulder.transform.rotation = deltaRot * this.shoulder.transform.rotation; + this.shoulder.transform.rotation = new Quaternion().multiplyQuaternions(deltaRot, this.shoulder.transform.rotation); this.positionShoulderRelative(); } positionShoulderRelative() { const deltaRot = Quaternion.AngleAxis(this.shoulderRightRotation, this.shoulder.transform.right); - const shoulderHeadDiff = this.shoulder.transform.position - this.avatarTrackingReferences.head.transform.position; - this.shoulder.transform.position = deltaRot * shoulderHeadDiff + this.avatarTrackingReferences.head.transform.position; + const shoulderHeadDiff = new Vector3().subVectors(this.shoulder.transform.position, this.avatarTrackingReferences.head.transform.position); + this.shoulder.transform.position = new Vector3().addVectors(shoulderHeadDiff.clone().applyQuaternion(deltaRot), this.avatarTrackingReferences.head.transform.position); } getCombinedDirectionAngleUp() @@ -205,19 +206,19 @@ class ShoulderPoser extends MonoBehavior const leftHand = this.avatarTrackingReferences.leftHand.transform; const rightHand = this.avatarTrackingReferences.rightHand.transform; - const distanceLeftHand = leftHand.position - this.shoulder.transform.position; - const distanceRightHand = rightHand.position - this.shoulder.transform.position; + const distanceLeftHand = new Vector3().subVectors(leftHand.position, this.shoulder.transform.position); + const distanceRightHand = new Vector3().subVectors(rightHand.position, this.shoulder.transform.position); if (this.ignoreYPos) { - this.distanceLeftHand.y = 0; - this.distanceRightHand.y = 0; + distanceLeftHand.y = 0; + distanceRightHand.y = 0; } const directionLeftHand = distanceLeftHand.normalized; const directionRightHand = distanceRightHand.normalized; - const combinedDirection = directionLeftHand + directionRightHand; + const combinedDirection = new Vector3().addVectors(directionLeftHand, directionRightHand); return Mathf.Atan2(combinedDirection.x, combinedDirection.z) * 180 / Mathf.PI; } @@ -263,8 +264,8 @@ class ShoulderPoser extends MonoBehavior clampShoulderHandDistance() { - const leftHandVector = this.avatarTrackingReferences.leftHand.transform.position - this.shoulder.leftShoulderAnchor.position; - const rightHandVector = this.avatarTrackingReferences.rightHand.transform.position - this.shoulder.rightShoulderAnchor.position; + const leftHandVector = new Vector3().subVectors(this.avatarTrackingReferences.leftHand.transform.position, this.shoulder.leftShoulderAnchor.position); + const rightHandVector = new Vector3().subVectors(this.avatarTrackingReferences.rightHand.transform.position, this.shoulder.rightShoulderAnchor.position); const leftShoulderHandDistance = leftHandVector.magnitude; const rightShoulderHandDistance = rightHandVector.magnitude; this.shoulderDislocated = false; @@ -274,9 +275,8 @@ class ShoulderPoser extends MonoBehavior if (leftShoulderHandDistance > this.shoulder.leftArm.armLength * startBeforeFactor) { this.shoulderDislocated = true; - this.shoulder.leftArm.transform.position = this.shoulder.leftShoulderAnchor.position + - leftHandVector.normalized * - (leftShoulderHandDistance - this.shoulder.leftArm.armLength * startBeforeFactor); + this.shoulder.leftArm.transform.position = new Vector3().addVectors(this.shoulder.leftShoulderAnchor.position, + leftHandVector.normalized.multiplyScalar(leftShoulderHandDistance - this.shoulder.leftArm.armLength * startBeforeFactor)); } else { @@ -286,10 +286,8 @@ class ShoulderPoser extends MonoBehavior if (rightShoulderHandDistance > this.shoulder.rightArm.armLength * this.startBeforeFactor) { this.shoulderDislocated = true; - this.shoulder.rightArm.transform.position = this.shoulder.rightShoulderAnchor.position + - rightHandVector.normalized * - (rightShoulderHandDistance - - this.shoulder.rightArm.armLength * startBeforeFactor); + this.shoulder.rightArm.transform.position = new Vector3().addVectors(this.shoulder.rightShoulderAnchor.position, + rightHandVector.normalized.multiplyScalar(rightShoulderHandDistance - this.shoulder.rightArm.armLength * startBeforeFactor)); } else { diff --git a/vrarmik/StaticOffsetTransform.js b/vrarmik/StaticOffsetTransform.js index eb59320..cd54812 100644 --- a/vrarmik/StaticOffsetTransform.js +++ b/vrarmik/StaticOffsetTransform.js @@ -68,27 +68,27 @@ class StaticOffsetTransform extends MonoBehavior if (this.reference === null) return; - const rot = this.switchAxis(this.referenceLocalRotation ? this.reference.localEulerAngles : this.reference.eulerAngles, this.axisOrder) + - this.offsetRotation; - rot.Scale(referenceRotationMultiplicator); + const rot = new Vector3().addVectors(this.switchAxis(this.referenceLocalRotation ? this.reference.localEulerAngles : this.reference.eulerAngles, this.axisOrder), + this.offsetRotation); + rot.multiply(this.referenceRotationMultiplicator); const pos = this.referenceLocalPosition ? this.reference.localPosition : this.reference.position; if (this.applyForwardOffsetAfterRotationOffset) { - pos += Quaternion.Euler(rot) * Vector3.right * this.orientationalOffset.x; - pos += Quaternion.Euler(rot) * Vector3.up * othis.rientationalOffset.y; - pos += Quaternion.Euler(rot) * Vector3.forward * this.orientationalOffset.z; + pos.add(Vector3.right.applyQuaternion(Quaternion.Euler(rot)).multiplyScalar(this.orientationalOffset.x)); + pos.add(Vector3.up.applyQuaternion(Quaternion.Euler(rot)).multiplyScalar(this.orientationalOffset.y)); + pos.add(Vector3.forward.applyQuaternion(Quaternion.Euler(rot)).multiplyScalar(this.orientationalOffset.z)); } else { - pos += this.reference.right * this.orientationalOffset.x; - pos += this.reference.up * this.orientationalOffset.y; - pos += this.reference.forward * this.orientationalOffset.z; + pos.add(this.reference.right.multiplyScalar(this.orientationalOffset.x)); + pos.add(this.reference.up.multiplyScalar(this.orientationalOffset.y)); + pos.add(this.reference.forward.multiplyScalar(this.orientationalOffset.z)); } - pos += offsetPosition; + pos.add(this.offsetPosition); if (this.applyPosition) { diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index fb661cd..f9b71a9 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -155,7 +155,7 @@ function toPositiveEulerAngle(n) calcElbowInnerAngle() { const eulerAngles = new Vector3(); - const targetShoulderDistance = (this.target.position - this.upperArmPos).magnitude; + const targetShoulderDistance = new Vector3().subVectors(this.target.position, this.upperArmPos).magnitude; let innerAngle; if (targetShoulderDistance > this.arm.armLength) @@ -184,8 +184,8 @@ function toPositiveEulerAngle(n) rotateShoulder() { const eulerAngles = new Vector3(); - const targetShoulderDirection = (target.position - upperArmPos).normalized; - const targetShoulderDistance = (target.position - upperArmPos).magnitude; + const targetShoulderDirection = new Vector3().subVectors(this.target.position, this.upperArmPos).normalized; + const targetShoulderDistance = new Vector3().subVectors(this.target.position, this.upperArmPos).magnitude; eulerAngles.y = (this.left ? -1 : 1) * Mathf.Acos(Mathf.Clamp((Mathf.Pow(targetShoulderDistance, 2) + Mathf.Pow(this.arm.upperArmLength, 2) - @@ -193,16 +193,15 @@ function toPositiveEulerAngle(n) if (isNaN(eulerAngles.y)) eulerAngles.y = 0; - const shoulderRightRotation = Quaternion.FromToRotation(this.armDirection, targetShoulderDirection); this.setUpperArmRotation(shoulderRightRotation); - this.arm.upperArm.rotation = Quaternion.AngleAxis(eulerAngles.y, lowerArmRotation * Vector3.up) * this.arm.upperArm.rotation; + this.arm.upperArm.rotation = new Quaternion().multiplyQuaternions(Quaternion.AngleAxis(eulerAngles.y, new Vector3().multiplyVectors(this.lowerArmRotation, Vector3.up)), this.arm.upperArm.rotation); this.setLowerArmLocalRotation(Quaternion.Euler(this.nextLowerArmAngle)); } getElbowTargetAngle() { - const localHandPosNormalized = this.shoulderAnker.InverseTransformPoint(this.handPos) / this.arm.armLength; + const localHandPosNormalized = this.shoulderAnker.InverseTransformPoint(this.handPos).divideScalar(this.arm.armLength); // angle from Y let angle = this.elbowSettings.yWeight * localHandPosNormalized.y + this.elbowSettings.offsetAngle; @@ -243,9 +242,9 @@ function toPositiveEulerAngle(n) correctElbowRotation() { - const s = beforePositioningSettings; + const s = this.beforePositioningSettings; - const localTargetPos = this.shoulderAnker.InverseTransformPoint(target.position) / arm.armLength; + const localTargetPos = this.shoulderAnker.InverseTransformPoint(this.target.position).divideScalar(this.arm.armLength); const elbowOutsideFactor = Mathf.Clamp01( Mathf.Clamp01((s.startBelowZ - localTargetPos.z) / Mathf.Abs(s.startBelowZ) * .5) * @@ -254,16 +253,16 @@ function toPositiveEulerAngle(n) Mathf.Clamp01(1 - localTargetPos.x * (this.left ? -1 : 1)) ) * s.weight; - const shoulderHandDirection = (this.upperArmPos - this.handPos).normalized; - const targetDir = this.shoulder.transform.rotation * (Vector3.up + (s.correctElbowOutside ? (this.armDirection + Vector3.forward * -.2) * elbowOutsideFactor : Vector3.zero)); + const shoulderHandDirection = new Vector3().subVectors(this.upperArmPos, this.handPos).normalized; + const targetDir = new Vector3().addVectors(Vector3.up, (s.correctElbowOutside ? new Vector3().addVectors(this.armDirection, Vector3.forward.multiplyScalar(-.2)) * elbowOutsideFactor : Vector3.zero)).applyQuaternion(this.shoulder.transform.rotation); const cross = Vector3.Cross(shoulderHandDirection, targetDir * 1000); - const upperArmUp = upperArmRotation * Vector3.up; + const upperArmUp = new Vector3().multiplyVectors(this.upperArmRotation, Vector3.up); const elbowTargetUp = Vector3.Dot(upperArmUp, targetDir); const elbowAngle = Vector3.Angle(cross, upperArmUp) + (this.left ? 0 : 180); const rotation = Quaternion.AngleAxis(elbowAngle * Mathf.Sign(elbowTargetUp), shoulderHandDirection); - this.arm.upperArm.rotation = rotation * this.arm.upperArm.rotation; + this.arm.upperArm.rotation = new Quaternion().multiplyQuaternions(rotation, this.arm.upperArm.rotation); } /// @@ -272,39 +271,39 @@ function toPositiveEulerAngle(n) correctElbowAfterPositioning() { const s = this.elbowCorrectionSettings; - const localTargetPos = this.shoulderAnker.InverseTransformPoint(this.target.position) / this.arm.armLength; - const shoulderHandDirection = (this.upperArmPos - this.handPos).normalized; + const localTargetPos = this.shoulderAnker.InverseTransformPoint(this.target.position).divideScalar(this.arm.armLength); + const shoulderHandDirection = new Vector3().subVectors(this.upperArmPos, this.handPos).normalized; const elbowPos = s.localElbowPos; if (this.left) elbowPos.x *= -1; - const targetDir = this.shoulder.transform.rotation * elbowPos.normalized; + const targetDir = elbowPos.normalized.applyQuaternion(this.shoulder.transform.rotation); const cross = Vector3.Cross(shoulderHandDirection, targetDir); - const upperArmUp = this.upperArmRotation * Vector3.up; + const upperArmUp = new Vector3().multiplyVectors(this.upperArmRotation, Vector3.up); - let distance = this.target.position - this.upperArmPos; - distance = distance.magnitude * this.shoulder.transform.InverseTransformDirection(distance / distance.magnitude); + let distance = new Vector3().subVectors(this.target.position, this.upperArmPos); + distance = this.shoulder.transform.InverseTransformDirection(distance.clone().divideScalar(distance.magnitude)).multiplyScalar(distance.magnitude); - const weight = Mathf.Clamp01(Mathf.Clamp01((s.startBelowDistance - distance.xz().magnitude / arm.armLength) / + const weight = Mathf.Clamp01(Mathf.Clamp01((s.startBelowDistance - distance.xz().magnitude / this.arm.armLength) / s.startBelowDistance) * s.weight + Mathf.Clamp01((-distance.z + .1) * 3)) * Mathf.Clamp01((s.startBelowY - localTargetPos.y) / s.startBelowY); const elbowTargetUp = Vector3.Dot(upperArmUp, targetDir); const elbowAngle2 = Vector3.Angle(cross, upperArmUp) + (this.left ? 0 : 180); - const rotation = Quaternion.AngleAxis((elbowAngle2 * Mathf.Sign(elbowTargetUp)).toSignedEulerAngle() * Mathf.Clamp(weight, 0, 1), shoulderHandDirection); - this.arm.upperArm.rotation = rotation * this.arm.upperArm.rotation; + const rotation = Quaternion.AngleAxis(toSignedEulerAngle(elbowAngle2 * Mathf.Sign(elbowTargetUp)) * Mathf.Clamp(weight, 0, 1), shoulderHandDirection); + this.arm.upperArm.rotation = new Quaternion().multiplyQuaternions(rotation, this.arm.upperArm.rotation); } rotateElbow(angle) { - const shoulderHandDirection = (this.upperArmPos - this.handPos).normalized; + const shoulderHandDirection = new Vector3().subVectors(this.upperArmPos, this.handPos).normalized; const rotation = Quaternion.AngleAxis(angle, shoulderHandDirection); - this.setUpperArmRotation(rotation * this.upperArmRotation); + this.setUpperArmRotation(new Quaternion().multiplyQuaternions(rotation, this.upperArmRotation)); } //source: https://github.com/NickHardeman/ofxIKArm/blob/master/src/ofxIKArm.cpp @@ -318,16 +317,16 @@ function toPositiveEulerAngle(n) rotateElbowWithHandRight() { const s = this.handSettings; - let handUpVec = this.target.rotation * Vector3.up; - const forwardAngle = VectorHelpers.getAngleBetween(lowerArmRotation * Vector3.right, this.target.rotation * Vector3.right, - lowerArmRotation * Vector3.up, lowerArmRotation * Vector3.forward); + let handUpVec = Vector3.up.applyQuaternion(this.target.rotation); + const forwardAngle = VectorHelpers.getAngleBetween(Vector3.right.applyQuaternion(this.lowerArmRotation), Vector3.right.applyQuaternion(this.target.rotation), + Vector3.up.applyQuaternion(this.lowerArmRotation), Vector3.forward.applyQuaternion(this.lowerArmRotation)); // todo reduce influence if hand local forward rotation is high (hand tilted inside) - const handForwardRotation = Quaternion.AngleAxis(-forwardAngle, lowerArmRotation * Vector3.forward); - handUpVec = handForwardRotation * handUpVec; + const handForwardRotation = Quaternion.AngleAxis(-forwardAngle, this.lowerArmRotation * Vector3.forward); + handUpVec = handUpVec.applyQuaternion(handForwardRotation); - const elbowTargetAngle = VectorHelpers.getAngleBetween(this.lowerArmRotation * Vector3.up, handUpVec, - this.lowerArmRotation * Vector3.forward, this.lowerArmRotation * this.armDirection); + const elbowTargetAngle = VectorHelpers.getAngleBetween(Vector3.up.applyQuaternion(this.lowerArmRotation), handUpVec, + Vector3.forward.applyQuaternion(this.lowerArmRotation), this.armDirection.clone().applyQuaternion(this.lowerArmRotation)); let deltaElbow = (elbowTargetAngle + (this.left ? -s.handDeltaOffset : s.handDeltaOffset)) / 180; @@ -340,10 +339,10 @@ function toPositiveEulerAngle(n) rotateElbowWithHandFoward() { const s = this.handSettings; - const handRightVec = this.target.rotation * this.armDirection; + const handRightVec = this.armDirection.clone().applyQuaternion(this.target.rotation); - const elbowTargetAngleForward = VectorHelpers.getAngleBetween(this.lowerArmRotation * armDirection, handRightVec, - this.lowerArmRotation * Vector3.up, this.lowerArmRotation * Vector3.forward); + const elbowTargetAngleForward = VectorHelpers.getAngleBetween(this.armDirection.clone().applyQuaternion(this.lowerArmRotation), handRightVec, + Vector3.up.applyQuaternion(this.lowerArmRotation), Vector3.forward.applyQuaternion(this.lowerArmRotation)); let deltaElbowForward = (elbowTargetAngleForward + (this.left ? -s.handDeltaForwardOffset : s.handDeltaForwardOffset)) / 180; @@ -357,7 +356,7 @@ function toPositiveEulerAngle(n) deltaElbowForward = Mathf.Sign(deltaElbowForward) * Mathf.Pow(Mathf.Abs(deltaElbowForward), s.handDeltaForwardPow) * 180; this.interpolatedDeltaElbowForward = Mathf.LerpAngle(this.interpolatedDeltaElbowForward, deltaElbowForward, Time.deltaTime / s.rotateElbowWithHandDelay); - const signedInterpolated = this.interpolatedDeltaElbowForward.toSignedEulerAngle(); + const signedInterpolated = toSignedEulerAngle(this.interpolatedDeltaElbowForward); this.rotateElbow(signedInterpolated * s.handDeltaForwardFactor); } @@ -365,28 +364,28 @@ function toPositiveEulerAngle(n) { if (this.handSettings.useWristRotation) { - let handUpVec = this.target.rotation * Vector3.up; - const forwardAngle = VectorHelpers.getAngleBetween(this.lowerArmRotation * Vector3.right, target.rotation * Vector3.right, - this.lowerArmRotation * Vector3.up, this.lowerArmRotation * Vector3.forward); + let handUpVec = Vector3.up.applyQuaternion(this.target.rotation); + const forwardAngle = VectorHelpers.getAngleBetween(Vector3.right.applyQuaternion(this.lowerArmRotation), Vector3.right.applyQuaternion(this.target.rotation), + Vector3.up.applyQuaternion(this.lowerArmRotation), Vector3.forward.applyQuaternion(this.lowerArmRotation)); // todo reduce influence if hand local forward rotation is high (hand tilted inside) - const handForwardRotation = Quaternion.AngleAxis(-forwardAngle, this.lowerArmRotation * Vector3.forward); - handUpVec = handForwardRotation * handUpVec; + const handForwardRotation = Quaternion.AngleAxis(-forwardAngle, Vector3.forward.applyQuaternion(this.lowerArmRotation)); + handUpVec = handUpVec.applyQuaternion(handForwardRotation); - let elbowTargetAngle = VectorHelpers.getAngleBetween(this.lowerArmRotation * Vector3.up, handUpVec, - this.lowerArmRotation * Vector3.forward, this.lowerArmRotation * this.armDirection); + let elbowTargetAngle = VectorHelpers.getAngleBetween(Vector3.up.applyQuaternion(this.lowerArmRotation), handUpVec, + Vector3.forward.applyQuaternion(this.lowerArmRotation), this.armDirection.clone().applyQuaternion(this.lowerArmRotation)); elbowTargetAngle = Mathf.Clamp(elbowTargetAngle, -90, 90); - if (arm.wrist1 != null) - this.setWrist1Rotation(Quaternion.AngleAxis(elbowTargetAngle * .3, this.lowerArmRotation * this.armDirection) * this.lowerArmRotation); - if (arm.wrist2 != null) - this.setWrist2Rotation(Quaternion.AngleAxis(elbowTargetAngle * .8, this.lowerArmRotation * this.armDirection) * this.lowerArmRotation); + if (this.arm.wrist1 !== null) + this.setWrist1Rotation(new Quaternion().multiplyQuaternions(Quaternion.AngleAxis(elbowTargetAngle * .3, this.armDirection.clone().applyQuaternion(this.lowerArmRotation)), this.lowerArmRotation)); + if (this.arm.wrist2 !== null) + this.setWrist2Rotation(new Quaternion().multiplyQuaternions(Quaternion.AngleAxis(elbowTargetAngle * .8, this.armDirection.clone().applyQuaternion(this.lowerArmRotation)), this.lowerArmRotation)); } - this.setHandRotation(target.rotation); + this.setHandRotation(this.target.rotation); } removeShoulderRightRotation(direction) { - return Quaternion.AngleAxis(-this.shoulderPoser.shoulderRightRotation, this.shoulder.transform.right) * this.direction; + return this.direction.clone().applyQuaternion(Quaternion.AngleAxis(-this.shoulderPoser.shoulderRightRotation, this.shoulder.transform.right)); } get armDirection() { @@ -406,35 +405,35 @@ function toPositiveEulerAngle(n) } get upperArmRotation() { - return this.arm.upperArm.rotation * Quaternion.Inverse(this.upperArmStartRotation); + return new Quaternion().multiplyQuaternions(this.arm.upperArm.rotation, Quaternion.Inverse(this.upperArmStartRotation)); } get lowerArmRotation() { - return this.arm.lowerArm.rotation * Quaternion.Inverse(this.lowerArmStartRotation); + return new Quaternion().multiplyQuaternions(this.arm.lowerArm.rotation, Quaternion.Inverse(this.lowerArmStartRotation)); } get handRotation() { - return this.arm.hand.rotation * Quaternion.Inverse(this.handStartRotation); + return new Quaternion().multiplyQuaternions(this.arm.hand.rotation, Quaternion.Inverse(this.handStartRotation)); } setUpperArmRotation(rotation) { - return this.arm.upperArm.rotation = rotation * this.upperArmStartRotation; + return this.arm.upperArm.rotation = new Quaternion().multiplyQuaternions(rotation, this.upperArmStartRotation); } setLowerArmRotation(rotation) { - return this.arm.lowerArm.rotation = rotation * this.lowerArmStartRotation; + return this.arm.lowerArm.rotation = new Quaternion().multiplyQuaternions(rotation, this.lowerArmStartRotation); } setLowerArmLocalRotation(rotation) { - return this.arm.lowerArm.rotation = this.upperArmRotation * rotation * this.lowerArmStartRotation; + return this.arm.lowerArm.rotation = new Quaternion().multiplyQuaternions(new Quaternion().multiplyQuaternions(this.upperArmRotation, rotation), this.lowerArmStartRotation); } setWrist1Rotation(rotation) { - return this.arm.wrist1.rotation = rotation * this.wristStartRotation; + return this.arm.wrist1.rotation = new Quaternion().multiplyQuaternions(rotation, this.wristStartRotation); } setWrist2Rotation(rotation) { - return this.arm.wrist2.rotation = rotation * this.wristStartRotation; + return this.arm.wrist2.rotation = new Quaternion().multiplyQuaternions(rotation, this.wristStartRotation); } setWristLocalRotation(rotation) { - return this.arm.wrist1.rotation = this.arm.lowerArm.rotation * rotation * this.wristStartRotation; + return this.arm.wrist1.rotation = new Quaternion().multiplyQuaternions(new Quaternion().multiplyQuaternions(this.arm.lowerArm.rotation, rotation), this.wristStartRotation); } setHandRotation(rotation) { - return this.arm.hand.rotation = this.arm.hand.rotation = rotation * this.handStartRotation; + return this.arm.hand.rotation = /* this.arm.hand.rotation = */ new Quaternion().multiplyQuaternions(rotation, this.handStartRotation); } } From c6052a33f9917b2ab9643a2f4429bf2288a3b66e Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 23:50:11 -0400 Subject: [PATCH 079/562] Small spacing cleanup --- vrarmik/AvatarVRTrackingReferences.js | 8 ++------ 1 file changed, 2 insertions(+), 6 deletions(-) diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index e60936c..de5a758 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -77,12 +77,8 @@ import StaticOffsetTransform from './StaticOffsetTransform.js'; this.head.reference = this.head.reference !== null ? this.head.reference : PoseManager.Instance.vrTransforms.head; this.hmd.reference = this.hmd.reference !== null ? this.hmd.reference : PoseManager.Instance.vrTransforms.hmd; - this.leftHand.reference = this.leftHand.reference !== null - ? this.leftHand.reference - : PoseManager.Instance.vrTransforms.leftHand; - this.rightHand.reference = this.rightHand.reference !== null - ? this.rightHand.reference - : PoseManager.Instance.vrTransforms.rightHand; + this.leftHand.reference = this.leftHand.reference !== null ? this.leftHand.reference : PoseManager.Instance.vrTransforms.leftHand; + this.rightHand.reference = this.rightHand.reference !== null ? this.rightHand.reference : PoseManager.Instance.vrTransforms.rightHand; } } From 476877d57c40bf97676463b132de7dbeb7d1b43f Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 23:50:43 -0400 Subject: [PATCH 080/562] Dead code cleanup --- vrarmik/ShoulderPoser.js | 1 - 1 file changed, 1 deletion(-) diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index 827fe20..9031d57 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -1,7 +1,6 @@ import {Vector3, Quaternion, GameObject, MonoBehavior, Mathf} from './Unity.js'; import ShoulderTransforms from './ShoulderTransforms.js'; import VRTrackingReferences from './VRTrackingReferences.js'; -import AvatarVRTrackingReferences from './AvatarVRTrackingReferences.js'; import PoseManager from './PoseManager.js'; import VectorHelpers from './Utils/VectorHelpers.js'; From f470a7137d80a7df05bb64863bb4c86b40a238b4 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 23:52:15 -0400 Subject: [PATCH 081/562] Bugfix transforms propagation --- vrarmik/AvatarVRTrackingReferences.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/vrarmik/AvatarVRTrackingReferences.js b/vrarmik/AvatarVRTrackingReferences.js index de5a758..eea7fd6 100644 --- a/vrarmik/AvatarVRTrackingReferences.js +++ b/vrarmik/AvatarVRTrackingReferences.js @@ -43,7 +43,7 @@ import StaticOffsetTransform from './StaticOffsetTransform.js'; if (t === null) { t = new GameObject(name).AddComponent(StaticOffsetTransform); - t.transform.parent = this.transform; + this.transform.AddChild(t.transform); this.setStaticOffsetSettings(t); this[k] = t; } @@ -55,7 +55,7 @@ import StaticOffsetTransform from './StaticOffsetTransform.js'; { t = new GameObject(name).transform; t.transform.localPosition = Vector3.zero; - t.transform.parent = parent; + this.transform.AddChild(t.transform); } } */ From 5319ebf1dcd5bab52a7df9a53ad09896b4a0fc48 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 23:52:48 -0400 Subject: [PATCH 082/562] Bugfix unity shoulder/shoulder base methods ambiguity --- vrarmik/ShoulderPoser.js | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index 9031d57..6f7f1ee 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -80,8 +80,8 @@ class ShoulderPoser extends MonoBehavior { this.shoulder.transform.rotation = Quaternion.identity; this.positionShoulder(); - this.rotateShoulderUp(); - this.rotateShoulderRight(); + this.rotateShoulderUpBase(); + this.rotateShoulderRightBase(); if (this.enableDistinctShoulderRotation) { @@ -152,7 +152,7 @@ class ShoulderPoser extends MonoBehavior new Vector3().addVectors(this.shoulder.transform.parent.InverseTransformPoint(targetPosition), this.neckShoulderDistance); } - rotateShoulderUp() + rotateShoulderUpBase() { const angle = this.getCombinedDirectionAngleUp(); @@ -171,7 +171,7 @@ class ShoulderPoser extends MonoBehavior this.shoulder.transform.eulerAngles = targetRotation; } - rotateShoulderRight() + rotateShoulderRightBase() { const heightDiff = this.vrTrackingReferences.hmd.position.y - PoseManager.Instance.vrSystemOffsetHeight; From 86c5b57efd4849c1a640879c2547c3bd7b769985 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Mon, 14 Oct 2019 23:59:02 -0400 Subject: [PATCH 083/562] References bugfixing --- vrarmik/ShoulderPoser.js | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/vrarmik/ShoulderPoser.js b/vrarmik/ShoulderPoser.js index 6f7f1ee..a8dc7d5 100644 --- a/vrarmik/ShoulderPoser.js +++ b/vrarmik/ShoulderPoser.js @@ -119,7 +119,7 @@ class ShoulderPoser extends MonoBehavior initialShoulderLocalPos, angleSign) { const initialShoulderPos = this.shoulder.transform.TransformPoint(initialShoulderLocalPos); - const handShoulderOffset = new Vector3().subVectors(targetHand.position, this.initialShoulderPos); + const handShoulderOffset = new Vector3().subVectors(targetHand.position, initialShoulderPos); const armLength = arm.armLength; const targetAngle = Vector3.zero; @@ -137,8 +137,8 @@ class ShoulderPoser extends MonoBehavior -this.distinctShoulderRotationLimitBackward, 0); } - targetAngle.z = Mathf.Clamp(-(upwardDistanceRatio - 0.5) * distinctShoulderRotationMultiplier, - -distinctShoulderRotationLimitUpward, 0); + targetAngle.z = Mathf.Clamp(-(upwardDistanceRatio - 0.5) * this.distinctShoulderRotationMultiplier, + -this.distinctShoulderRotationLimitUpward, 0); shoulderSide.localEulerAngles = targetAngle * angleSign; } From 448eadd4f0e31a90c1368b7d11e5f33f70c72098 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Tue, 15 Oct 2019 00:00:35 -0400 Subject: [PATCH 084/562] Arm transforms debugging --- vrarmik/ArmTransforms.js | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/vrarmik/ArmTransforms.js b/vrarmik/ArmTransforms.js index c7891c7..5628e74 100644 --- a/vrarmik/ArmTransforms.js +++ b/vrarmik/ArmTransforms.js @@ -1,4 +1,4 @@ -import {Vector3, Transform, MonoBehavior, Mathf} from './Unity.js'; +import {Vector3, Quaternion, Transform, MonoBehavior, Mathf} from './Unity.js'; import PoseManager from './PoseManager.js'; class ArmTransforms extends MonoBehavior @@ -7,10 +7,19 @@ class ArmTransforms extends MonoBehavior super(...args); this.upperArm = new Transform(); + this.upperArm.localPosition = new Vector3(0, 0, 0); this.lowerArm = new Transform(); + this.lowerArm.localPosition = new Vector3(-0.3, 0, 0); this.wrist1 = new Transform(); this.wrist2 = new Transform(); this.hand = new Transform(); + this.hand.localPosition = new Vector3(-0.3, 0, 0); + + this.transform.AddChild(this.upperArm); + this.upperArm.AddChild(this.lowerArm); + // this.lowerArm.AddChild(this.wrist1); + // this.lowerArm.AddChild(this.wrist2); + this.lowerArm.AddChild(this.hand); this.armLengthByScale = false; this.scaleAxis = Vector3.one; From abadabed98261a9834eb055083f2a16e205c7c69 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Tue, 15 Oct 2019 00:06:15 -0400 Subject: [PATCH 085/562] Bugfix shoulder transforms --- vrarmik/ShoulderTransforms.js | 17 +++++++++++++++-- 1 file changed, 15 insertions(+), 2 deletions(-) diff --git a/vrarmik/ShoulderTransforms.js b/vrarmik/ShoulderTransforms.js index 06b5ff4..222dbd2 100644 --- a/vrarmik/ShoulderTransforms.js +++ b/vrarmik/ShoulderTransforms.js @@ -11,33 +11,46 @@ class ShoulderTransforms extends MonoBehavior super(...args); this.leftShoulder = new Transform(); + this.transform.AddChild(this.leftShoulder); this.rightShoulder = new Transform(); + this.transform.AddChild(this.rightShoulder); /* this.leftShoulderRenderer = new Transform(); this.rightShoulderRenderer = new Transform(); */ this.leftShoulderAnchor = new Transform(); + this.leftShoulderAnchor.localPosition = new Vector3(-0.3, 0, 0); + this.transform.AddChild(this.leftShoulderAnchor); this.rightShoulderAnchor = new Transform(); + this.rightShoulderAnchor.localPosition = new Vector3(0.3, 0, 0); + this.transform.AddChild(this.rightShoulderAnchor); this.leftArm = null; this.rightArm = null; } OnEnable() { + const shoulderPoser = this.GetComponent(ShoulderPoser); if (this.leftArm === null) { this.leftArm = new GameObject().AddComponent(ArmTransforms); const armIk = this.leftArm.GetComponentInChildren(VRArmIK); armIk.shoulder = this; - armIk.shoulderPoser = this.GetComponent(ShoulderPoser); + armIk.shoulderPoser = shoulderPoser; armIk.target = armIk.shoulderPoser.avatarTrackingReferences.leftHand.transform; + this.leftShoulderAnchor.AddChild(this.leftArm.transform); } if (this.rightArm === null) { this.rightArm = new GameObject().AddComponent(ArmTransforms); const armIk = this.rightArm.GetComponentInChildren(VRArmIK); armIk.shoulder = this; - armIk.shoulderPoser = this.GetComponent(ShoulderPoser); + armIk.shoulderPoser = shoulderPoser; armIk.target = armIk.shoulderPoser.avatarTrackingReferences.rightHand.transform; + + armIk.left = false; + this.rightArm.lowerArm.localPosition = new Vector3(0.3, 0, 0); + this.rightArm.hand.localPosition = new Vector3(0.3, 0, 0); + this.rightShoulderAnchor.AddChild(this.rightArm.transform); } } From a7efc076ffde313226c833d702d20110368119a8 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Tue, 15 Oct 2019 00:10:11 -0400 Subject: [PATCH 086/562] Major Unity port debugging --- vrarmik/Unity.js | 56 +++++++++++++++++++++++++++++++++++++----------- 1 file changed, 43 insertions(+), 13 deletions(-) diff --git a/vrarmik/Unity.js b/vrarmik/Unity.js index f0374d7..9530d44 100644 --- a/vrarmik/Unity.js +++ b/vrarmik/Unity.js @@ -1,5 +1,6 @@ const DEG2RAD = Math.PI/180; const RAD2DEG = 180/Math.PI; +const ORDER = 'ZXY'; class Vector2 extends THREE.Vector2 { get magnitude() { @@ -159,7 +160,7 @@ class Quaternion extends THREE.Quaternion { return new Quaternion().setFromUnitVectors(a, b); } static Euler(v) { - return new Quaternion().setFromEuler(new THREE.Euler(v.x, v.y, v.z, 'ZXY')); + return new Quaternion().setFromEuler(new THREE.Euler(v.x * DEG2RAD, v.y * DEG2RAD, v.z * DEG2RAD, ORDER)); } static Inverse(q) { return q.clone().inverse(); @@ -184,6 +185,7 @@ class Transform { this._localScale = new Vector3(1, 1, 1); this._localScale.bindOnchange(localChange); + this._children = []; this._parent = null; this._matrix = new THREE.Matrix4(); @@ -195,21 +197,27 @@ class Transform { return this._position.clone(); } set position(position) { + this.updateMatrixWorld(); this._position.copy(position); + this.updateLocalMatrix(); } get rotation() { this.updateMatrixWorld(); return this._rotation.clone(); } set rotation(rotation) { + this.updateMatrixWorld(); this._rotation.copy(rotation); + this.updateLocalMatrix(); } get scale() { this.updateMatrixWorld(); return this._scale.clone(); } set scale(scale) { + this.updateMatrixWorld(); this._scale.copy(scale); + this.updateLocalMatrix(); } get localPosition() { @@ -236,7 +244,7 @@ class Transform { } set parent(parent) { this._parent = parent; - this.matrixWorldNeedsUpdate = true; + this.localChange(); } get right() { @@ -249,9 +257,23 @@ class Transform { return this.TransformPoint(Vector3.forward); } + AddChild(child) { + this._children.push(child); + child.parent = this; + } + + updateLocalMatrix() { + this._matrixWorld.compose(this._position, this._rotation, this._scale); + this._matrix.copy(this._matrixWorld); + if (this._parent) { + this._matrix.premultiply(new THREE.Matrix4().getInverse(this._parent._matrixWorld)); + } + this._matrix.decompose(this._localPosition, this._localRotation, this._localScale); + this.matrixWorldNeedsUpdate = false; + } updateMatrixWorld() { if (this.matrixWorldNeedsUpdate) { - this._matrix.compose(this._position, this._rotation, this._scale); + this._matrix.compose(this._localPosition, this._localRotation, this._localScale); this._matrixWorld.copy(this._matrix); if (this._parent) { @@ -266,21 +288,24 @@ class Transform { } localChange() { this.matrixWorldNeedsUpdate = true; + for (let i = 0; i < this._children.length; i++) { + this._children[i].localChange(); + } } get eulerAngles() { - const e = new THREE.Euler().setFromQuaternion(this.rotation, 'ZXY'); - return new Vector3(e.x, e.y, e.z); + const e = new THREE.Euler().setFromQuaternion(this.rotation, ORDER); + return new Vector3(e.x * RAD2DEG, e.y * RAD2DEG, e.z * RAD2DEG); } set eulerAngles(v) { - this.rotation = new THREE.Quaternion().setFromEuler(new THREE.Euler(v.x, v.y, v.z, 'ZXY')); + this.rotation = new THREE.Quaternion().setFromEuler(new THREE.Euler(v.x * DEG2RAD, v.y * DEG2RAD, v.z * DEG2RAD, ORDER)); } get localEulerAngles() { - const e = new THREE.Euler().setFromQuaternion(this.localRotation, 'ZXY'); - return new Vector3(e.x, e.y, e.z); + const e = new THREE.Euler().setFromQuaternion(this.localRotation, ORDER); + return new Vector3(e.x * RAD2DEG, e.y * RAD2DEG, e.z * RAD2DEG); } set localEulerAngles(v) { - this.localRotation = new THREE.Quaternion().setFromEuler(new THREE.Euler(v.x, v.y, v.z, 'ZXY')); + this.localRotation = new THREE.Quaternion().setFromEuler(new THREE.Euler(v.x * DEG2RAD, v.y * DEG2RAD, v.z * DEG2RAD, ORDER)); } TransformPoint(v) { @@ -308,7 +333,6 @@ class GameObject { this.transform = new Transform(); this.components = new Map(); - this.children = []; gameObjects.push(this); } @@ -321,8 +345,7 @@ class GameObject { return component; } AddChild(child) { - this.children.push(child); - child.transform.parent = this.transform; + this.transform.AddChild(child.transform); } static startAll() { for (let i = 0; i < gameObjects.length; i++) { @@ -411,11 +434,18 @@ const Mathf = { Abs(v) { return Math.abs(v); }, + Log(a, b) { + let result = Math.log(a); + if (b !== undefined) { + result /= Math.log(b); + } + return result; + }, Lerp(a, b, t) { return a*(1-v) + b*v; }, LerpAngle(a, b, t) { - const num = Mathf.Repeat(b - a, 360); + let num = Mathf.Repeat(b - a, 360); if (num > 180) { num -= 360; } From b715562d66dd6390eee2dcf1735171d8162ffcdc Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Tue, 15 Oct 2019 00:19:25 -0400 Subject: [PATCH 087/562] VRArmIK debugging --- vrarmik/VRArmIK.js | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index f9b71a9..effb6dc 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -81,7 +81,7 @@ function toPositiveEulerAngle(n) constructor(...args) { super(...args); - this.arm = new GameObject().AddComponent(ArmTransforms); + this.arm = this.GetOrAddComponent(ArmTransforms); this.shoulder = null; this.shoulderPoser = null; this.target = new Transform(); @@ -126,6 +126,7 @@ function toPositiveEulerAngle(n) this.calcElbowInnerAngle(); this.rotateShoulder(); this.correctElbowRotation(); + if (this.elbowSettings.calcElbowAngle) { this.positionElbow(); @@ -149,7 +150,7 @@ function toPositiveEulerAngle(n) updateUpperArmPosition() { - //arm.upperArm.position = shoulderAnker.transform.position; + //this.arm.upperArm.position = this.shoulderAnker.position; } calcElbowInnerAngle() From f6f00913b66658db54ebbe43896d408cdb967290 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Tue, 15 Oct 2019 00:20:19 -0400 Subject: [PATCH 088/562] More VRArmIK debugging --- vrarmik/VRArmIK.js | 37 +++++++++++++++++++------------------ 1 file changed, 19 insertions(+), 18 deletions(-) diff --git a/vrarmik/VRArmIK.js b/vrarmik/VRArmIK.js index effb6dc..2eecd6e 100644 --- a/vrarmik/VRArmIK.js +++ b/vrarmik/VRArmIK.js @@ -184,20 +184,21 @@ function toPositiveEulerAngle(n) //source: https://github.com/NickHardeman/ofxIKArm/blob/master/src/ofxIKArm.cpp rotateShoulder() { - const eulerAngles = new Vector3(); - const targetShoulderDirection = new Vector3().subVectors(this.target.position, this.upperArmPos).normalized; - const targetShoulderDistance = new Vector3().subVectors(this.target.position, this.upperArmPos).magnitude; + const eulerAngles = new Vector3(); + const targetShoulderDirection = new Vector3().subVectors(this.target.position, this.upperArmPos).normalized; + const targetShoulderDistance = new Vector3().subVectors(this.target.position, this.upperArmPos).magnitude; + + eulerAngles.y = (this.left ? -1 : 1) * + Mathf.Acos(Mathf.Clamp((Mathf.Pow(targetShoulderDistance, 2) + Mathf.Pow(this.arm.upperArmLength, 2) - + Mathf.Pow(this.arm.lowerArmLength, 2)) / (2 * targetShoulderDistance * this.arm.upperArmLength), -1, 1)) * Mathf.Rad2Deg; + if (isNaN(eulerAngles.y)) + eulerAngles.y = 0; - eulerAngles.y = (this.left ? -1 : 1) * - Mathf.Acos(Mathf.Clamp((Mathf.Pow(targetShoulderDistance, 2) + Mathf.Pow(this.arm.upperArmLength, 2) - - Mathf.Pow(this.arm.lowerArmLength, 2)) / (2 * targetShoulderDistance * this.arm.upperArmLength), -1, 1)) * Mathf.Rad2Deg; - if (isNaN(eulerAngles.y)) - eulerAngles.y = 0; - const shoulderRightRotation = Quaternion.FromToRotation(this.armDirection, targetShoulderDirection); - this.setUpperArmRotation(shoulderRightRotation); - this.arm.upperArm.rotation = new Quaternion().multiplyQuaternions(Quaternion.AngleAxis(eulerAngles.y, new Vector3().multiplyVectors(this.lowerArmRotation, Vector3.up)), this.arm.upperArm.rotation); - this.setLowerArmLocalRotation(Quaternion.Euler(this.nextLowerArmAngle)); + const shoulderRightRotation = Quaternion.FromToRotation(this.armDirection, targetShoulderDirection); + this.setUpperArmRotation(shoulderRightRotation); + this.arm.upperArm.rotation = new Quaternion().multiplyQuaternions(Quaternion.AngleAxis(eulerAngles.y, Vector3.up.applyQuaternion(this.lowerArmRotation)), this.arm.upperArm.rotation); + this.setLowerArmLocalRotation(Quaternion.Euler(this.nextLowerArmAngle)); } getElbowTargetAngle() @@ -255,13 +256,13 @@ function toPositiveEulerAngle(n) ) * s.weight; const shoulderHandDirection = new Vector3().subVectors(this.upperArmPos, this.handPos).normalized; - const targetDir = new Vector3().addVectors(Vector3.up, (s.correctElbowOutside ? new Vector3().addVectors(this.armDirection, Vector3.forward.multiplyScalar(-.2)) * elbowOutsideFactor : Vector3.zero)).applyQuaternion(this.shoulder.transform.rotation); - const cross = Vector3.Cross(shoulderHandDirection, targetDir * 1000); + const targetDir = new Vector3().addVectors(Vector3.up, (s.correctElbowOutside ? new Vector3().addVectors(this.armDirection, Vector3.forward.multiplyScalar(-.2)).multiplyScalar(elbowOutsideFactor) : Vector3.zero)).applyQuaternion(this.shoulder.transform.rotation); + const cross = Vector3.Cross(shoulderHandDirection, targetDir.clone().multiplyScalar(1000)); - const upperArmUp = new Vector3().multiplyVectors(this.upperArmRotation, Vector3.up); + const upperArmUp = Vector3.up.applyQuaternion(this.upperArmRotation); const elbowTargetUp = Vector3.Dot(upperArmUp, targetDir); - const elbowAngle = Vector3.Angle(cross, upperArmUp) + (this.left ? 0 : 180); + const elbowAngle = (cross.equals(Vector3.zero) ? 0 : Vector3.Angle(cross, upperArmUp)) + (this.left ? 0 : 180); const rotation = Quaternion.AngleAxis(elbowAngle * Mathf.Sign(elbowTargetUp), shoulderHandDirection); this.arm.upperArm.rotation = new Quaternion().multiplyQuaternions(rotation, this.arm.upperArm.rotation); } @@ -282,7 +283,7 @@ function toPositiveEulerAngle(n) const targetDir = elbowPos.normalized.applyQuaternion(this.shoulder.transform.rotation); const cross = Vector3.Cross(shoulderHandDirection, targetDir); - const upperArmUp = new Vector3().multiplyVectors(this.upperArmRotation, Vector3.up); + const upperArmUp = Vector3.up.applyQuaternion(this.upperArmRotation); let distance = new Vector3().subVectors(this.target.position, this.upperArmPos); @@ -323,7 +324,7 @@ function toPositiveEulerAngle(n) Vector3.up.applyQuaternion(this.lowerArmRotation), Vector3.forward.applyQuaternion(this.lowerArmRotation)); // todo reduce influence if hand local forward rotation is high (hand tilted inside) - const handForwardRotation = Quaternion.AngleAxis(-forwardAngle, this.lowerArmRotation * Vector3.forward); + const handForwardRotation = Quaternion.AngleAxis(-forwardAngle, Vector3.forward.applyQuaternion(this.lowerArmRotation)); handUpVec = handUpVec.applyQuaternion(handForwardRotation); const elbowTargetAngle = VectorHelpers.getAngleBetween(Vector3.up.applyQuaternion(this.lowerArmRotation), handUpVec, From 0eb076eda6fd4208ca12ec4a8393ee09cb7ffd92 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Tue, 15 Oct 2019 00:23:30 -0400 Subject: [PATCH 089/562] Add vrarmik test HTML --- vrarmik.html | 176 +++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 176 insertions(+) create mode 100644 vrarmik.html diff --git a/vrarmik.html b/vrarmik.html new file mode 100644 index 0000000..37576e4 --- /dev/null +++ b/vrarmik.html @@ -0,0 +1,176 @@ + + + + + + + + + From 45cfe146bc0c4ae5e108d76c180f17fdb6b78e15 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Tue, 15 Oct 2019 01:16:43 -0400 Subject: [PATCH 090/562] Extract rigging to VRArmIK --- vrarmik.html | 113 ++++++++++++++++--------------------------------- vrarmik/Rig.js | 57 +++++++++++++++++++++++++ 2 files changed, 93 insertions(+), 77 deletions(-) create mode 100644 vrarmik/Rig.js diff --git a/vrarmik.html b/vrarmik.html index 37576e4..50b4340 100644 --- a/vrarmik.html +++ b/vrarmik.html @@ -9,22 +9,7 @@ + --> - - From 52b46e589b138c9860b929b0722350ed6243ed85 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 18 Oct 2019 23:06:39 -0400 Subject: [PATCH 139/562] Add Reflector.js --- Reflector.js | 271 +++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 271 insertions(+) create mode 100644 Reflector.js diff --git a/Reflector.js b/Reflector.js new file mode 100644 index 0000000..3f9be10 --- /dev/null +++ b/Reflector.js @@ -0,0 +1,271 @@ +/** + * @author Slayvin / http://slayvin.net + */ + +THREE.Reflector = function ( geometry, options ) { + + THREE.Mesh.call( this, geometry ); + + this.type = 'Reflector'; + + var scope = this; + + options = options || {}; + + var color = ( options.color !== undefined ) ? new THREE.Color( options.color ) : new THREE.Color( 0x7F7F7F ); + var addColor = ( options.addColor !== undefined ) ? new THREE.Color( options.addColor ) : new THREE.Color( 0x000000 ); + var textureWidth = options.textureWidth || 512; + var textureHeight = options.textureHeight || 512; + var clipBias = options.clipBias || 0; + var shader = options.shader || THREE.Reflector.ReflectorShader; + var recursion = options.recursion !== undefined ? options.recursion : 0; + + // + + var reflectorPlane = new THREE.Plane(); + var normal = new THREE.Vector3(); + var reflectorWorldPosition = new THREE.Vector3(); + var cameraWorldPosition = new THREE.Vector3(); + var rotationMatrix = new THREE.Matrix4(); + var lookAtPosition = new THREE.Vector3( 0, 0, - 1 ); + var clipPlane = new THREE.Vector4(); + var viewport = new THREE.Vector4(); + + var view = new THREE.Vector3(); + var target = new THREE.Vector3(); + var q = new THREE.Vector4(); + + var textureMatrix = new THREE.Matrix4(); + var virtualCamera = new THREE.PerspectiveCamera(); + + var parameters = { + minFilter: THREE.LinearFilter, + magFilter: THREE.LinearFilter, + format: THREE.RGBFormat, + stencilBuffer: false + }; + + var renderTarget = new THREE.WebGLRenderTarget( textureWidth, textureHeight, parameters ); + + if ( ! THREE.Math.isPowerOfTwo( textureWidth ) || ! THREE.Math.isPowerOfTwo( textureHeight ) ) { + + renderTarget.texture.generateMipmaps = false; + + } + + var material = new THREE.ShaderMaterial( { + uniforms: THREE.UniformsUtils.clone( shader.uniforms ), + fragmentShader: shader.fragmentShader, + vertexShader: shader.vertexShader + } ); + + material.uniforms.tDiffuse.value = renderTarget.texture; + material.uniforms.color.value = color; + material.uniforms.addColor.value = addColor; + material.uniforms.textureMatrix.value = textureMatrix; + + this.material = material; + this.renderOrder = - Infinity; // render first + + this.onBeforeRender = function ( renderer, scene, camera ) { + + if ( 'recursion' in camera.userData ) { + + if ( camera.userData.recursion === recursion ) return; + + camera.userData.recursion ++; + + } + + reflectorWorldPosition.setFromMatrixPosition( scope.matrixWorld ); + cameraWorldPosition.setFromMatrixPosition( camera.matrixWorld ); + + rotationMatrix.extractRotation( scope.matrixWorld ); + + normal.set( 0, 0, -1 ); + normal.applyMatrix4( rotationMatrix ); + + view.subVectors( reflectorWorldPosition, cameraWorldPosition ); + + // Avoid rendering when reflector is facing away + + if ( view.dot( normal ) < 0 ) return; + + view.copy(cameraWorldPosition); + + rotationMatrix.extractRotation( camera.matrixWorld ); + + lookAtPosition.set( 0, 0, - 1 ); + lookAtPosition.applyMatrix4( rotationMatrix ); + lookAtPosition.add( cameraWorldPosition ); + + target.subVectors( reflectorWorldPosition, lookAtPosition ); + target.reflect( normal ).negate(); + target.add( reflectorWorldPosition ); + + virtualCamera.position.copy( view ); + virtualCamera.up.set( 0, 1, 0 ); + virtualCamera.up.applyMatrix4( rotationMatrix ); + virtualCamera.up.reflect( normal ); + virtualCamera.lookAt( target ); + + virtualCamera.far = camera.far; // Used in WebGLBackground + + virtualCamera.updateMatrixWorld(); + virtualCamera.projectionMatrix.copy( camera.projectionMatrix ); + + virtualCamera.userData.recursion = 0; + + // Update the texture matrix + textureMatrix.set( + 0.5, 0.0, 0.0, 0.5, + 0.0, 0.5, 0.0, 0.5, + 0.0, 0.0, 0.5, 0.5, + 0.0, 0.0, 0.0, 1.0 + ); + textureMatrix.multiply( virtualCamera.projectionMatrix ); + textureMatrix.multiply( virtualCamera.matrixWorldInverse ); + textureMatrix.multiply( scope.matrixWorld ); + + // Now update projection matrix with new clip plane, implementing code from: http://www.terathon.com/code/oblique.html + // Paper explaining this technique: http://www.terathon.com/lengyel/Lengyel-Oblique.pdf + reflectorPlane.setFromNormalAndCoplanarPoint( normal, reflectorWorldPosition ); + reflectorPlane.applyMatrix4( virtualCamera.matrixWorldInverse ); + + clipPlane.set( reflectorPlane.normal.x, reflectorPlane.normal.y, reflectorPlane.normal.z, reflectorPlane.constant ); + + var projectionMatrix = virtualCamera.projectionMatrix; + + q.x = ( Math.sign( clipPlane.x ) + projectionMatrix.elements[ 8 ] ) / projectionMatrix.elements[ 0 ]; + q.y = ( Math.sign( clipPlane.y ) + projectionMatrix.elements[ 9 ] ) / projectionMatrix.elements[ 5 ]; + q.z = - 1.0; + q.w = ( 1.0 + projectionMatrix.elements[ 10 ] ) / projectionMatrix.elements[ 14 ]; + + // Calculate the scaled plane vector + clipPlane.multiplyScalar( 2.0 / clipPlane.dot( q ) ); + + // Replacing the third row of the projection matrix + projectionMatrix.elements[ 2 ] = clipPlane.x; + projectionMatrix.elements[ 6 ] = clipPlane.y; + projectionMatrix.elements[ 10 ] = clipPlane.z + 1.0 - clipBias; + projectionMatrix.elements[ 14 ] = clipPlane.w; + + // Render + + scope.visible = false; + + var currentRenderTarget = renderer.getRenderTarget(); + + var currentVrEnabled = renderer.vr.enabled; + var currentShadowAutoUpdate = renderer.shadowMap.autoUpdate; + + renderer.vr.enabled = false; // Avoid camera modification and recursion + renderer.shadowMap.autoUpdate = false; // Avoid re-computing shadows + + renderer.render( scene, virtualCamera, renderTarget, true ); + + renderer.vr.enabled = currentVrEnabled; + renderer.shadowMap.autoUpdate = currentShadowAutoUpdate; + + renderer.setRenderTarget( currentRenderTarget ); + + // Restore viewport + + var bounds = camera.bounds; + + if ( bounds !== undefined ) { + + var size = renderer.getSize(); + var pixelRatio = renderer.getPixelRatio(); + + viewport.x = bounds.x * size.width * pixelRatio; + viewport.y = bounds.y * size.height * pixelRatio; + viewport.z = bounds.z * size.width * pixelRatio; + viewport.w = bounds.w * size.height * pixelRatio; + + renderer.state.viewport( viewport ); + + } + + scope.visible = true; + + }; + + this.getRenderTarget = function () { + + return renderTarget; + + }; + +}; + +THREE.Reflector.prototype = Object.create( THREE.Mesh.prototype ); +THREE.Reflector.prototype.constructor = THREE.Reflector; + +THREE.Reflector.ReflectorShader = { + + uniforms: { + + 'color': { + type: 'c', + value: null + }, + + 'addColor': { + type: 'c', + value: null + }, + + 'tDiffuse': { + type: 't', + value: null + }, + + 'textureMatrix': { + type: 'm4', + value: null + } + + }, + + vertexShader: [ + 'uniform mat4 textureMatrix;', + 'varying vec4 vUv;', + + 'void main() {', + + ' vUv = textureMatrix * vec4( position, 1.0 );', + + ' gl_Position = projectionMatrix * modelViewMatrix * vec4( position, 1.0 );', + + '}' + ].join( '\n' ), + + fragmentShader: [ + 'uniform vec3 color;', + 'uniform vec3 addColor;', + 'uniform sampler2D tDiffuse;', + 'varying vec4 vUv;', + + 'float blendOverlay( float base, float blend ) {', + + ' return( base < 0.5 ? ( 2.0 * base * blend ) : ( 1.0 - 2.0 * ( 1.0 - base ) * ( 1.0 - blend ) ) );', + + '}', + + 'vec3 blendOverlay( vec3 base, vec3 blend ) {', + + ' return vec3( blendOverlay( base.r, blend.r ), blendOverlay( base.g, blend.g ), blendOverlay( base.b, blend.b ) );', + + '}', + + 'void main() {', + + ' vec4 base = texture2DProj( tDiffuse, vUv );', + ' vec3 addColorC;', + ' if (base.r > 0.01 || base.g > 0.01 || base.b > 0.01) {addColorC = addColor;} else {addColorC = vec3(0.0);}', + ' gl_FragColor = vec4( blendOverlay( base.rgb, color ) + addColorC, 1.0 );', + + '}' + ].join( '\n' ) +}; From c189f3d5bf9959ca4c23bea45e072dd4b08285ed Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 18 Oct 2019 23:06:50 -0400 Subject: [PATCH 140/562] Add initial mirror mesh --- app.html | 23 +++++++++++++++++++++++ 1 file changed, 23 insertions(+) diff --git a/app.html b/app.html index 3eb303c..0ba7ecf 100644 --- a/app.html +++ b/app.html @@ -9,6 +9,7 @@ + @@ -6958,6 +6959,28 @@ }; _loadDefaultGltf(); +const mirrorMesh = (() => { + const geometry = new THREE.PlaneBufferGeometry(1, 1); + const mesh = new THREE.Reflector(geometry, { + clipBias: 0.003, + textureWidth: 1024 * window.devicePixelRatio, + textureHeight: 1024 * window.devicePixelRatio, + color: 0x889999, + addColor: 0x300000, + recursion: 1 + }); + // mesh.position.set(-1, 1.5, -2.1); + mesh.position.set(0, 1, 1); + /* mesh.rotation.order = 'YXZ'; + mesh.rotation.y = Math.PI; */ + /* const material = new THREE.MeshBasicMaterial({ + color: 0xFF0000, + }); + const mesh = new THREE.Mesh(geometry, material); */ + return mesh; +})(); +scene.add(mirrorMesh); + function _updateCoord() { const {c} = parseQuery(window.location.search); let match; From 430c054e09e56a17f22e6d712cf28b1fef7f06b3 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 18 Oct 2019 23:24:55 -0400 Subject: [PATCH 141/562] Update Reflector.js --- Reflector.js | 46 +++++++++++++--------------------------------- 1 file changed, 13 insertions(+), 33 deletions(-) diff --git a/Reflector.js b/Reflector.js index 3f9be10..47fdc2b 100644 --- a/Reflector.js +++ b/Reflector.js @@ -13,7 +13,6 @@ THREE.Reflector = function ( geometry, options ) { options = options || {}; var color = ( options.color !== undefined ) ? new THREE.Color( options.color ) : new THREE.Color( 0x7F7F7F ); - var addColor = ( options.addColor !== undefined ) ? new THREE.Color( options.addColor ) : new THREE.Color( 0x000000 ); var textureWidth = options.textureWidth || 512; var textureHeight = options.textureHeight || 512; var clipBias = options.clipBias || 0; @@ -29,7 +28,6 @@ THREE.Reflector = function ( geometry, options ) { var rotationMatrix = new THREE.Matrix4(); var lookAtPosition = new THREE.Vector3( 0, 0, - 1 ); var clipPlane = new THREE.Vector4(); - var viewport = new THREE.Vector4(); var view = new THREE.Vector3(); var target = new THREE.Vector3(); @@ -59,13 +57,11 @@ THREE.Reflector = function ( geometry, options ) { vertexShader: shader.vertexShader } ); - material.uniforms.tDiffuse.value = renderTarget.texture; - material.uniforms.color.value = color; - material.uniforms.addColor.value = addColor; - material.uniforms.textureMatrix.value = textureMatrix; + material.uniforms[ "tDiffuse" ].value = renderTarget.texture; + material.uniforms[ "color" ].value = color; + material.uniforms[ "textureMatrix" ].value = textureMatrix; this.material = material; - this.renderOrder = - Infinity; // render first this.onBeforeRender = function ( renderer, scene, camera ) { @@ -82,16 +78,17 @@ THREE.Reflector = function ( geometry, options ) { rotationMatrix.extractRotation( scope.matrixWorld ); - normal.set( 0, 0, -1 ); + normal.set( 0, 0, 1 ); normal.applyMatrix4( rotationMatrix ); view.subVectors( reflectorWorldPosition, cameraWorldPosition ); // Avoid rendering when reflector is facing away - if ( view.dot( normal ) < 0 ) return; + if ( view.dot( normal ) > 0 ) return; - view.copy(cameraWorldPosition); + view.reflect( normal ).negate(); + view.add( reflectorWorldPosition ); rotationMatrix.extractRotation( camera.matrixWorld ); @@ -162,7 +159,9 @@ THREE.Reflector = function ( geometry, options ) { renderer.vr.enabled = false; // Avoid camera modification and recursion renderer.shadowMap.autoUpdate = false; // Avoid re-computing shadows - renderer.render( scene, virtualCamera, renderTarget, true ); + renderer.setRenderTarget( renderTarget ); + renderer.clear(); + renderer.render( scene, virtualCamera ); renderer.vr.enabled = currentVrEnabled; renderer.shadowMap.autoUpdate = currentShadowAutoUpdate; @@ -171,17 +170,9 @@ THREE.Reflector = function ( geometry, options ) { // Restore viewport - var bounds = camera.bounds; + var viewport = camera.viewport; - if ( bounds !== undefined ) { - - var size = renderer.getSize(); - var pixelRatio = renderer.getPixelRatio(); - - viewport.x = bounds.x * size.width * pixelRatio; - viewport.y = bounds.y * size.height * pixelRatio; - viewport.z = bounds.z * size.width * pixelRatio; - viewport.w = bounds.w * size.height * pixelRatio; + if ( viewport !== undefined ) { renderer.state.viewport( viewport ); @@ -207,22 +198,14 @@ THREE.Reflector.ReflectorShader = { uniforms: { 'color': { - type: 'c', - value: null - }, - - 'addColor': { - type: 'c', value: null }, 'tDiffuse': { - type: 't', value: null }, 'textureMatrix': { - type: 'm4', value: null } @@ -243,7 +226,6 @@ THREE.Reflector.ReflectorShader = { fragmentShader: [ 'uniform vec3 color;', - 'uniform vec3 addColor;', 'uniform sampler2D tDiffuse;', 'varying vec4 vUv;', @@ -262,9 +244,7 @@ THREE.Reflector.ReflectorShader = { 'void main() {', ' vec4 base = texture2DProj( tDiffuse, vUv );', - ' vec3 addColorC;', - ' if (base.r > 0.01 || base.g > 0.01 || base.b > 0.01) {addColorC = addColor;} else {addColorC = vec3(0.0);}', - ' gl_FragColor = vec4( blendOverlay( base.rgb, color ) + addColorC, 1.0 );', + ' gl_FragColor = vec4( blendOverlay( base.rgb, color ), 1.0 );', '}' ].join( '\n' ) From 6b359fb0c8373faa736f9bc2c8762256a48ba096 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 18 Oct 2019 23:27:45 -0400 Subject: [PATCH 142/562] Updte reflector size --- app.html | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/app.html b/app.html index 0ba7ecf..6d7cd88 100644 --- a/app.html +++ b/app.html @@ -6960,11 +6960,11 @@ _loadDefaultGltf(); const mirrorMesh = (() => { - const geometry = new THREE.PlaneBufferGeometry(1, 1); + const geometry = new THREE.PlaneBufferGeometry(1, 2); const mesh = new THREE.Reflector(geometry, { clipBias: 0.003, textureWidth: 1024 * window.devicePixelRatio, - textureHeight: 1024 * window.devicePixelRatio, + textureHeight: 2048 * window.devicePixelRatio, color: 0x889999, addColor: 0x300000, recursion: 1 From baf5ad92e99ad7d372b16f1a606c8137eabfb043 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Fri, 18 Oct 2019 23:31:24 -0400 Subject: [PATCH 143/562] Dead code cleanup --- vrarmik.html | 25 +------------------------ 1 file changed, 1 insertion(+), 24 deletions(-) diff --git a/vrarmik.html b/vrarmik.html index 44f39d8..33de864 100644 --- a/vrarmik.html +++ b/vrarmik.html @@ -160,26 +160,14 @@ window.bones = bones; o.bind(window.skeleton); - // window.bindMatrix = bindMatrix.clone(); } }); - fixSkeletonZForward(window.bones[0], { - /* preRotations: { - 'Left_arm': new Quaternion().setFromAxisAngle(new Vector3(0, 0, 1), Math.PI*0.25).inverse(), - 'Right_arm': new Quaternion().setFromAxisAngle(new Vector3(0, 0, 1), -Math.PI*0.25).inverse(), - }, */ - }); + fixSkeletonZForward(window.bones[0]); model.traverse(o => { if (o.isSkinnedMesh) { o.bind(window.skeleton); } }); - /* ['Left_arm', 'Right_arm'].forEach((name, i) => { - const bone = bones.find(bone => bone.name === name); - if (bone) { - bone.quaternion.premultiply(new Quaternion().setFromAxisAngle(new Vector3(0, 0, 1), (i === 0 ? 1 : -1) * Math.PI*0.25)); - } - }); */ model.updateMatrixWorld(true); model.traverse(o => { @@ -189,15 +177,9 @@ if (o.isSkinnedMesh) { const {skeleton, bindMatrix} = o; const {bones} = skeleton; - /* fixSkeletonZForward(bones[0], { - exclude: ['Hips', 'Head'], - }); - o.bind(skeleton, bindMatrix); */ - // o.updateMatrixWorld(true); for (const k in modelBones) { if (!modelBones[k]) { modelBones[k] = bones.find(bone => bone.name === k); - // modelBones[k].initialDirectionQuaternion = _getChildDirectionQuaternion(modelBones[k]); modelBones[k].initialPosition = modelBones[k].position.clone(); modelBones[k].initialQuaternion = modelBones[k].quaternion.clone(); console.log('found bone', k, modelBones[k], modelBones[k].children, new THREE.Euler().setFromQuaternion(modelBones[k].initialQuaternion, 'YXZ'), 'YXZ'); @@ -205,11 +187,6 @@ modelBones[k].initialMatrixWorld = modelBones[k].matrixWorld.clone(); } } - /* for (const k in modelBones) { - if (modelBones[k] && modelBones[k].parent) { - modelBones[k].parent.remove(modelBones[k]); - } - } */ } }); From c60a7a86134cac8acfbdd1ec5bf9444dcb93b426 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sat, 19 Oct 2019 00:21:17 -0400 Subject: [PATCH 144/562] Add missing neck bone binding --- vrarmik/Rig.js | 1 + 1 file changed, 1 insertion(+) diff --git a/vrarmik/Rig.js b/vrarmik/Rig.js index 9509451..b278bbc 100644 --- a/vrarmik/Rig.js +++ b/vrarmik/Rig.js @@ -49,6 +49,7 @@ class Rig { hips: this.legsManager.hips, spine: this.shoulderTransforms.spine, chest: this.shoulderTransforms.transform, + neck: this.shoulderTransforms.neck, leftShoulder: this.shoulderTransforms.leftShoulderAnchor, leftUpperArm: this.shoulderTransforms.leftArm.upperArm, leftLowerArm: this.shoulderTransforms.leftArm.lowerArm, From 274d9dab1274a5dd330334977c2b7d3ab474af95 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sat, 19 Oct 2019 00:24:06 -0400 Subject: [PATCH 145/562] Major API cleanup --- vrarmik.html | 247 ++++++------------------------------------------- vrarmik/Rig.js | 214 +++++++++++++++++++++++++++++++++++++++++- 2 files changed, 235 insertions(+), 226 deletions(-) diff --git a/vrarmik.html b/vrarmik.html index 33de864..2bb67bd 100644 --- a/vrarmik.html +++ b/vrarmik.html @@ -96,7 +96,7 @@ scene.remove(meshes.lowerLegRight); scene.remove(meshes.footRight); */ -let rig, model, modelBones, modelBonesMappings; +let rig, model; const _getChildDirectionQuaternion = o => { // const worldPosition = o.getWorldPosition(new Vector3()); const vector = new Vector3(0, 0, 0); @@ -125,124 +125,37 @@ } }; new THREE.GLTFLoader().load('model4.glb', object => { - modelBones = { - Hips: null, - Spine: null, - Chest: null, - Neck: null, - Head: null, - - Left_shoulder: null, - Left_arm: null, - Left_elbow: null, - Left_wrist: null, - Left_leg: null, - Left_knee: null, - Left_ankle: null, - - Right_shoulder: null, - Right_arm: null, - Right_elbow: null, - Right_wrist: null, - Right_leg: null, - Right_knee: null, - Right_ankle: null, - }; - model = object.scene; - model.updateMatrixWorld(true); - model.traverse(o => { - if (o.isSkinnedMesh) { - const {skeleton, bindMatrix} = o; - const {bones} = skeleton; - window.skeleton = skeleton; - window.bones = bones; - - o.bind(window.skeleton); - } - }); - fixSkeletonZForward(window.bones[0]); - model.traverse(o => { - if (o.isSkinnedMesh) { - o.bind(window.skeleton); - } - }); - model.updateMatrixWorld(true); - - model.traverse(o => { - if (o.isMesh) { - o.frustumCulled = false; - } - if (o.isSkinnedMesh) { - const {skeleton, bindMatrix} = o; - const {bones} = skeleton; - for (const k in modelBones) { - if (!modelBones[k]) { - modelBones[k] = bones.find(bone => bone.name === k); - modelBones[k].initialPosition = modelBones[k].position.clone(); - modelBones[k].initialQuaternion = modelBones[k].quaternion.clone(); - console.log('found bone', k, modelBones[k], modelBones[k].children, new THREE.Euler().setFromQuaternion(modelBones[k].initialQuaternion, 'YXZ'), 'YXZ'); - modelBones[k].initialScale = modelBones[k].scale.clone(); - modelBones[k].initialMatrixWorld = modelBones[k].matrixWorld.clone(); - } - } - } + rig = new Rig(model, { + boneMappings: { + Hips: 'Hips', + Spine: 'Spine', + Chest: 'Chest', + Neck: 'Neck', + Head: 'Head', + + Left_shoulder: 'Left_shoulder', + Left_arm: 'Left_arm', + Left_elbow: 'Left_elbow', + Left_wrist: 'Left_wrist', + Left_leg: 'Left_leg', + Left_knee: 'Left_knee', + Left_ankle: 'Left_ankle', + + Right_shoulder: 'Right_shoulder', + Right_arm: 'Right_arm', + Right_elbow: 'Right_elbow', + Right_wrist: 'Right_wrist', + Right_leg: 'Right_leg', + Right_knee: 'Right_knee', + Right_ankle: 'Right_ankle', + }, }); - - const _getOffset = (bone, parent = bone.parent) => bone.getWorldPosition(new Vector3()).sub(parent.getWorldPosition(new Vector3())); - const setups = { - spine: _getOffset(modelBones.Spine), - hips: _getOffset(modelBones.Spine, modelBones.Head), - neck: _getOffset(modelBones.Neck), - head: _getOffset(modelBones.Head), - - leftShoulder: _getOffset(modelBones.Right_shoulder), - leftUpperArm: _getOffset(modelBones.Right_arm), - leftLowerArm: _getOffset(modelBones.Right_elbow), - leftHand: _getOffset(modelBones.Right_wrist), - - rightShoulder: _getOffset(modelBones.Left_shoulder), - rightUpperArm: _getOffset(modelBones.Left_arm), - rightLowerArm: _getOffset(modelBones.Left_elbow), - rightHand: _getOffset(modelBones.Left_wrist), - - leftUpperLeg: _getOffset(modelBones.Right_leg), - leftLowerLeg: _getOffset(modelBones.Right_knee), - leftFoot: _getOffset(modelBones.Right_ankle), - - rightUpperLeg: _getOffset(modelBones.Left_leg), - rightLowerLeg: _getOffset(modelBones.Left_knee), - rightFoot: _getOffset(modelBones.Left_ankle), - }; - rig = new Rig(setups); /* for (const k in setups) { setups[k] = setups[k].toArray(); } console.log(JSON.stringify(setups, null, 2)); */ - modelBonesMappings = { - Hips: rig.outputs.hips, - Spine: rig.outputs.spine, - Chest: rig.outputs.chest, - Neck: rig.outputs.neck, - Head: rig.outputs.hmd, - - Left_shoulder: rig.outputs.rightShoulder, - Left_arm: rig.outputs.rightUpperArm, - Left_elbow: rig.outputs.rightLowerArm, - Left_wrist: rig.outputs.rightHand, - Left_leg: rig.outputs.rightUpperLeg, - Left_knee: rig.outputs.rightLowerLeg, - Left_ankle: rig.outputs.rightFoot, - - Right_shoulder: rig.outputs.leftShoulder, - Right_arm: rig.outputs.leftUpperArm, - Right_elbow: rig.outputs.leftLowerArm, - Right_wrist: rig.outputs.leftHand, - Right_leg: rig.outputs.leftUpperLeg, - Right_knee: rig.outputs.leftLowerLeg, - Right_ankle: rig.outputs.leftFoot, - }; // debugger; scene.add(model); @@ -338,115 +251,7 @@ meshes.lowerLegRight.quaternion.copy(rig.outputs.rightLowerLeg.rotation); meshes.footRight.position.copy(rig.outputs.rightFoot.position); meshes.footRight.quaternion.copy(rig.outputs.rightFoot.rotation); - - for (const k in modelBones) { - const modelBone = modelBones[k]; - const modelBoneMapping = modelBonesMappings[k]; - /* if (modelBone) { - if (['Chest', 'Left_arm', 'Left_elbow', 'Left_wrist'].includes(k)) { - if (!modelBone.axesHelper) { - modelBone.axesHelper = new THREE.AxesHelper(); - scene.add(modelBone.axesHelper); - } - modelBone.getWorldPosition(modelBone.axesHelper.position); - modelBone.getWorldQuaternion(modelBone.axesHelper.quaternion); - modelBone.getWorldScale(modelBone.axesHelper.scale); - } - } */ - if (modelBone && modelBoneMapping) { - if (k === 'Hips') { - modelBone.position.copy(modelBoneMapping.position); - } - modelBone.quaternion - .copy(modelBone.initialQuaternion) - - if (['Hips', 'Spine', 'Chest', 'Neck', 'Head'].includes(k)) { - modelBone.quaternion - .multiply(modelBoneMapping.localRotation) - } - - if (['Left_leg'].includes(k)) { - modelBone.quaternion - .multiply(modelBoneMapping.localRotation) - // .multiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), -Math.PI/2)) - } - if (['Left_knee'].includes(k)) { - modelBone.quaternion - .multiply(modelBoneMapping.localRotation) - // .premultiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), Math.PI)) - } - if (['Left_ankle'].includes(k)) { - modelBone.quaternion - .premultiply(modelBoneMapping.localRotation) - .multiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), Math.PI/2)) - } - - if (['Right_leg'].includes(k)) { - modelBone.quaternion - .multiply(modelBoneMapping.localRotation) - // .multiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), -Math.PI/2)) - } - if (['Right_knee'].includes(k)) { - modelBone.quaternion - .multiply(modelBoneMapping.localRotation) - // .premultiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), Math.PI)) - } - if (['Right_ankle'].includes(k)) { - modelBone.quaternion - .premultiply(modelBoneMapping.localRotation) - .multiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), Math.PI/2)) - } - - if (['Left_shoulder'].includes(k)) { - modelBone.quaternion - .multiply(modelBoneMapping.localRotation) - } - if (['Left_arm'].includes(k)) { - modelBone.quaternion - .premultiply(modelBoneMapping.localRotation) - // .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 1, 0), -Math.PI/2)) // forward - // .multiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), -Math.PI*0.6)) - // .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 0, 1), Math.PI/4)) // up - // .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 0, 1), Math.PI/4)) // down - } - if (['Left_elbow'].includes(k)) { - modelBone.quaternion - .premultiply(modelBoneMapping.localRotation) - } - if (['Left_wrist'].includes(k)) { - modelBone.quaternion - .premultiply(modelBoneMapping.localRotation) - .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 1, 0), -Math.PI/2)) // center - .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 0, 1), Math.PI/4)) // ip - } - - if (['Right_shoulder'].includes(k)) { - modelBone.quaternion - .multiply(modelBoneMapping.localRotation) - } - if (['Right_arm'].includes(k)) { - modelBone.quaternion - .premultiply(modelBoneMapping.localRotation) - // .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 1, 0), -Math.PI/2)) // forward - // .multiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), -Math.PI*0.6)) - // .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 0, 1), Math.PI/4)) // up - // .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 0, 1), Math.PI/4)) // down - } - if (['Right_elbow'].includes(k)) { - modelBone.quaternion - .premultiply(modelBoneMapping.localRotation) - } - if (['Right_wrist'].includes(k)) { - modelBone.quaternion - .premultiply(modelBoneMapping.localRotation) - .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 1, 0), Math.PI/2)) // center - .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 0, 1), -Math.PI/8)) // up - } - - modelBone.updateMatrixWorld(); - } - } -} + } renderer.render(scene, camera); } diff --git a/vrarmik/Rig.js b/vrarmik/Rig.js index b278bbc..e7393ad 100644 --- a/vrarmik/Rig.js +++ b/vrarmik/Rig.js @@ -10,11 +10,97 @@ const poses = { }; class Rig { - constructor(setups) { - const rig = new GameObject('rig'); - this.poseManager = rig.AddComponent(PoseManager); - this.shoulderTransforms = rig.AddComponent(ShoulderTransforms); - this.legsManager = rig.AddComponent(LegsManager); + constructor(model, {boneMappings}) { + const modelBones = { + Hips: null, + Spine: null, + Chest: null, + Neck: null, + Head: null, + + Left_shoulder: null, + Left_arm: null, + Left_elbow: null, + Left_wrist: null, + Left_leg: null, + Left_knee: null, + Left_ankle: null, + + Right_shoulder: null, + Right_arm: null, + Right_elbow: null, + Right_wrist: null, + Right_leg: null, + Right_knee: null, + Right_ankle: null, + }; + this.modelBones = modelBones; + + model.updateMatrixWorld(true); + let skeleton; + model.traverse(o => { + if (o.isSkinnedMesh) { + if (!skeleton) { + skeleton = o.skeleton; + } + + o.bind(skeleton); + } + }); + fixSkeletonZForward(skeleton.bones[0]); + model.traverse(o => { + if (o.isSkinnedMesh) { + o.bind(skeleton); + } + }); + model.updateMatrixWorld(true); + + model.traverse(o => { + if (o.isMesh) { + o.frustumCulled = false; + } + if (o.isSkinnedMesh) { + for (const k in modelBones) { + if (!modelBones[k]) { + const userlandBoneName = boneMappings[k]; + modelBones[k] = skeleton.bones.find(bone => bone.name === userlandBoneName); + modelBones[k].initialQuaternion = modelBones[k].quaternion.clone(); + console.log('found bone', k, userlandBoneName, modelBones[k], modelBones[k].children); + } + } + } + }); + + const _getOffset = (bone, parent = bone.parent) => bone.getWorldPosition(new Vector3()).sub(parent.getWorldPosition(new Vector3())); + const setups = { + spine: _getOffset(modelBones.Spine), + hips: _getOffset(modelBones.Spine, modelBones.Head), + neck: _getOffset(modelBones.Neck), + head: _getOffset(modelBones.Head), + + leftShoulder: _getOffset(modelBones.Right_shoulder), + leftUpperArm: _getOffset(modelBones.Right_arm), + leftLowerArm: _getOffset(modelBones.Right_elbow), + leftHand: _getOffset(modelBones.Right_wrist), + + rightShoulder: _getOffset(modelBones.Left_shoulder), + rightUpperArm: _getOffset(modelBones.Left_arm), + rightLowerArm: _getOffset(modelBones.Left_elbow), + rightHand: _getOffset(modelBones.Left_wrist), + + leftUpperLeg: _getOffset(modelBones.Right_leg), + leftLowerLeg: _getOffset(modelBones.Right_knee), + leftFoot: _getOffset(modelBones.Right_ankle), + + rightUpperLeg: _getOffset(modelBones.Left_leg), + rightLowerLeg: _getOffset(modelBones.Left_knee), + rightFoot: _getOffset(modelBones.Left_ankle), + }; + + const rigObject = new GameObject('rig'); + this.poseManager = rigObject.AddComponent(PoseManager); + this.shoulderTransforms = rigObject.AddComponent(ShoulderTransforms); + this.legsManager = rigObject.AddComponent(LegsManager); this.shoulderTransforms.spine.localPosition = setups.spine; this.shoulderTransforms.localPosition = setups.hips; @@ -65,11 +151,129 @@ class Rig { rightLowerLeg: this.legsManager.rightLeg.lowerLeg, rightFoot: this.legsManager.rightLeg.foot, }; + this.modelBoneOutputs = { + Hips: this.outputs.hips, + Spine: this.outputs.spine, + Chest: this.outputs.chest, + Neck: this.outputs.neck, + Head: this.outputs.hmd, + + Left_shoulder: this.outputs.rightShoulder, + Left_arm: this.outputs.rightUpperArm, + Left_elbow: this.outputs.rightLowerArm, + Left_wrist: this.outputs.rightHand, + Left_leg: this.outputs.rightUpperLeg, + Left_knee: this.outputs.rightLowerLeg, + Left_ankle: this.outputs.rightFoot, + + Right_shoulder: this.outputs.leftShoulder, + Right_arm: this.outputs.leftUpperArm, + Right_elbow: this.outputs.leftLowerArm, + Right_wrist: this.outputs.leftHand, + Right_leg: this.outputs.leftUpperLeg, + Right_knee: this.outputs.leftLowerLeg, + Right_ankle: this.outputs.leftFoot, + }; GameObject.startAll(); } update() { GameObject.updateAll(); + + for (const k in this.modelBones) { + const modelBone = this.modelBones[k]; + const modelBoneOutput = this.modelBoneOutputs[k]; + + if (k === 'Hips') { + modelBone.position.copy(modelBoneOutput.position); + } + modelBone.quaternion + .copy(modelBone.initialQuaternion) + + if (['Hips', 'Spine', 'Chest', 'Neck', 'Head'].includes(k)) { + modelBone.quaternion + .multiply(modelBoneOutput.localRotation) + } + + if (['Left_leg'].includes(k)) { + modelBone.quaternion + .multiply(modelBoneOutput.localRotation) + // .multiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), -Math.PI/2)) + } + if (['Left_knee'].includes(k)) { + modelBone.quaternion + .multiply(modelBoneOutput.localRotation) + // .premultiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), Math.PI)) + } + if (['Left_ankle'].includes(k)) { + modelBone.quaternion + .premultiply(modelBoneOutput.localRotation) + .multiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), Math.PI/2)) + } + + if (['Right_leg'].includes(k)) { + modelBone.quaternion + .multiply(modelBoneOutput.localRotation) + // .multiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), -Math.PI/2)) + } + if (['Right_knee'].includes(k)) { + modelBone.quaternion + .multiply(modelBoneOutput.localRotation) + // .premultiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), Math.PI)) + } + if (['Right_ankle'].includes(k)) { + modelBone.quaternion + .premultiply(modelBoneOutput.localRotation) + .multiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), Math.PI/2)) + } + + if (['Left_shoulder'].includes(k)) { + modelBone.quaternion + .multiply(modelBoneOutput.localRotation) + } + if (['Left_arm'].includes(k)) { + modelBone.quaternion + .premultiply(modelBoneOutput.localRotation) + // .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 1, 0), -Math.PI/2)) // forward + // .multiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), -Math.PI*0.6)) + // .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 0, 1), Math.PI/4)) // up + // .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 0, 1), Math.PI/4)) // down + } + if (['Left_elbow'].includes(k)) { + modelBone.quaternion + .premultiply(modelBoneOutput.localRotation) + } + if (['Left_wrist'].includes(k)) { + modelBone.quaternion + .premultiply(modelBoneOutput.localRotation) + .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 1, 0), -Math.PI/2)) // center + .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 0, 1), Math.PI/4)) // ip + } + + if (['Right_shoulder'].includes(k)) { + modelBone.quaternion + .multiply(modelBoneOutput.localRotation) + } + if (['Right_arm'].includes(k)) { + modelBone.quaternion + .premultiply(modelBoneOutput.localRotation) + // .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 1, 0), -Math.PI/2)) // forward + // .multiply(new Quaternion().setFromAxisAngle(new Vector3(1, 0, 0), -Math.PI*0.6)) + // .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 0, 1), Math.PI/4)) // up + // .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 0, 1), Math.PI/4)) // down + } + if (['Right_elbow'].includes(k)) { + modelBone.quaternion + .premultiply(modelBoneOutput.localRotation) + } + if (['Right_wrist'].includes(k)) { + modelBone.quaternion + .premultiply(modelBoneOutput.localRotation) + .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 1, 0), Math.PI/2)) // center + .multiply(new Quaternion().setFromAxisAngle(new Vector3(0, 0, 1), -Math.PI/8)) // up + } + modelBone.updateMatrixWorld(); + } } } export default Rig; \ No newline at end of file From 2c88d2caa1def584c1c68c97585f3d66bebd5b32 Mon Sep 17 00:00:00 2001 From: Avaer Kazmer Date: Sat, 19 Oct 2019 00:47:52 -0400 Subject: [PATCH 146/562] Bugfix missing avatar rig file import --- app.html | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/app.html b/app.html index 6d7cd88..6a1c256 100644 --- a/app.html +++ b/app.html @@ -26,9 +26,8 @@ - - + + + + + + @@ -14,6 +204,31 @@ + + + + + + Viewer + Documentation + Browser + + Exoland + Coming soon! + + + + + Examples + + model.glb + model2.vrm + model3.unitypackage + + + + +
Coming soon!
+ model.glb + model2.vrm + model3.unitypackage +