Files
rdkit/Code/JavaWrappers/Depict.h
2011-06-20 03:01:34 +00:00

169 lines
6.2 KiB
C++

/*
*
* Copyright (c) 2011, Novartis Institutes for BioMedical Research Inc.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following
* disclaimer in the documentation and/or other materials provided
* with the distribution.
* * Neither the name of Novartis Institutes for BioMedical Research Inc.
* nor the names of its contributors may be used to endorse or promote
* products derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include <Demos/RDKit/Draw/MolDrawing.h>
namespace {
std::string getColor(int atNum){
static std::map<int,std::string> colors;
if(colors.empty()){
colors[7]="#0000FF";
colors[8]="#FF0000";
colors[9]="#33CCCC";
colors[15]="#FF7F00";
colors[16]="#CCCC00";
colors[17]="#00CC00";
colors[35]="#7F4C19";
colors[0]="#7F7F7F";
}
std::string res="#000000";
if(colors.find(atNum)!=colors.end()) res= colors[atNum];
return res;
}
void drawLine(std::vector<int>::const_iterator &pos,std::ostringstream &sstr,
unsigned int lineWidthMult){
int width=*pos++;
width*=lineWidthMult;
int dashed=*pos++;
std::string dashString="";
if(dashed){
dashString=";stroke-dasharray:6, 6";
}
int an1=*pos++;
int an2=*pos++;
std::string c1=getColor(an1);
std::string c2=getColor(an2);
if(c1==c2){
sstr<<"<svg:path ";
sstr<<"d='M "<<*pos<<","<<*(pos+1)<<" "<<*(pos+2)<<","<<*(pos+3)<<"' ";
pos+=4;
sstr<<"style='fill:none;fill-rule:evenodd;stroke:"<<c1<<";stroke-width:"<<width<<"px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1"<<dashString<<"'";
sstr<<" />\n";
} else {
int xp1 = *pos++;
int yp1 = *pos++;
int xp2 = *pos++;
int yp2 = *pos++;
int mx = xp1+(xp2-xp1)/2;
int my = yp1+(yp2-yp1)/2;
sstr<<"<svg:path ";
sstr<<"d='M "<<xp1<<","<<yp1<<" "<<mx<<","<<my<<"' ";
sstr<<"style='fill:none;fill-rule:evenodd;stroke:"<<c1<<";stroke-width:"<<width<<"px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1"<<dashString<<"'";
sstr<<" />\n";
sstr<<"<svg:path ";
sstr<<"d='M "<<mx<<","<<my<<" "<<xp2<<","<<yp2<<"' ";
sstr<<"style='fill:none;fill-rule:evenodd;stroke:"<<c2<<";stroke-width:"<<width<<"px;stroke-linecap:butt;stroke-linejoin:miter;stroke-opacity:1"<<dashString<<"'";
sstr<<" />\n";
}
}
void drawAtom(std::vector<int>::const_iterator &pos,std::ostringstream &sstr,unsigned int fontSz){
int atNum=*pos++;
int xp=*pos++;
int yp=*pos++;
int slen=*pos++;
std::string label="";
for(unsigned int i=0;i<slen;++i){
label+=(char)*pos++;
}
int orient=*pos++;
int width=fontSz*label.length();
int height=fontSz;
sstr<<"<svg:g transform='translate("<<xp<<","<<yp<<")'><svg:rect ";
sstr<<"style='opacity:1.0;fill:#FFFFFF;stroke:none'";
sstr<<" width='"<<width<<"' height='"<<height<<"'";
sstr<<" x='-"<<width/2<<"' y='-"<<height/2<<"'";
sstr<<"> </svg:rect>\n";
sstr<<"<svg:text";
sstr<<" style='font-size:"<<fontSz<<"px;font-style:normal;font-weight:normal;line-height:125%;letter-spacing:0px;word-spacing:0px;fill-opacity:1;stroke:none;font-family:Sans;text-anchor:middle"<<";fill:"<<getColor(atNum)<<"'";
sstr<<" y='"<<.75*fontSz/2<<"'>";
sstr<<"<svg:tspan>";
sstr<<label<<"</svg:tspan>";
sstr<<"</svg:text>";
sstr<<"</svg:g>\n";
}
std::string DrawingToSVG(const std::vector<int> &drawing,
unsigned int lineWidthMult=2,unsigned int fontSize=50){
std::vector<int>::const_iterator pos=drawing.begin()+2;
if(*pos!= RDKit::Drawing::BOUNDS){
std::cerr<<"no bounds token found"<<std::endl;
return "";
}
pos+=3;
int width=300,height=300;
width = *pos++;
height = *pos++;
std::ostringstream sstr;
sstr<<"<?xml version='1.0' encoding='iso-8859-1'?>\n";
sstr << "<svg:svg version='1.1' baseProfile='full'\n \
xmlns:svg='http://www.w3.org/2000/svg'\n \
xmlns:xlink='http://www.w3.org/1999/xlink'\n \
xml:space='preserve'\n";
sstr<<"width='"<<width<<"px' height='"<<height<<"px' >\n";
sstr<<"<svg:g transform='translate("<<width*.05<<","<<height*.05<<") scale(.9,.9)'>";
while(pos!=drawing.end()){
int token=*pos++;
switch(token){
case RDKit::Drawing::LINE:
drawLine(pos,sstr,lineWidthMult);
break;
case RDKit::Drawing::ATOM:
drawAtom(pos,sstr,fontSize);
break;
default:
std::cerr<<"unrecognized token: "<<token<<std::endl;
}
}
sstr<<"</svg:g></svg:svg>";
return sstr.str();
}
}
std::vector<int> MolToDrawing(const RDKit::ROMol &mol,const std::vector<int> *highlightAtoms=0){
RDKit::RWMol *cp = new RDKit::RWMol(mol);
try{
RDKit::MolOps::Kekulize(*cp);
} catch (...) {
delete cp;
cp = new RDKit::RWMol(mol);
}
if(!mol.getNumConformers()) {
RDDepict::compute2DCoords(*cp);
}
std::vector<int> drawing=RDKit::Drawing::DrawMol(*cp,-1,highlightAtoms);
delete cp;
return drawing;
}