All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jmol.renderbio.RocketRenderer Maven / Gradle / Ivy

There is a newer version: 14.31.10
Show newest version
package org.jmol.renderbio;

import javajs.util.P3;
import javajs.util.V3;

import org.jmol.api.JmolRendererInterface;
import org.jmol.c.STR;
import org.jmol.java.BS;
import org.jmol.modelsetbio.AlphaMonomer;
import org.jmol.modelsetbio.AminoPolymer;
import org.jmol.modelsetbio.Helix;
import org.jmol.modelsetbio.ProteinStructure;
import org.jmol.modelsetbio.Sheet;
import org.jmol.util.GData;
import org.jmol.util.MeshSurface;
import org.jmol.viewer.TransformManager;
import org.jmol.viewer.Viewer;

public class RocketRenderer {
  private boolean tPending;
  private ProteinStructure proteinstructurePending;
  private int startIndexPending;
  private int endIndexPending;

  private V3 vtemp;
  private P3 screenA, screenB, screenC;
  private short colix;
  private short mad;
  private RocketsRenderer rr;
  private Viewer vwr;
  private JmolRendererInterface g3d;
  private TransformManager tm;
  private boolean renderArrowHeads;
  private boolean isRockets;

  public RocketRenderer(){
  }
  
  RocketRenderer set(RocketsRenderer rr) {
    screenA = new P3();
    screenB = new P3();
    screenC = new P3();
    vtemp = new V3();
    this.rr = rr;
    vwr = rr.vwr;
    tm = rr.vwr.tm;
    isRockets = rr.isRockets;
    return this;
  }

  void renderRockets() {
    // doing the cylinders separately because we want to connect them if we can.

    // Key structures that must render properly
    // include 1crn and 7hvp

    g3d = rr.g3d;
    tPending = false;
    renderArrowHeads = rr.renderArrowHeads;
    BS bsVisible = rr.bsVisible;
    for (int i = bsVisible.nextSetBit(0); i >= 0; i = bsVisible
        .nextSetBit(i + 1)) {
      if (rr.structureTypes[i] == STR.HELIX || isRockets && rr.structureTypes[i] == STR.SHEET) {
        renderSpecialSegment((AlphaMonomer) rr.monomers[i], rr.getLeadColix(i), rr.mads[i]);
      } else if (isRockets) {
        renderPending();
        rr.renderHermiteConic(i, true, 7);
      }
    }
    renderPending();
  }

  private void renderSpecialSegment(AlphaMonomer monomer, short thisColix,
                                      short thisMad) {
    ProteinStructure proteinstructure = monomer.proteinStructure;
    if (tPending) {
      if (proteinstructure == proteinstructurePending && thisMad == mad
          && thisColix == colix
          && proteinstructure.getIndex(monomer) == endIndexPending + 1) {
        ++endIndexPending;
        return;
      }
      renderPending();
    }
    proteinstructurePending = proteinstructure;
    startIndexPending = endIndexPending = proteinstructure.getIndex(monomer);
    colix = thisColix;
    mad = thisMad;
    tPending = true;
  }

  private void renderPending() {
    if (!tPending)
      return;
    P3[] segments = proteinstructurePending.getSegments();
    boolean renderArrowHead = (renderArrowHeads && endIndexPending == proteinstructurePending.nRes - 1);
    if (proteinstructurePending instanceof Helix)
      renderPendingRocketSegment(endIndexPending, segments[startIndexPending],
          segments[endIndexPending], segments[endIndexPending + 1],
          renderArrowHead);
    else if (proteinstructurePending instanceof Sheet 
        && ((Sheet)proteinstructurePending).apolymer instanceof AminoPolymer)
      renderPendingSheetPlank(segments[startIndexPending],
          segments[endIndexPending], segments[endIndexPending + 1],
          renderArrowHead);
    tPending = false;
  }

  /**
   * @param i
   * @param pointStart
   * @param pointBeforeEnd
   *        ignored now that arrow heads protrude beyond end of rocket
   * @param pointEnd
   * @param renderArrowHead
   */
  private void renderPendingRocketSegment(int i, P3 pointStart,
                                          P3 pointBeforeEnd, P3 pointEnd,
                                          boolean renderArrowHead) {
    if (g3d.setC(colix)) {
      tm.transformPt3f(pointStart, screenA);
      tm.transformPt3f((renderArrowHead ? pointBeforeEnd : pointEnd), screenB);
      int zMid = (int) Math.floor((screenA.z + screenB.z) / 2f);
      int diameter = ((int) vwr.tm.scaleToScreen(zMid, mad));
      if (!renderArrowHead || pointStart != pointBeforeEnd)
        g3d.fillCylinderBits(GData.ENDCAPS_FLAT, diameter, screenA, screenB);
      if (renderArrowHead) {
        screenA.sub2(pointEnd, pointBeforeEnd);
        tm.transformPt3f(pointEnd, screenC);
        int coneDiameter = (mad << 1) - (mad >> 1);
        coneDiameter = (int) vwr.tm.scaleToScreen(
            (int) Math.floor(screenB.z), coneDiameter);
        g3d.fillConeScreen3f(GData.ENDCAPS_FLAT, coneDiameter, screenB,
            screenC, false);
      } else {
        
      }
      if (startIndexPending == endIndexPending)
        return;
      P3 t = screenB;
      screenB = screenC;
      screenC = t;
    }
  }

  private final static int[][] boxFaces =
  {
    { 0, 1, 3, 2 },
    { 0, 2, 6, 4 },
    { 0, 4, 5, 1 },
    { 7, 5, 4, 6 },
    { 7, 6, 2, 3 },
    { 7, 3, 1, 5 } };

  private final static int[][] arrowHeadFaces =
  {
    { 1, 0, 4 },
    { 2, 3, 5 },
    { 0, 1, 3, 2 },
    { 2, 5, 4, 0 },
    { 1, 4, 5, 3 }
  };
  
  private P3 ptC, ptTip;
  private P3[] corners, screenCorners;
  private V3 vW, vH;
  private MeshSurface meshSurface;

  private void renderPendingSheetPlank(P3 ptStart, P3 pointBeforeEnd, P3 ptEnd,
                                       boolean renderArrowHead) {
    if (!g3d.setC(colix))
      return;
    if (corners == null) {
      ptC = new P3();
      ptTip = new P3();
      vW = new V3();
      vH = new V3();
      screenCorners = new P3[8];
      corners = new P3[8];
      for (int i = 8; --i >= 0;) {
        corners[i] = new P3();
        screenCorners[i] = new P3();
      }
    }
    if (renderArrowHead) {
      setBox(1.25f, 0.333f, pointBeforeEnd);
      ptTip.scaleAdd2(-0.5f, vH, ptEnd);
      for (int i = 4; --i >= 0;) {
        P3 corner = corners[i];
        corner.setT(ptC);
        if ((i & 1) != 0)
          corner.add(vW);
        if ((i & 2) != 0)
          corner.add(vH);
      }
      corners[4].setT(ptTip);
      corners[5].add2(ptTip, vH);
      renderPart(arrowHeadFaces);
      ptEnd = pointBeforeEnd;
    }
    setBox(1f, 0.25f, ptStart);
    vtemp.sub2(ptEnd, ptStart);
    if (vtemp.lengthSquared() == 0)
      return;
    buildBox(ptC, vW, vH, vtemp);
    renderPart(boxFaces);
  }
  
  private void setBox(float w, float h, P3 pt) {
    ((Sheet) proteinstructurePending).setBox(w, h, pt, vW, vH, ptC, mad / 1000f);
  }

  private void buildBox(P3 pointCorner, V3 scaledWidthVector,
                        V3 scaledHeightVector, V3 lengthVector) {
    for (int i = 8; --i >= 0;) {
      P3 corner = corners[i];
      corner.setT(pointCorner);
      if ((i & 1) != 0)
        corner.add(scaledWidthVector);
      if ((i & 2) != 0)
        corner.add(scaledHeightVector);
      if ((i & 4) != 0)
        corner.add(lengthVector);
    }
  }

  private void renderPart(int[][] planes) {
    if (rr.exportType == GData.EXPORT_CARTESIAN) {
      if (meshSurface == null) {
        meshSurface = new MeshSurface();
        meshSurface.vs = corners;
        meshSurface.haveQuads = true;
        meshSurface.vc = corners.length;
      }
      meshSurface.pis = planes;
      meshSurface.pc = planes.length;
      g3d.drawSurface(meshSurface, colix);
    } else {
      for (int i = 8; --i >= 0;)
        tm.transformPt3f(corners[i], screenCorners[i]);
      for (int i = planes.length; --i >= 0;) {
        int[] f = planes[i];
        if (f.length == 3)
          g3d.fillTriangle3f(screenCorners[f[0]], screenCorners[f[1]],
              screenCorners[f[2]], true);
        else
          g3d.fillQuadrilateral(screenCorners[f[0]], screenCorners[f[1]],
              screenCorners[f[2]], screenCorners[f[3]], true);
      }
    }
  }

  
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy