use of ffx.crystal.Crystal in project ffx by mjschnie.
the class TransitionTemperedOSRW method optimization.
private void optimization(double e, double[] x, double[] gradient) {
if (energyCount % osrwOptimizationFrequency == 0) {
logger.info(String.format(" OSRW Minimization (Step %d)", energyCount));
// Set the underlying Potential's Lambda value to 1.0.
lambdaInterface.setLambda(1.0);
potential.setEnergyTermState(Potential.STATE.BOTH);
if (barostat != null) {
barostat.setActive(false);
}
try {
// Optimize the system.
double startingEnergy = potential.energy(x);
Minimize minimize = new Minimize(null, potential, null);
minimize.minimize(osrwOptimizationEps);
// Collect the minimum energy.
double minEnergy = potential.getTotalEnergy();
// Check for a new minimum within an energy window of the lowest energy structure found.
if (minEnergy < osrwOptimum + osrwOptimizationEnergyWindow) {
if (minEnergy < osrwOptimum) {
osrwOptimum = minEnergy;
}
int n = potential.getNumberOfVariables();
osrwOptimumCoords = new double[n];
osrwOptimumCoords = potential.getCoordinates(osrwOptimumCoords);
double mass = molecularAssembly.getMass();
double density = potential.getCrystal().getDensity(mass);
systemFilter.writeFile(optFile, false);
Crystal uc = potential.getCrystal().getUnitCell();
logger.info(String.format(" Minimum: %12.6f %s (%12.6f g/cc) optimized from %12.6f at step %d.", minEnergy, uc.toShortString(), density, startingEnergy, energyCount));
}
} catch (EnergyException ex) {
String message = ex.getMessage();
logger.info(format(" Energy exception minimizing coordinates at lambda=%8.6f\n %s.", lambda, message));
logger.info(format(" TT-OSRW sampling will continue."));
}
// Set the underlying Potential's Lambda value back to current lambda value.
lambdaInterface.setLambda(lambda);
// Remove the scaling of coordinates & gradient set by the minimizer.
potential.setScaling(null);
// Reset the Potential State
potential.setEnergyTermState(state);
// Reset the Barostat
if (barostat != null) {
barostat.setActive(true);
}
// Revert to the coordinates and gradient prior to optimization.
double eCheck = potential.energyAndGradient(x, gradient);
if (abs(eCheck - e) > osrwOptimizationTolerance) {
logger.warning(String.format(" TT-OSRW optimization could not revert coordinates %16.8f vs. %16.8f.", e, eCheck));
}
}
}
use of ffx.crystal.Crystal in project ffx by mjschnie.
the class ParticleMeshEwaldQI method initSoftCore.
/**
* Initialize a boolean array of soft atoms and, if requested, ligand vapor
* electrostatics.
*/
private void initSoftCore(boolean print) {
if (initSoftCore) {
return;
}
/**
* Initialize a boolean array that marks soft atoms.
*/
StringBuilder sb = new StringBuilder("\n Softcore Atoms:\n");
int count = 0;
for (int i = 0; i < nAtoms; i++) {
Atom ai = atoms[i];
if (ai.applyLambda()) {
isSoft[i] = true;
if (print) {
sb.append(ai.toString()).append("\n");
}
count++;
}
}
if (count > 0 && print) {
logger.info(sb.toString());
}
/**
* Initialize boundary conditions, an n^2 neighbor list and parallel
* scheduling for ligand vapor electrostatics.
*/
if (doLigandVaporElec) {
double maxr = 10.0;
for (int i = 0; i < nAtoms; i++) {
Atom ai = atoms[i];
if (ai.applyLambda()) {
/**
* Determine ligand size.
*/
for (int j = i + 1; j < nAtoms; j++) {
Atom aj = atoms[j];
if (aj.applyLambda()) {
double dx = ai.getX() - aj.getX();
double dy = ai.getY() - aj.getY();
double dz = ai.getZ() - aj.getZ();
double r = sqrt(dx * dx + dy * dy + dz * dz);
maxr = max(r, maxr);
}
}
}
}
double vacuumOff = 2 * maxr;
vaporCrystal = new Crystal(3 * vacuumOff, 3 * vacuumOff, 3 * vacuumOff, 90.0, 90.0, 90.0, "P1");
vaporCrystal.setAperiodic(true);
NeighborList vacuumNeighborList = new NeighborList(null, vaporCrystal, atoms, vacuumOff, 2.0, parallelTeam);
vacuumNeighborList.setIntermolecular(false, molecule);
vaporLists = new int[1][nAtoms][];
double[][] coords = new double[1][nAtoms * 3];
for (int i = 0; i < nAtoms; i++) {
coords[0][i * 3] = atoms[i].getX();
coords[0][i * 3 + 1] = atoms[i].getY();
coords[0][i * 3 + 2] = atoms[i].getZ();
}
vacuumNeighborList.buildList(coords, vaporLists, isSoft, true, true);
vaporPermanentSchedule = vacuumNeighborList.getPairwiseSchedule();
vaporEwaldSchedule = vaporPermanentSchedule;
vacuumRanges = new Range[maxThreads];
vacuumNeighborList.setDisableUpdates(forceField.getBoolean(ForceField.ForceFieldBoolean.DISABLE_NEIGHBOR_UPDATES, false));
} else {
vaporCrystal = null;
vaporLists = null;
vaporPermanentSchedule = null;
vaporEwaldSchedule = null;
vacuumRanges = null;
}
/**
* Set this flag to true to avoid re-initialization.
*/
initSoftCore = true;
}
use of ffx.crystal.Crystal in project ffx by mjschnie.
the class MainPanel method savePDBSymMates.
public void savePDBSymMates(File file, String suffix) {
FFXSystem system = hierarchy.getActive();
if (system == null) {
logger.log(Level.INFO, " No active system to save.");
return;
}
if (system.isClosing()) {
logger.log(Level.INFO, " {0} is being closed and can no longer be saved.", system);
return;
}
File saveFile = file;
if (saveFile == null) {
resetFileChooser();
fileChooser.setCurrentDirectory(pwd);
fileChooser.setFileFilter(pdbFileFilter);
fileChooser.setAcceptAllFileFilterUsed(false);
int result = fileChooser.showSaveDialog(this);
if (result == JFileChooser.APPROVE_OPTION) {
saveFile = fileChooser.getSelectedFile();
pwd = saveFile.getParentFile();
}
}
if (saveFile == null) {
logger.log(Level.INFO, " No filename is defined for {0}.", system);
return;
}
String filename = FilenameUtils.removeExtension(file.getName());
PDBFilter pdbFilter = new PDBFilter(saveFile, system, null, null);
if (pdbFilter.writeFile(saveFile, false)) {
// Refresh Panels with the new System name
hierarchy.setActive(system);
activeFilter = pdbFilter;
} else {
logger.log(Level.INFO, " Save failed for: {0}", system);
}
Crystal crystal = system.getCrystal();
int nSymOps = crystal.spaceGroup.getNumberOfSymOps();
logger.info(String.format(" Writing %d symmetry mates for %s", nSymOps, system.toString()));
for (int i = 1; i < nSymOps; i++) {
pdbFilter.setSymOp(i);
String saveFileName = filename + suffix + "_" + i + ".pdb";
saveFile = new File(saveFileName);
for (int j = 1; j < 1000; j++) {
if (!saveFile.exists()) {
break;
}
saveFile = new File(saveFileName + "_" + j);
}
StringBuilder symSb = new StringBuilder();
String[] symopLines = crystal.spaceGroup.getSymOp(i).toString().split("\\r?\\n");
int nLines = symopLines.length;
symSb.append("REMARK 350\nREMARK 350 SYMMETRY OPERATORS");
for (int j = 0; j < nLines; j++) {
symSb.append("\nREMARK 350 ").append(symopLines[j]);
}
symopLines = crystal.spaceGroup.getSymOp(i).toXYZString().split("\\r?\\n");
nLines = symopLines.length;
symSb.append("\nREMARK 350\nREMARK 350 SYMMETRY OPERATORS XYZ FORM");
for (int j = 0; j < nLines; j++) {
symSb.append("\nREMARK 350 ").append(symopLines[j]);
}
if (saveFile.exists()) {
logger.warning(String.format(" Could not successfully version file " + "%s: appending to file %s", saveFileName, saveFile.getName()));
if (!pdbFilter.writeFileWithHeader(saveFile, symSb.toString(), true)) {
logger.log(Level.INFO, " Save failed for: {0}", system);
}
} else {
if (!pdbFilter.writeFileWithHeader(saveFile, symSb.toString(), false)) {
logger.log(Level.INFO, " Save failed for: {0}", system);
}
}
}
}
use of ffx.crystal.Crystal in project ffx by mjschnie.
the class PDBFilter method writeFile.
/**
* <p>
* writeFile</p>
*
* @param saveFile a {@link java.io.File} object.
* @param append a {@link java.lang.StringBuilder} object.
* @param printLinear Whether to print atoms linearly or by element
* @return Success of writing.
*/
public boolean writeFile(File saveFile, boolean append, boolean printLinear) {
if (Boolean.parseBoolean(System.getProperty("standardizeAtomNames", "false"))) {
renameAtomsToPDBStandard(activeMolecularAssembly);
}
if (saveFile == null) {
return false;
}
if (vdwH) {
logger.info(" Printing hydrogens to van der Waals centers instead of nuclear locations.");
}
if (nSymOp != 0) {
logger.info(String.format(" Printing atoms with symmetry operator %s\n", activeMolecularAssembly.getCrystal().spaceGroup.getSymOp(nSymOp).toString()));
}
/**
* Create StringBuilders for ATOM, ANISOU and TER records that can be
* reused.
*/
StringBuilder sb = new StringBuilder("ATOM ");
StringBuilder anisouSB = new StringBuilder("ANISOU");
StringBuilder terSB = new StringBuilder("TER ");
StringBuilder model = null;
for (int i = 6; i < 80; i++) {
sb.append(' ');
anisouSB.append(' ');
terSB.append(' ');
}
FileWriter fw;
BufferedWriter bw;
try {
File newFile = saveFile;
if (!append) {
if (!noVersioning) {
newFile = version(saveFile);
}
} else if (modelsWritten >= 0) {
model = new StringBuilder(String.format("MODEL %-4d", ++modelsWritten));
for (int i = 15; i < 80; i++) {
model.append(' ');
}
}
activeMolecularAssembly.setFile(newFile);
activeMolecularAssembly.setName(newFile.getName());
if (logWrites) {
logger.log(Level.INFO, " Saving {0}", activeMolecularAssembly.getName());
}
fw = new FileWriter(newFile, append);
bw = new BufferedWriter(fw);
/**
* Will come before CRYST1 and ATOM records, but after anything
* written by writeFileWithHeader (particularly X-ray refinement
* statistics).
*/
String[] headerLines = activeMolecularAssembly.getHeaderLines();
for (String line : headerLines) {
bw.write(String.format("%s\n", line));
}
if (model != null) {
if (!listMode) {
bw.write(model.toString());
bw.newLine();
} else {
listOutput.add(model.toString());
}
}
// =============================================================================
// The CRYST1 record presents the unit cell parameters, space group, and Z
// value. If the structure was not determined by crystallographic means, CRYST1
// simply provides the unitary values, with an appropriate REMARK.
//
// 7 - 15 Real(9.3) a a (Angstroms).
// 16 - 24 Real(9.3) b b (Angstroms).
// 25 - 33 Real(9.3) c c (Angstroms).
// 34 - 40 Real(7.2) alpha alpha (degrees).
// 41 - 47 Real(7.2) beta beta (degrees).
// 48 - 54 Real(7.2) gamma gamma (degrees).
// 56 - 66 LString sGroup Space group.
// 67 - 70 Integer z Z value.
// =============================================================================
Crystal crystal = activeMolecularAssembly.getCrystal();
if (crystal != null && !crystal.aperiodic()) {
Crystal c = crystal.getUnitCell();
if (!listMode) {
bw.write(format("CRYST1%9.3f%9.3f%9.3f%7.2f%7.2f%7.2f %10s\n", c.a, c.b, c.c, c.alpha, c.beta, c.gamma, padRight(c.spaceGroup.pdbName, 10)));
} else {
listOutput.add(format("CRYST1%9.3f%9.3f%9.3f%7.2f%7.2f%7.2f %10s", c.a, c.b, c.c, c.alpha, c.beta, c.gamma, padRight(c.spaceGroup.pdbName, 10)));
}
}
// =============================================================================
// The SSBOND record identifies each disulfide bond in protein and polypeptide
// structures by identifying the two residues involved in the bond.
// The disulfide bond distance is included after the symmetry operations at
// the end of the SSBOND record.
//
// 8 - 10 Integer serNum Serial number.
// 12 - 14 LString(3) "CYS" Residue name.
// 16 Character chainID1 Chain identifier.
// 18 - 21 Integer seqNum1 Residue sequence number.
// 22 AChar icode1 Insertion code.
// 26 - 28 LString(3) "CYS" Residue name.
// 30 Character chainID2 Chain identifier.
// 32 - 35 Integer seqNum2 Residue sequence number.
// 36 AChar icode2 Insertion code.
// 60 - 65 SymOP sym1 Symmetry oper for 1st resid
// 67 - 72 SymOP sym2 Symmetry oper for 2nd resid
// 74 – 78 Real(5.2) Length Disulfide bond distance
//
// If SG of cysteine is disordered then there are possible alternate linkages.
// wwPDB practice is to put together all possible SSBOND records. This is
// problematic because the alternate location identifier is not specified in
// the SSBOND record.
// =============================================================================
int serNum = 1;
Polymer[] polymers = activeMolecularAssembly.getChains();
if (polymers != null) {
for (Polymer polymer : polymers) {
ArrayList<Residue> residues = polymer.getResidues();
for (Residue residue : residues) {
if (residue.getName().equalsIgnoreCase("CYS")) {
List<Atom> cysAtoms = residue.getAtomList();
Atom SG1 = null;
for (Atom atom : cysAtoms) {
String atName = atom.getName().toUpperCase();
if (atName.equals("SG") || atName.equals("SH")) {
SG1 = atom;
break;
}
}
List<Bond> bonds = SG1.getBonds();
for (Bond bond : bonds) {
Atom SG2 = bond.get1_2(SG1);
if (SG2.getName().equalsIgnoreCase("SG")) {
if (SG1.getIndex() < SG2.getIndex()) {
bond.energy(false);
if (!listMode) {
bw.write(format("SSBOND %3d CYS %1s %4s CYS %1s %4s %36s %5.2f\n", serNum++, SG1.getChainID().toString(), Hybrid36.encode(4, SG1.getResidueNumber()), SG2.getChainID().toString(), Hybrid36.encode(4, SG2.getResidueNumber()), "", bond.getValue()));
} else {
listOutput.add(format("SSBOND %3d CYS %1s %4s CYS %1s %4s %36s %5.2f\n", serNum++, SG1.getChainID().toString(), Hybrid36.encode(4, SG1.getResidueNumber()), SG2.getChainID().toString(), Hybrid36.encode(4, SG2.getResidueNumber()), "", bond.getValue()));
}
}
}
}
}
}
}
}
// =============================================================================
//
// 7 - 11 Integer serial Atom serial number.
// 13 - 16 Atom name Atom name.
// 17 Character altLoc Alternate location indicator.
// 18 - 20 Residue name resName Residue name.
// 22 Character chainID Chain identifier.
// 23 - 26 Integer resSeq Residue sequence number.
// 27 AChar iCode Code for insertion of residues.
// 31 - 38 Real(8.3) x Orthogonal coordinates for X in Angstroms.
// 39 - 46 Real(8.3) y Orthogonal coordinates for Y in Angstroms.
// 47 - 54 Real(8.3) z Orthogonal coordinates for Z in Angstroms.
// 55 - 60 Real(6.2) occupancy Occupancy.
// 61 - 66 Real(6.2) tempFactor Temperature factor.
// 77 - 78 LString(2) element Element symbol, right-justified.
// 79 - 80 LString(2) charge Charge on the atom.
// =============================================================================
// 1 2 3 4 5 6 7
// 123456789012345678901234567890123456789012345678901234567890123456789012345678
// ATOM 1 N ILE A 16 60.614 71.140 -10.592 1.00 7.38 N
// ATOM 2 CA ILE A 16 60.793 72.149 -9.511 1.00 6.91 C
MolecularAssembly[] molecularAssemblies = this.getMolecularAssemblys();
int serial = 1;
// Loop over biomolecular chains
if (polymers != null) {
for (Polymer polymer : polymers) {
currentSegID = polymer.getName();
currentChainID = polymer.getChainID();
sb.setCharAt(21, currentChainID);
// Loop over residues
ArrayList<Residue> residues = polymer.getResidues();
for (Residue residue : residues) {
String resName = residue.getName();
if (resName.length() > 3) {
resName = resName.substring(0, 3);
}
int resID = residue.getResidueNumber();
sb.replace(17, 20, padLeft(resName.toUpperCase(), 3));
sb.replace(22, 26, String.format("%4s", Hybrid36.encode(4, resID)));
// Loop over atoms
ArrayList<Atom> residueAtoms = residue.getAtomList();
ArrayList<Atom> backboneAtoms = residue.getBackboneAtoms();
boolean altLocFound = false;
for (Atom atom : backboneAtoms) {
writeAtom(atom, serial++, sb, anisouSB, bw);
Character altLoc = atom.getAltLoc();
if (altLoc != null && !altLoc.equals(' ')) {
altLocFound = true;
}
residueAtoms.remove(atom);
}
for (Atom atom : residueAtoms) {
writeAtom(atom, serial++, sb, anisouSB, bw);
Character altLoc = atom.getAltLoc();
if (altLoc != null && !altLoc.equals(' ')) {
altLocFound = true;
}
}
// Write out alternate conformers
if (altLocFound) {
for (int ma = 1; ma < molecularAssemblies.length; ma++) {
MolecularAssembly altMolecularAssembly = molecularAssemblies[ma];
Polymer altPolymer = altMolecularAssembly.getPolymer(currentChainID, currentSegID, false);
Residue altResidue = altPolymer.getResidue(resName, resID, false);
backboneAtoms = altResidue.getBackboneAtoms();
residueAtoms = altResidue.getAtomList();
for (Atom atom : backboneAtoms) {
if (atom.getAltLoc() != null && !atom.getAltLoc().equals(' ') && !atom.getAltLoc().equals('A')) {
writeAtom(atom, serial++, sb, anisouSB, bw);
}
residueAtoms.remove(atom);
}
for (Atom atom : residueAtoms) {
if (atom.getAltLoc() != null && !atom.getAltLoc().equals(' ') && !atom.getAltLoc().equals('A')) {
writeAtom(atom, serial++, sb, anisouSB, bw);
}
}
}
}
}
terSB.replace(6, 11, String.format("%5s", Hybrid36.encode(5, serial++)));
terSB.replace(12, 16, " ");
terSB.replace(16, 26, sb.substring(16, 26));
if (!listMode) {
bw.write(terSB.toString());
bw.newLine();
} else {
listOutput.add(terSB.toString());
}
}
}
sb.replace(0, 6, "HETATM");
sb.setCharAt(21, 'A');
int resID = 1;
Polymer polymer = activeMolecularAssembly.getPolymer('A', "A", false);
if (polymer != null) {
ArrayList<Residue> residues = polymer.getResidues();
for (Residue residue : residues) {
int resID2 = residue.getResidueNumber();
if (resID2 >= resID) {
resID = resID2 + 1;
}
}
}
/**
* Loop over molecules, ions and then water.
*/
ArrayList<Molecule> molecules = activeMolecularAssembly.getMolecules();
for (int i = 0; i < molecules.size(); i++) {
Molecule molecule = (Molecule) molecules.get(i);
Character chainID = molecule.getChainID();
sb.setCharAt(21, chainID);
String resName = molecule.getResidueName();
if (resName.length() > 3) {
resName = resName.substring(0, 3);
}
sb.replace(17, 20, padLeft(resName.toUpperCase(), 3));
sb.replace(22, 26, String.format("%4s", Hybrid36.encode(4, resID)));
ArrayList<Atom> moleculeAtoms = molecule.getAtomList();
boolean altLocFound = false;
for (Atom atom : moleculeAtoms) {
writeAtom(atom, serial++, sb, anisouSB, bw);
Character altLoc = atom.getAltLoc();
if (altLoc != null && !altLoc.equals(' ')) {
altLocFound = true;
}
}
// Write out alternate conformers
if (altLocFound) {
for (int ma = 1; ma < molecularAssemblies.length; ma++) {
MolecularAssembly altMolecularAssembly = molecularAssemblies[ma];
MSNode altmolecule = altMolecularAssembly.getMolecules().get(i);
moleculeAtoms = altmolecule.getAtomList();
for (Atom atom : moleculeAtoms) {
if (atom.getAltLoc() != null && !atom.getAltLoc().equals(' ') && !atom.getAltLoc().equals('A')) {
writeAtom(atom, serial++, sb, anisouSB, bw);
}
}
}
}
resID++;
}
ArrayList<MSNode> ions = activeMolecularAssembly.getIons();
for (int i = 0; i < ions.size(); i++) {
Molecule ion = (Molecule) ions.get(i);
Character chainID = ion.getChainID();
sb.setCharAt(21, chainID);
String resName = ion.getResidueName();
if (resName.length() > 3) {
resName = resName.substring(0, 3);
}
sb.replace(17, 20, padLeft(resName.toUpperCase(), 3));
sb.replace(22, 26, String.format("%4s", Hybrid36.encode(4, resID)));
ArrayList<Atom> ionAtoms = ion.getAtomList();
boolean altLocFound = false;
for (Atom atom : ionAtoms) {
writeAtom(atom, serial++, sb, anisouSB, bw);
Character altLoc = atom.getAltLoc();
if (altLoc != null && !altLoc.equals(' ')) {
altLocFound = true;
}
}
// Write out alternate conformers
if (altLocFound) {
for (int ma = 1; ma < molecularAssemblies.length; ma++) {
MolecularAssembly altMolecularAssembly = molecularAssemblies[ma];
MSNode altion = altMolecularAssembly.getIons().get(i);
ionAtoms = altion.getAtomList();
for (Atom atom : ionAtoms) {
if (atom.getAltLoc() != null && !atom.getAltLoc().equals(' ') && !atom.getAltLoc().equals('A')) {
writeAtom(atom, serial++, sb, anisouSB, bw);
}
}
}
}
resID++;
}
ArrayList<MSNode> waters = activeMolecularAssembly.getWaters();
for (int i = 0; i < waters.size(); i++) {
Molecule water = (Molecule) waters.get(i);
Character chainID = water.getChainID();
sb.setCharAt(21, chainID);
String resName = water.getResidueName();
if (resName.length() > 3) {
resName = resName.substring(0, 3);
}
sb.replace(17, 20, padLeft(resName.toUpperCase(), 3));
sb.replace(22, 26, String.format("%4s", Hybrid36.encode(4, resID)));
ArrayList<Atom> waterAtoms = water.getAtomList();
boolean altLocFound = false;
for (Atom atom : waterAtoms) {
writeAtom(atom, serial++, sb, anisouSB, bw);
Character altLoc = atom.getAltLoc();
if (altLoc != null && !altLoc.equals(' ')) {
altLocFound = true;
}
}
// Write out alternate conformers
if (altLocFound) {
for (int ma = 1; ma < molecularAssemblies.length; ma++) {
MolecularAssembly altMolecularAssembly = molecularAssemblies[ma];
MSNode altwater = altMolecularAssembly.getWaters().get(i);
waterAtoms = altwater.getAtomList();
for (Atom atom : waterAtoms) {
if (atom.getAltLoc() != null && !atom.getAltLoc().equals(' ') && !atom.getAltLoc().equals('A')) {
writeAtom(atom, serial++, sb, anisouSB, bw);
}
}
}
}
resID++;
}
String end = model != null ? "ENDMDL" : "END";
if (!listMode) {
bw.write(end);
bw.newLine();
} else {
listOutput.add(end);
}
bw.close();
} catch (Exception e) {
String message = "Exception writing to file: " + saveFile.toString();
logger.log(Level.WARNING, message, e);
return false;
}
return true;
}
use of ffx.crystal.Crystal in project ffx by mjschnie.
the class PDBFilter method writeSIFTFile.
public boolean writeSIFTFile(File saveFile, boolean append, String[] resAndScore) {
if (saveFile == null) {
return false;
}
if (vdwH) {
logger.info(" Printing hydrogens to van der Waals centers instead of nuclear locations.");
}
if (nSymOp != 0) {
logger.info(String.format(" Printing atoms with symmetry operator %s", activeMolecularAssembly.getCrystal().spaceGroup.getSymOp(nSymOp).toString()));
}
/**
* Create StringBuilders for ATOM, ANISOU and TER records that can be
* reused.
*/
StringBuilder sb = new StringBuilder("ATOM ");
StringBuilder anisouSB = new StringBuilder("ANISOU");
StringBuilder terSB = new StringBuilder("TER ");
for (int i = 6; i < 80; i++) {
sb.append(' ');
anisouSB.append(' ');
terSB.append(' ');
}
FileWriter fw;
BufferedWriter bw;
try {
File newFile = saveFile;
if (!append && !noVersioning) {
newFile = version(saveFile);
}
activeMolecularAssembly.setFile(newFile);
activeMolecularAssembly.setName(newFile.getName());
if (logWrites) {
logger.log(Level.INFO, " Saving {0}", newFile.getName());
}
fw = new FileWriter(newFile, append);
bw = new BufferedWriter(fw);
// =============================================================================
// The CRYST1 record presents the unit cell parameters, space group, and Z
// value. If the structure was not determined by crystallographic means, CRYST1
// simply provides the unitary values, with an appropriate REMARK.
//
// 7 - 15 Real(9.3) a a (Angstroms).
// 16 - 24 Real(9.3) b b (Angstroms).
// 25 - 33 Real(9.3) c c (Angstroms).
// 34 - 40 Real(7.2) alpha alpha (degrees).
// 41 - 47 Real(7.2) beta beta (degrees).
// 48 - 54 Real(7.2) gamma gamma (degrees).
// 56 - 66 LString sGroup Space group.
// 67 - 70 Integer z Z value.
// =============================================================================
Crystal crystal = activeMolecularAssembly.getCrystal();
if (crystal != null && !crystal.aperiodic()) {
Crystal c = crystal.getUnitCell();
if (!listMode) {
bw.write(format("CRYST1%9.3f%9.3f%9.3f%7.2f%7.2f%7.2f %10s\n", c.a, c.b, c.c, c.alpha, c.beta, c.gamma, padRight(c.spaceGroup.pdbName, 10)));
} else {
listOutput.add(format("CRYST1%9.3f%9.3f%9.3f%7.2f%7.2f%7.2f %10s", c.a, c.b, c.c, c.alpha, c.beta, c.gamma, padRight(c.spaceGroup.pdbName, 10)));
}
}
// =============================================================================
// The SSBOND record identifies each disulfide bond in protein and polypeptide
// structures by identifying the two residues involved in the bond.
// The disulfide bond distance is included after the symmetry operations at
// the end of the SSBOND record.
//
// 8 - 10 Integer serNum Serial number.
// 12 - 14 LString(3) "CYS" Residue name.
// 16 Character chainID1 Chain identifier.
// 18 - 21 Integer seqNum1 Residue sequence number.
// 22 AChar icode1 Insertion code.
// 26 - 28 LString(3) "CYS" Residue name.
// 30 Character chainID2 Chain identifier.
// 32 - 35 Integer seqNum2 Residue sequence number.
// 36 AChar icode2 Insertion code.
// 60 - 65 SymOP sym1 Symmetry oper for 1st resid
// 67 - 72 SymOP sym2 Symmetry oper for 2nd resid
// 74 – 78 Real(5.2) Length Disulfide bond distance
//
// If SG of cysteine is disordered then there are possible alternate linkages.
// wwPDB practice is to put together all possible SSBOND records. This is
// problematic because the alternate location identifier is not specified in
// the SSBOND record.
// =============================================================================
int serNum = 1;
Polymer[] polymers = activeMolecularAssembly.getChains();
if (polymers != null) {
for (Polymer polymer : polymers) {
ArrayList<Residue> residues = polymer.getResidues();
for (Residue residue : residues) {
if (residue.getName().equalsIgnoreCase("CYS")) {
List<Atom> cysAtoms = residue.getAtomList();
Atom SG1 = null;
for (Atom atom : cysAtoms) {
if (atom.getName().equalsIgnoreCase("SG")) {
SG1 = atom;
break;
}
}
List<Bond> bonds = SG1.getBonds();
for (Bond bond : bonds) {
Atom SG2 = bond.get1_2(SG1);
if (SG2.getName().equalsIgnoreCase("SG")) {
if (SG1.getIndex() < SG2.getIndex()) {
bond.energy(false);
if (!listMode) {
bw.write(format("SSBOND %3d CYS %1s %4s CYS %1s %4s %36s %5.2f\n", serNum++, SG1.getChainID().toString(), Hybrid36.encode(4, SG1.getResidueNumber()), SG2.getChainID().toString(), Hybrid36.encode(4, SG2.getResidueNumber()), "", bond.getValue()));
} else {
listOutput.add(format("SSBOND %3d CYS %1s %4s CYS %1s %4s %36s %5.2f\n", serNum++, SG1.getChainID().toString(), Hybrid36.encode(4, SG1.getResidueNumber()), SG2.getChainID().toString(), Hybrid36.encode(4, SG2.getResidueNumber()), "", bond.getValue()));
}
}
}
}
}
}
}
}
// =============================================================================
//
// 7 - 11 Integer serial Atom serial number.
// 13 - 16 Atom name Atom name.
// 17 Character altLoc Alternate location indicator.
// 18 - 20 Residue name resName Residue name.
// 22 Character chainID Chain identifier.
// 23 - 26 Integer resSeq Residue sequence number.
// 27 AChar iCode Code for insertion of residues.
// 31 - 38 Real(8.3) x Orthogonal coordinates for X in Angstroms.
// 39 - 46 Real(8.3) y Orthogonal coordinates for Y in Angstroms.
// 47 - 54 Real(8.3) z Orthogonal coordinates for Z in Angstroms.
// 55 - 60 Real(6.2) occupancy Occupancy.
// 61 - 66 Real(6.2) tempFactor Temperature factor.
// 77 - 78 LString(2) element Element symbol, right-justified.
// 79 - 80 LString(2) charge Charge on the atom.
// =============================================================================
// 1 2 3 4 5 6 7
// 123456789012345678901234567890123456789012345678901234567890123456789012345678
// ATOM 1 N ILE A 16 60.614 71.140 -10.592 1.00 7.38 N
// ATOM 2 CA ILE A 16 60.793 72.149 -9.511 1.00 6.91 C
MolecularAssembly[] molecularAssemblies = this.getMolecularAssemblys();
int serial = 1;
// Loop over biomolecular chains
if (polymers != null) {
for (Polymer polymer : polymers) {
currentSegID = polymer.getName();
currentChainID = polymer.getChainID();
sb.setCharAt(21, currentChainID);
// Loop over residues
ArrayList<Residue> residues = polymer.getResidues();
for (Residue residue : residues) {
String resName = residue.getName();
if (resName.length() > 3) {
resName = resName.substring(0, 3);
}
int resID = residue.getResidueNumber();
int i = 0;
String[] entries = null;
for (; i < resAndScore.length; i++) {
entries = resAndScore[i].split("\\t");
if (!entries[0].equals(entries[0].replaceAll("\\D+", ""))) {
String[] subEntries = entries[0].split("[^0-9]");
entries[0] = subEntries[0];
}
if (entries[0].equals(String.valueOf(resID)) && !".".equals(entries[1])) {
break;
}
}
sb.replace(17, 20, padLeft(resName.toUpperCase(), 3));
sb.replace(22, 26, String.format("%4s", Hybrid36.encode(4, resID)));
// Loop over atoms
ArrayList<Atom> residueAtoms = residue.getAtomList();
boolean altLocFound = false;
for (Atom atom : residueAtoms) {
if (i != resAndScore.length) {
writeSIFTAtom(atom, serial++, sb, anisouSB, bw, entries[1]);
} else {
writeSIFTAtom(atom, serial++, sb, anisouSB, bw, null);
}
Character altLoc = atom.getAltLoc();
if (altLoc != null && !altLoc.equals(' ')) {
altLocFound = true;
}
}
// Write out alternate conformers
if (altLocFound) {
for (int ma = 1; ma < molecularAssemblies.length; ma++) {
MolecularAssembly altMolecularAssembly = molecularAssemblies[ma];
Polymer altPolymer = altMolecularAssembly.getPolymer(currentChainID, currentSegID, false);
Residue altResidue = altPolymer.getResidue(resName, resID, false);
residueAtoms = altResidue.getAtomList();
for (Atom atom : residueAtoms) {
if (atom.getAltLoc() != null && !atom.getAltLoc().equals(' ') && !atom.getAltLoc().equals('A')) {
if (i != resAndScore.length) {
writeSIFTAtom(atom, serial++, sb, anisouSB, bw, entries[1]);
} else {
writeSIFTAtom(atom, serial++, sb, anisouSB, bw, null);
}
}
}
}
}
}
terSB.replace(6, 11, String.format("%5s", Hybrid36.encode(5, serial++)));
terSB.replace(12, 16, " ");
terSB.replace(16, 26, sb.substring(16, 26));
if (!listMode) {
bw.write(terSB.toString());
bw.newLine();
} else {
listOutput.add(terSB.toString());
}
}
}
sb.replace(0, 6, "HETATM");
sb.setCharAt(21, 'A');
int resID = 1;
Polymer polymer = activeMolecularAssembly.getPolymer('A', "A", false);
if (polymer != null) {
ArrayList<Residue> residues = polymer.getResidues();
for (Residue residue : residues) {
int resID2 = residue.getResidueNumber();
if (resID2 >= resID) {
resID = resID2 + 1;
}
}
}
/**
* Loop over molecules, ions and then water.
*/
ArrayList<Molecule> molecules = activeMolecularAssembly.getMolecules();
for (int i = 0; i < molecules.size(); i++) {
Molecule molecule = (Molecule) molecules.get(i);
Character chainID = molecule.getChainID();
sb.setCharAt(21, chainID);
String resName = molecule.getResidueName();
if (resName.length() > 3) {
resName = resName.substring(0, 3);
}
sb.replace(17, 20, padLeft(resName.toUpperCase(), 3));
sb.replace(22, 26, String.format("%4s", Hybrid36.encode(4, resID)));
ArrayList<Atom> moleculeAtoms = molecule.getAtomList();
boolean altLocFound = false;
for (Atom atom : moleculeAtoms) {
writeSIFTAtom(atom, serial++, sb, anisouSB, bw, null);
Character altLoc = atom.getAltLoc();
if (altLoc != null && !altLoc.equals(' ')) {
altLocFound = true;
}
}
// Write out alternate conformers
if (altLocFound) {
for (int ma = 1; ma < molecularAssemblies.length; ma++) {
MolecularAssembly altMolecularAssembly = molecularAssemblies[ma];
MSNode altmolecule = altMolecularAssembly.getMolecules().get(i);
moleculeAtoms = altmolecule.getAtomList();
for (Atom atom : moleculeAtoms) {
if (atom.getAltLoc() != null && !atom.getAltLoc().equals(' ') && !atom.getAltLoc().equals('A')) {
writeSIFTAtom(atom, serial++, sb, anisouSB, bw, null);
}
}
}
}
resID++;
}
ArrayList<MSNode> ions = activeMolecularAssembly.getIons();
for (int i = 0; i < ions.size(); i++) {
Molecule ion = (Molecule) ions.get(i);
Character chainID = ion.getChainID();
sb.setCharAt(21, chainID);
String resName = ion.getResidueName();
if (resName.length() > 3) {
resName = resName.substring(0, 3);
}
sb.replace(17, 20, padLeft(resName.toUpperCase(), 3));
sb.replace(22, 26, String.format("%4s", Hybrid36.encode(4, resID)));
ArrayList<Atom> ionAtoms = ion.getAtomList();
boolean altLocFound = false;
for (Atom atom : ionAtoms) {
writeSIFTAtom(atom, serial++, sb, anisouSB, bw, null);
Character altLoc = atom.getAltLoc();
if (altLoc != null && !altLoc.equals(' ')) {
altLocFound = true;
}
}
// Write out alternate conformers
if (altLocFound) {
for (int ma = 1; ma < molecularAssemblies.length; ma++) {
MolecularAssembly altMolecularAssembly = molecularAssemblies[ma];
MSNode altion = altMolecularAssembly.getIons().get(i);
ionAtoms = altion.getAtomList();
for (Atom atom : ionAtoms) {
if (atom.getAltLoc() != null && !atom.getAltLoc().equals(' ') && !atom.getAltLoc().equals('A')) {
writeSIFTAtom(atom, serial++, sb, anisouSB, bw, null);
}
}
}
}
resID++;
}
ArrayList<MSNode> waters = activeMolecularAssembly.getWaters();
for (int i = 0; i < waters.size(); i++) {
Molecule water = (Molecule) waters.get(i);
Character chainID = water.getChainID();
sb.setCharAt(21, chainID);
String resName = water.getResidueName();
if (resName.length() > 3) {
resName = resName.substring(0, 3);
}
sb.replace(17, 20, padLeft(resName.toUpperCase(), 3));
sb.replace(22, 26, String.format("%4s", Hybrid36.encode(4, resID)));
ArrayList<Atom> waterAtoms = water.getAtomList();
boolean altLocFound = false;
for (Atom atom : waterAtoms) {
writeSIFTAtom(atom, serial++, sb, anisouSB, bw, null);
Character altLoc = atom.getAltLoc();
if (altLoc != null && !altLoc.equals(' ')) {
altLocFound = true;
}
}
// Write out alternate conformers
if (altLocFound) {
for (int ma = 1; ma < molecularAssemblies.length; ma++) {
MolecularAssembly altMolecularAssembly = molecularAssemblies[ma];
MSNode altwater = altMolecularAssembly.getWaters().get(i);
waterAtoms = altwater.getAtomList();
for (Atom atom : waterAtoms) {
if (atom.getAltLoc() != null && !atom.getAltLoc().equals(' ') && !atom.getAltLoc().equals('A')) {
writeSIFTAtom(atom, serial++, sb, anisouSB, bw, null);
}
}
}
}
resID++;
}
if (!listMode) {
bw.write("END");
bw.newLine();
} else {
listOutput.add("END");
}
bw.close();
} catch (Exception e) {
String message = "Exception writing to file: " + saveFile.toString();
logger.log(Level.WARNING, message, e);
return false;
}
return true;
}
Aggregations