/** Copyright (c) 2014-present, Facebook, Inc. All rights reserved. This source code is licensed under the BSD-style license found in the LICENSE file in the root directory of this source tree. An additional grant of patent rights can be found in the PATENTS file in the same directory. */ #import #import "POPVector.h" namespace POP { template struct SSState { T p; T v; }; template struct SSDerivative { T dp; T dv; }; typedef SSState SSState4d; typedef SSDerivative SSDerivative4d; const CFTimeInterval solverDt = 0.001f; const CFTimeInterval maxSolverDt = 30.0f; /** Templated spring solver class. */ template class SpringSolver { double _k; // stiffness double _b; // dampening double _m; // mass double _tp; // threshold double _tv; // threshold velocity double _ta; // threshold acceleration CFTimeInterval _accumulatedTime; SSState _lastState; T _lastDv; bool _started; public: SpringSolver(double k, double b, double m = 1) : _k(k), _b(b), _m(m), _started(false) { _accumulatedTime = 0; _lastState.p = T::Zero(); _lastState.v = T::Zero(); _lastDv = T::Zero(); setThreshold(1.); } ~SpringSolver() { } bool started() { return _started; } void setConstants(double k, double b, double m) { _k = k; _b = b; _m = m; } void setThreshold(double t) { _tp = t / 2; // half a unit _tv = 25.0 * t; // 5 units per second, squared for comparison _ta = 625.0 * t * t; // 5 units per second squared, squared for comparison } T acceleration(const SSState &state, double t) { return state.p*(-_k/_m) - state.v*(_b/_m); } SSDerivative evaluate(const SSState &initial, double t) { SSDerivative output; output.dp = initial.v; output.dv = acceleration(initial, t); return output; } SSDerivative evaluate(const SSState &initial, double t, double dt, const SSDerivative &d) { SSState state; state.p = initial.p + d.dp*dt; state.v = initial.v + d.dv*dt; SSDerivative output; output.dp = state.v; output.dv = acceleration(state, t+dt); return output; } void integrate(SSState &state, double t, double dt) { SSDerivative a = evaluate(state, t); SSDerivative b = evaluate(state, t, dt*0.5, a); SSDerivative c = evaluate(state, t, dt*0.5, b); SSDerivative d = evaluate(state, t, dt, c); T dpdt = (a.dp + (b.dp + c.dp)*2.0 + d.dp) * (1.0/6.0); T dvdt = (a.dv + (b.dv + c.dv)*2.0 + d.dv) * (1.0/6.0); state.p = state.p + dpdt*dt; state.v = state.v + dvdt*dt; _lastDv = dvdt; } SSState interpolate(const SSState &previous, const SSState ¤t, double alpha) { SSState state; state.p = current.p*alpha + previous.p*(1-alpha); state.v = current.v*alpha + previous.v*(1-alpha); return state; } void advance(SSState &state, double t, double dt) { _started = true; if (dt > maxSolverDt) { // excessive time step, force shut down _lastDv = _lastState.v = _lastState.p = T::Zero(); } else { _accumulatedTime += dt; SSState previousState = state, currentState = state; while (_accumulatedTime >= solverDt) { previousState = currentState; this->integrate(currentState, t, solverDt); t += solverDt; _accumulatedTime -= solverDt; } CFTimeInterval alpha = _accumulatedTime / solverDt; _lastState = state = this->interpolate(previousState, currentState, alpha); } } bool hasConverged() { if (!_started) { return false; } for (size_t idx = 0; idx < _lastState.p.size(); idx++) { if (fabs(_lastState.p(idx)) >= _tp) { return false; } } return (_lastState.v.squaredNorm() < _tv) && (_lastDv.squaredNorm() < _ta); } void reset() { _accumulatedTime = 0; _lastState.p = T::Zero(); _lastState.v = T::Zero(); _lastDv = T::Zero(); _started = false; } }; /** Convenience spring solver type definitions. */ typedef SpringSolver SpringSolver2d; typedef SpringSolver SpringSolver3d; typedef SpringSolver SpringSolver4d; }