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

org.jmol.render.SticksRenderer Maven / Gradle / Ivy

There is a newer version: 14.31.10
Show newest version
/* $RCSfile$
 * $Author: hansonr $
 * $Date: 2017-04-06 07:09:34 -0500 (Thu, 06 Apr 2017) $
 * $Revision: 21485 $

 *
 * Copyright (C) 2002-2005  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 St, Fifth Floor, Boston, MA 02110-1301 USA.
 */

package org.jmol.render;

import org.jmol.c.PAL;
import org.jmol.java.BS;
import org.jmol.modelset.Atom;
import org.jmol.modelset.Bond;
import org.jmol.script.T;
import org.jmol.util.C;
import org.jmol.util.GData;
import org.jmol.util.Edge;

import javajs.util.A4;
import javajs.util.M3;
import javajs.util.P3;
import javajs.util.V3;
import org.jmol.viewer.JC;

public class SticksRenderer extends FontLineShapeRenderer {

  private boolean showMultipleBonds;
  private float multipleBondSpacing;
  private float multipleBondRadiusFactor;
  private boolean bondsPerp;
  private boolean useBananas;
  private byte modeMultipleBond;
  private boolean isCartesian;
  //boolean showHydrogens;
  private byte endcaps;

  private boolean ssbondsBackbone;
  private boolean hbondsBackbone;
  private boolean bondsBackbone;
  private boolean hbondsSolid;

  private Atom a, b;
  private Bond bond;
  private int xA, yA, zA;
  private int xB, yB, zB;
  private int dx, dy;
  private int mag2d;
  private int bondOrder;
  private boolean wireframeOnly;
  private boolean isAntialiased;
  private boolean slabbing;
  private boolean slabByAtom;

  private final V3 x = new V3();
  private final V3 y = new V3();
  private final V3 z = new V3();
  private final P3 p1 = new P3();
  private final P3 p2 = new P3();
  private final BS bsForPass2 = BS.newN(64);
  private boolean isPass2;
  private double rTheta;

  @Override
  protected boolean render() {
    Bond[] bonds = ms.bo;
    if (bonds == null)
      return false;
    isPass2 = vwr.gdata.isPass2;
    if (!isPass2)
      bsForPass2.clearAll();
    slabbing = tm.slabEnabled;
    slabByAtom = vwr.getBoolean(T.slabbyatom);
    endcaps = GData.ENDCAPS_SPHERICAL;
    dashDots = (vwr.getBoolean(T.partialdots) ? sixdots : dashes);
    isCartesian = (exportType == GData.EXPORT_CARTESIAN);
    getMultipleBondSettings(false);
    wireframeOnly = !vwr.checkMotionRendering(T.bonds);
    ssbondsBackbone = vwr.getBoolean(T.ssbondsbackbone);
    hbondsBackbone = vwr.getBoolean(T.hbondsbackbone);
    bondsBackbone = hbondsBackbone | ssbondsBackbone;
    hbondsSolid = vwr.getBoolean(T.hbondssolid);
    isAntialiased = g3d.isAntialiased();
    boolean needTranslucent = false;
    if (isPass2) {
      if (!isExport)
        for (int i = bsForPass2.nextSetBit(0); i >= 0; i = bsForPass2
            .nextSetBit(i + 1)) {
          bond = bonds[i];
          renderBond();
        }
    } else {
      for (int i = ms.bondCount; --i >= 0;) {
        bond = bonds[i];
        if ((bond.shapeVisibilityFlags & myVisibilityFlag) != 0 && renderBond()) {
          needTranslucent = true;
          bsForPass2.set(i);
        }
      }
    }
    return needTranslucent;
  }

  private void getMultipleBondSettings(boolean isPymol) {
    useBananas = (vwr.getBoolean(T.multiplebondbananas) && !isPymol);
    // negative spacing is relative, depending upon atom-atom distance;
    // positive spacing is absolute, for fixed in-plane (radiusFactor > 0) or perp-plane (radiusFactor < 0)
    multipleBondSpacing = (isPymol ? 0.15f : vwr
        .getFloat(T.multiplebondspacing));
    // negative radius factor indicates perpendicular fixed double bond
    multipleBondRadiusFactor = (isPymol ? 0.4f : vwr
        .getFloat(T.multiplebondradiusfactor));
    bondsPerp = (useBananas || multipleBondSpacing > 0
        && multipleBondRadiusFactor < 0);
    if (useBananas)
      multipleBondSpacing = (multipleBondSpacing < 0 ? -multipleBondSpacing * 0.4f
          : multipleBondSpacing);
    multipleBondRadiusFactor = Math.abs(multipleBondRadiusFactor);
    if (multipleBondSpacing == 0 && isCartesian)
      multipleBondSpacing = 0.2f;
    modeMultipleBond = vwr.g.modeMultipleBond;
    showMultipleBonds = (multipleBondSpacing != 0
        && modeMultipleBond != JC.MULTIBOND_NEVER && vwr
        .getBoolean(T.showmultiplebonds));
  }

  private boolean renderBond() {
    Atom atomA0, atomB0;

    a = atomA0 = bond.atom1;
    b = atomB0 = bond.atom2;

    int order = bond.order & ~Edge.BOND_NEW;
    if (bondsBackbone) {
      if (ssbondsBackbone && (order & Edge.BOND_SULFUR_MASK) != 0) {
        // for ssbonds, always render the sidechain,
        // then render the backbone version
        /*
         mth 2004 04 26
         No, we are not going to do this any more
         render(bond, atomA, atomB);
         */

        a = a.group.getLeadAtomOr(a);
        b = b.group.getLeadAtomOr(b);
      } else if (hbondsBackbone && Edge.isOrderH(order)) {
        a = a.group.getLeadAtomOr(a);
        b = b.group.getLeadAtomOr(b);
      }
    }
    if (!isPass2
        && (!a.isVisible(Atom.ATOM_INFRAME_NOTHIDDEN)
            || !b.isVisible(Atom.ATOM_INFRAME_NOTHIDDEN)
            || !g3d.isInDisplayRange(a.sX, a.sY) || !g3d.isInDisplayRange(b.sX,
            b.sY)))
      return false;

    if (slabbing) {
      boolean ba = vwr.gdata.isClippedZ(a.sZ);
      if (ba && vwr.gdata.isClippedZ(b.sZ) || slabByAtom
          && (ba || vwr.gdata.isClippedZ(b.sZ)))
        return false;
    }
    zA = a.sZ;
    zB = b.sZ;
    if (zA == 1 || zB == 1)
      return false;
    colixA = atomA0.colixAtom;
    colixB = atomB0.colixAtom;
    if (((colix = bond.colix) & C.OPAQUE_MASK) == C.USE_PALETTE) {
      colix = (short) (colix & ~C.OPAQUE_MASK);
      colixA = C.getColixInherited(
          (short) (colix | vwr.cm.getColixAtomPalette(atomA0, PAL.CPK.id)),
          colixA);
      colixB = C.getColixInherited(
          (short) (colix | vwr.cm.getColixAtomPalette(atomB0, PAL.CPK.id)),
          colixB);
    } else {
      colixA = C.getColixInherited(colix, colixA);
      colixB = C.getColixInherited(colix, colixB);
    }
    boolean needTranslucent = false;
    if (!isExport && !isPass2) {
      boolean doA = !C.renderPass2(colixA);
      boolean doB = !C.renderPass2(colixB);
      if (!doA || !doB) {
        if (!doA && !doB && !needTranslucent) {
          g3d.setC(!doA ? colixA : colixB);
          return true;
        }
        needTranslucent = true;
      }
    }

    // set the rendered bond order

    bondOrder = order & ~Edge.BOND_NEW;
    if ((bondOrder & Edge.BOND_PARTIAL_MASK) == 0) {
      if ((bondOrder & Edge.BOND_SULFUR_MASK) != 0)
        bondOrder &= ~Edge.BOND_SULFUR_MASK;
      if ((bondOrder & Edge.BOND_COVALENT_MASK) != 0) {
        if (!showMultipleBonds
            || (modeMultipleBond == JC.MULTIBOND_NOTSMALL && mad > JC.madMultipleBondSmallMaximum)
            || (bondOrder & Edge.BOND_PYMOL_MULT) == Edge.BOND_RENDER_SINGLE) {
          bondOrder = 1;
        }
      }
    }

    // set the mask

    int mask = 0;
    switch (bondOrder) {
    case 1:
    case 2:
    case 3:
    case 4:
    case 5:
    case 6:
      break;
    case Edge.BOND_ORDER_UNSPECIFIED:
    case Edge.BOND_AROMATIC_SINGLE:
      bondOrder = 1;
      mask = (order == Edge.BOND_AROMATIC_SINGLE ? 0 : 1);
      break;
    case Edge.BOND_AROMATIC:
    case Edge.BOND_AROMATIC_DOUBLE:
      bondOrder = 2;
      mask = (order == Edge.BOND_AROMATIC ? getAromaticDottedBondMask() : 0);
      break;
    default:
      if ((bondOrder & Edge.BOND_PARTIAL_MASK) != 0) {
        bondOrder = Edge.getPartialBondOrder(order);
        mask = Edge.getPartialBondDotted(order);
      } else if (Edge.isOrderH(bondOrder)) {
        bondOrder = 1;
        if (!hbondsSolid)
          mask = -1;
      } else if (bondOrder == Edge.BOND_STRUT) {
        bondOrder = 1;
      } else if ((bondOrder & Edge.BOND_PYMOL_MULT) == Edge.BOND_PYMOL_MULT) {
        getMultipleBondSettings(true);
        bondOrder &= 3;
        mask = -2;
      }
    }

    // set the diameter

    xA = a.sX;
    yA = a.sY;
    xB = b.sX;
    yB = b.sY;

    mad = bond.mad;
    if (multipleBondRadiusFactor > 0 && bondOrder > 1)
      mad *= multipleBondRadiusFactor;
    dx = xB - xA;
    dy = yB - yA;
    width = (int) vwr.tm.scaleToScreen((zA + zB) / 2, mad);
    if (wireframeOnly && width > 0)
      width = 1;
    if (!isCartesian) {
      asLineOnly = (width <= 1);
      if (asLineOnly && (isAntialiased)) {
        width = 3;
        asLineOnly = false;
      }
    }

    // draw the bond

    switch (mask) {
    case -2:
      drawBond(0);
      getMultipleBondSettings(false);
      break;
    case -1:
      drawDashed(xA, yA, zA, xB, yB, zB, hDashes);
      break;
    default:
      switch (bondOrder) {
      case 4: {
        bondOrder = 2;
        float f = multipleBondRadiusFactor;
        if (f == 0 && width > 1)
          width = (int) (width * 0.5);
        float m = multipleBondSpacing;
        if (m < 0)
          multipleBondSpacing = 0.30f;
        drawBond(mask);
        bondsPerp = !bondsPerp;
        bondOrder = 2;
        drawBond(mask >> 2);
        bondsPerp = !bondsPerp;
        multipleBondSpacing = m;
      }
        break;
      case 5: {
        bondOrder = 3;
        float f = multipleBondRadiusFactor;
        if (f == 0 && width > 1)
          width = (int) (width * 0.5);
        float m = multipleBondSpacing;
        if (m < 0)
          multipleBondSpacing = 0.20f;
        drawBond(mask);
        bondsPerp = !bondsPerp;
        bondOrder = 2;
        multipleBondSpacing *= 1.5f;
        drawBond(mask >> 3);
        bondsPerp = !bondsPerp;
        multipleBondSpacing = m;
      }
        break;
      case 6: {
        bondOrder = 4;
        float f = multipleBondRadiusFactor;
        if (f == 0 && width > 1)
          width = (int) (width * 0.5);
        float m = multipleBondSpacing;
        if (m < 0)
          multipleBondSpacing = 0.15f;
        drawBond(mask);
        bondsPerp = !bondsPerp;
        bondOrder = 2;
        multipleBondSpacing *= 1.5f;
        drawBond(mask >> 4);
        bondsPerp = !bondsPerp;
        multipleBondSpacing = m;
      }
        break;
      default:
        drawBond(mask);
      }
      break;
    }
    return needTranslucent;
  }

  private void drawBond(int dottedMask) {
    boolean isDashed = (dottedMask & 1) != 0;
    if (isCartesian && bondOrder == 1 && !isDashed) {
      // bypass screen rendering and just use the atoms themselves
      g3d.drawBond(a, b, colixA, colixB, endcaps, mad, -1);
      return;
    }
    boolean isEndOn = (dx == 0 && dy == 0);
    if (isEndOn && asLineOnly && !isCartesian)
      return;
    boolean doFixedSpacing = (bondOrder > 1 && multipleBondSpacing > 0);
    boolean isPiBonded = doFixedSpacing
        && (vwr.getHybridizationAndAxes(a.i, z, x, "pz") != null || vwr
            .getHybridizationAndAxes(b.i, z, x, "pz") != null)
        && !Float.isNaN(x.x);
    if (isEndOn && !doFixedSpacing) {
      // end-on view
      int space = width / 8 + 3;
      int step = width + space;
      int y = yA - (bondOrder - 1) * step / 2;
      do {
        fillCylinder(colixA, colixB, endcaps, width, xA, y, zA, xB, y, zB);
        y += step;
      } while (--bondOrder > 0);
      return;
    }
    if (bondOrder == 1) {
      if (isDashed)
        drawDashed(xA, yA, zA, xB, yB, zB, dashDots);
      else
        fillCylinder(colixA, colixB, endcaps, width, xA, yA, zA, xB, yB, zB);
      return;
    }
    if (doFixedSpacing) {
      if (!isPiBonded) // obscure point
        z.setT(P3.getUnlikely());
      x.sub2(b, a);
      y.cross(x, z);
      y.normalize();
      if (Float.isNaN(y.x)) {
        // in case x and z are parallel (O=C=O)
        z.setT(P3.getUnlikely());
        y.cross(x, z);
        y.cross(y, x);
        y.normalize();
      }
      if (bondsPerp)
        y.cross(y, x);
      y.scale(multipleBondSpacing);
      x.setT(y);
      x.scale((bondOrder - 1) / 2f);
      if (useBananas) {
        drawBanana(a, b, x, 0);
        switch (bondOrder) {
        case 4:
          drawBanana(a, b, x, 90);
          drawBanana(a, b, x, -90);
          //$FALL-THROUGH$
        case 2:
        default:
          drawBanana(a, b, x, 180);
          break;
        case 3:
          drawBanana(a, b, x, 120);
          drawBanana(a, b, x, -120);
          break;
        }
        return;
      }
      p1.sub2(a, x);
      p2.sub2(b, x);

      while (true) {
        if (isCartesian) {
          // bypass screen rendering and just use the atoms themselves
          g3d.drawBond(p1, p2, colixA, colixB, endcaps, mad, -2);
        } else {
          tm.transformPtScr(p1, s1);
          tm.transformPtScr(p2, s2);
          if (isDashed)
            drawDashed(s1.x, s1.y, s1.z, s2.x, s2.y, s2.z, dashDots);
          else
            fillCylinder(colixA, colixB, endcaps, width, s1.x, s1.y, s1.z,
                s2.x, s2.y, s2.z);
        }
        dottedMask >>= 1;
        isDashed = (dottedMask & 1) != 0;
        if (--bondOrder <= 0)
          break;
        p1.add(y);
        p2.add(y);
        stepAxisCoordinates();
      }
      return;
    }
    int dxB = dx * dx;
    int dyB = dy * dy;
    mag2d = (int) Math.round(Math.sqrt(dxB + dyB));
    resetAxisCoordinates();
    if (isCartesian && bondOrder == 3) {
      fillCylinder(colixA, colixB, endcaps, width, xAxis1, yAxis1, zA, xAxis2,
          yAxis2, zB);
      stepAxisCoordinates();
      x.sub2(b, a);
      x.scale(0.05f);
      p1.sub2(a, x);
      p2.add2(b, x);
      g3d.drawBond(p1, p2, colixA, colixB, endcaps, mad, -2);
      stepAxisCoordinates();
      fillCylinder(colixA, colixB, endcaps, width, xAxis1, yAxis1, zA, xAxis2,
          yAxis2, zB);
      return;
    }
    while (true) {
      if ((dottedMask & 1) != 0)
        drawDashed(xAxis1, yAxis1, zA, xAxis2, yAxis2, zB, dashDots);
      else
        fillCylinder(colixA, colixB, endcaps, width, xAxis1, yAxis1, zA,
            xAxis2, yAxis2, zB);
      dottedMask >>= 1;
      if (--bondOrder <= 0)
        break;
      stepAxisCoordinates();
    }
  }

  private int xAxis1, yAxis1, xAxis2, yAxis2, dxStep, dyStep;

  private void resetAxisCoordinates() {
    int space = mag2d >> 3;
    if (multipleBondSpacing != -1 && multipleBondSpacing < 0)
      space *= -multipleBondSpacing;
    int step = width + space;
    dxStep = step * dy / mag2d;
    dyStep = step * -dx / mag2d;
    xAxis1 = xA;
    yAxis1 = yA;
    xAxis2 = xB;
    yAxis2 = yB;
    int f = (bondOrder - 1);
    xAxis1 -= dxStep * f / 2;
    yAxis1 -= dyStep * f / 2;
    xAxis2 -= dxStep * f / 2;
    yAxis2 -= dyStep * f / 2;
  }

  private void stepAxisCoordinates() {
    xAxis1 += dxStep;
    yAxis1 += dyStep;
    xAxis2 += dxStep;
    yAxis2 += dyStep;
  }

  private int getAromaticDottedBondMask() {
    Atom atomC = b.findAromaticNeighbor(a.i);
    if (atomC == null)
      return 1;
    int dxAC = atomC.sX - xA;
    int dyAC = atomC.sY - yA;
    return ((dx * dyAC - dy * dxAC) < 0 ? 2 : 1);
  }

  private M3 rot;
  private A4 a4;

  private void drawBanana(Atom a, Atom b, V3 x, int deg) {
    g3d.addRenderer(T.hermitelevel);
    vectorT.sub2(b, a);
    if (rot == null) {
      rot = new M3();
      a4 = new A4();
    }
    a4.setVA(vectorT, (float) (deg * Math.PI / 180));
    rot.setAA(a4);
    pointT.setT(a);
    pointT3.setT(b);
    pointT2.ave(a, b);
    rot.rotate2(x, vectorT);
    pointT2.add(vectorT);
    tm.transformPtScrT3(a, pointT);
    tm.transformPtScrT3(pointT2, pointT2);
    tm.transformPtScrT3(b, pointT3);
    int w = Math.max(width, 1);
    g3d.setC(colixA);
    g3d.fillHermite(5, w, w, w, pointT, pointT, pointT2, pointT3);
    g3d.setC(colixB);
    g3d.fillHermite(5, w, w, w, pointT, pointT2, pointT3, pointT3);
  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy