
org.jmol.thread.SpinThread Maven / Gradle / Ivy
/* $RCSfile$
* $Author$
* $Date$
* $Revision$
*
* Copyright (C) 2011 The Jmol Development Team
*
* Contact: [email protected]
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
* 02110-1301, USA.
*/
package org.jmol.thread;
import org.jmol.java.BS;
import org.jmol.script.T;
import javajs.util.Lst;
import javajs.util.M4;
import org.jmol.util.Logger;
import javajs.util.P3;
import org.jmol.viewer.JC;
import org.jmol.viewer.TransformManager;
import org.jmol.viewer.Viewer;
public class SpinThread extends JmolThread {
/**
*
*/
private TransformManager transformManager;
private float endDegrees;
private Lst endPositions;
private float[] dihedralList;
private float nDegrees;
private BS bsAtoms;
private boolean isNav;
private boolean isGesture;
private float myFps;
private float angle;
private boolean haveNotified;
private int index;
//private boolean navigatingSurface;
private BS[] bsBranches;
private boolean isDone = false;
private M4 m4;
/**
* @j2sIgnore
*
*/
public SpinThread() {}
@SuppressWarnings("unchecked")
@Override
public int setManager(Object manager, Viewer vwr, Object params) {
transformManager = (TransformManager) manager;
setViewer(vwr, "SpinThread");
Object[] options = (Object[]) params;
//f//loat endDegrees, List endPositions, float[] dihedralList, BS bsAtoms, boolean isNav,
//boolean isGesture) {
//Float.valueOf(endDegrees), endPositions, dihedralList,
//bsAtoms, Boolean.valueOf(isGesture)} );
// spinThread = new SpinThread(this, vwr, NULL
// === 0, null, null, null, true, false);
if (options == null) {
isNav = true;
} else {
endDegrees = ((Float) options[0]).floatValue();
endPositions = (Lst) options[1];
dihedralList = (float[]) options[2];
if (dihedralList != null)
bsBranches = vwr.ms.getBsBranches(dihedralList);
bsAtoms = (BS) options[3];
isGesture = (options[4] != null);
}
return 0;
}
/**
* Java:
*
* run1(INIT) while(!interrupted()) { run1(MAIN) } run1(FINISH)
*
* JavaScript:
*
* run1(INIT) run1(MAIN) --> setTimeout to run1(CHECK) or run1(FINISH) and
* return run1(CHECK) --> setTimeout to run1(CHECK) or run1(MAIN) or
* run1(FINISH) and return
*
*/
@Override
protected void run1(int mode) throws InterruptedException {
while (true)
switch (mode) {
case INIT:
myFps = (isNav ? transformManager.navFps : transformManager.spinFps);
vwr.g.setB(isNav ? "_navigating" : "_spinning", true);
haveReference = true;
vwr.startHoverWatcher(false);
mode = MAIN;
break;
case MAIN:
if (isReset || checkInterrupted(transformManager.spinThread)) {
mode = FINISH;
break;
}
if (isNav && myFps != transformManager.navFps) {
myFps = transformManager.navFps;
index = 0;
startTime = System.currentTimeMillis();
} else if (!isNav && myFps != transformManager.spinFps
&& bsAtoms == null) {
myFps = transformManager.spinFps;
index = 0;
startTime = System.currentTimeMillis();
}
if (myFps == 0
|| !(isNav ? transformManager.navOn : transformManager.spinOn)) {
mode = FINISH;
break;
}
//navigatingSurface = vwr.getNavigateSurface();
boolean refreshNeeded = (endDegrees >= 1e10f ? true : isNav ? //navigatingSurface ||
transformManager.navX != 0 || transformManager.navY != 0
|| transformManager.navZ != 0
: transformManager.isSpinInternal
&& transformManager.internalRotationAxis.angle != 0
|| transformManager.isSpinFixed
&& transformManager.fixedRotationAxis.angle != 0
|| !transformManager.isSpinFixed
&& !transformManager.isSpinInternal
&& (transformManager.spinX != 0 || transformManager.spinY != 0 || transformManager.spinZ != 0));
targetTime = (long) (++index * 1000 / myFps);
currentTime = System.currentTimeMillis() - startTime;
sleepTime = (int) (targetTime - currentTime);
//System.out.println(index + " spin thread " + targetTime + " " + currentTime + " " + sleepTime);
if (sleepTime < 0) {
if (!haveNotified)
Logger.info("spinFPS is set too fast (" + myFps
+ ") -- can't keep up!");
haveNotified = true;
startTime -= sleepTime;
sleepTime = 0;
}
boolean isInMotion = (bsAtoms == null && vwr.getInMotion(false));
if (isInMotion) {
if (isGesture) {
mode = FINISH;
break;
}
sleepTime += 1000;
}
if (refreshNeeded && !isInMotion
&& (transformManager.spinOn || transformManager.navOn))
doTransform();
mode = CHECK1;
break;
case CHECK1: // cycling
while (!checkInterrupted(transformManager.spinThread) && !vwr.getRefreshing())
if (!runSleep(10, CHECK1))
return;
if (bsAtoms != null || vwr.g.waitForMoveTo && endDegrees != Float.MAX_VALUE)
vwr.requestRepaintAndWait("spin thread");
else
vwr.refresh(1, "SpinThread");
if (endDegrees >= 1e10f ? nDegrees/endDegrees > 0.99
: !isNav && endDegrees >= 0 ? nDegrees >= endDegrees - 0.001
: -nDegrees <= endDegrees + 0.001) {
isDone = true;
transformManager.setSpinOff();
}
if (!runSleep(sleepTime, MAIN))
return;
mode = MAIN;
break;
case FINISH:
if (dihedralList != null) {
vwr.setDihedrals(dihedralList, bsBranches, 0F);
} else if (bsAtoms != null && endPositions != null) {
// when the standard deviations of the end points was
// exact, we know that we want EXACTLY those final positions
vwr.setAtomCoords(bsAtoms, T.xyz, endPositions);
bsAtoms = null;
endPositions = null;
}
if (!isReset) {
transformManager.setSpinOff();
vwr.startHoverWatcher(true);
}
stopped = !isDone;
resumeEval();
stopped = true;
return;
}
}
private void doTransform() {
if (dihedralList != null) {
float f = 1f / myFps / endDegrees;
vwr.setDihedrals(dihedralList, bsBranches, f);
nDegrees += 1f / myFps;
} else if (isNav) {
transformManager.setNavigationOffsetRelative();//navigatingSurface);
} else if (transformManager.isSpinInternal
|| transformManager.isSpinFixed) {
angle = (transformManager.isSpinInternal ? transformManager.internalRotationAxis
: transformManager.fixedRotationAxis).angle
/ myFps;
if (transformManager.isSpinInternal) {
transformManager.rotateAxisAngleRadiansInternal(angle, bsAtoms, m4);
} else {
transformManager.rotateAxisAngleRadiansFixed(angle, bsAtoms);
}
nDegrees += Math.abs(angle * TransformManager.degreesPerRadian);
//System.out.println(i + " " + angle + " " + nDegrees);
} else { // old way: Rx * Ry * Rz
if (transformManager.spinX != 0) {
transformManager.rotateXRadians(transformManager.spinX
* JC.radiansPerDegree / myFps, null);
}
if (transformManager.spinY != 0) {
transformManager.rotateYRadians(transformManager.spinY
* JC.radiansPerDegree / myFps, null);
}
if (transformManager.spinZ != 0) {
transformManager.rotateZRadians(transformManager.spinZ
* JC.radiansPerDegree / myFps);
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy