00001 
00002 
00003 
00004 
00005 
00006 
00007 
00008 
00009 
00010 
00011 
00012 
00013 
00014 
00015 
00021 #include <math.h>
00022 #include <string.h>
00023 #include <stdlib.h>
00024 #include <stdio.h>
00025 #include "AtomColor.h"
00026 #include "DrawMolecule.h"
00027 #include "MoleculeList.h"
00028 #include "Scene.h"
00029 #include "VolumetricData.h"
00030 #include "PeriodicTable.h"
00031 #include "Inform.h"
00032 #include "utilities.h"
00033 #include "config.h"
00034 
00038 const char *AtomColorName[AtomColor::TOTAL] = { 
00039   "Name",     "Type",     "Element",
00040   "ResName",  "ResType",  "ResID",    
00041   "Chain",    "SegName",  "Conformation",
00042   "Molecule", "Structure", 
00043   "ColorID",  "Beta",     "Occupancy", 
00044   "Mass",     "Charge",   
00045   "Pos",      "PosX",     "PosY",  "PosZ", 
00046   "User",     "User2",    "User3", "User4", "Fragment",
00047   "Index",    "Backbone", "Throb",
00048   "PhysicalTime", "Timestep", "Velocity", "Volume"
00049   };
00050 
00052 const char *AtomColorMenuName[AtomColor::TOTAL] = { 
00053   "Name",          "Type",          "Element",
00054   "ResName",       "ResType",       "ResID",    
00055   "Chain",         "SegName",       "Conformation",
00056   "Molecule",      "Secondary Structure", 
00057   "ColorID",       "Beta",          "Occupancy", 
00058   "Mass",          "Charge",   
00059   "Position/Radial", "Position/X", "Position/Y", "Position/Z", 
00060   "Trajectory/User/User", "Trajectory/User/User2", "Trajectory/User/User3", "Trajectory/User/User4",
00061   "Fragment", "Index",  "Backbone", "Throb",
00062   "Trajectory/Physical Time", "Trajectory/Timestep", "Trajectory/Velocity", "Volume"
00063   };
00064 
00065 
00067 
00068 AtomColor::AtomColor(MoleculeList *mlist, const Scene *sc) {
00069 
00070   
00071   molList = mlist;
00072   scene = sc;
00073   color = NULL;
00074   colIndex = 0;
00075   volIndex = 0;
00076   nAtoms = 0;
00077   mol = NULL;
00078   colorMethod  = DEFAULT_ATOMCOLOR;
00079   strcpy(cmdStr, AtomColorName[DEFAULT_ATOMCOLOR]);
00080   do_update = 0;
00081   need_recalc_minmax = TRUE;
00082   minRange = maxRange = 0;
00083 }
00084 
00085 
00086 
00087 AtomColor::AtomColor(AtomColor& ar) {
00088   strcpy(cmdStr, ar.cmdStr);
00089   colorMethod = ar.colorMethod;
00090   colIndex = ar.colIndex;
00091   volIndex = ar.volIndex;
00092   scene = ar.scene;
00093   molList = ar.molList;
00094   mol = ar.mol;
00095   if(ar.color) {
00096     nAtoms = ar.nAtoms;
00097     color = new int[nAtoms];
00098     for(int i=0; i < nAtoms; i++)
00099       color[i] = ar.color[i];
00100   } else {
00101     nAtoms = 0;
00102     color = NULL;
00103   }
00104   do_update = ar.do_update;
00105   need_recalc_minmax = TRUE;
00106   minRange = maxRange = 0;
00107 }
00108 
00109 
00110 
00111 AtomColor::~AtomColor(void) {
00112   if(color)
00113     delete [] color;
00114 }
00115 
00116 
00118 
00119 
00120 int AtomColor::parse_cmd(const char *newcmd) {
00121   int argc, ok = TRUE;
00122   char *argv[128] = { NULL };
00123   char *cmdStrTok = NULL;
00124 
00125   ColorMethod newMethod = colorMethod;
00126   int newIndex = colIndex;
00127   int newVol   = volIndex;
00128  
00129   
00130   if(newcmd && strlen(newcmd) > MAX_ATOMCOLOR_CMD) {
00131     msgErr << "Atom coloring method string is too long (over ";
00132     msgErr << MAX_ATOMCOLOR_CMD << " characters)." << sendmsg;
00133     return FALSE;
00134   }
00135 
00136   
00137   if(!newcmd || !(cmdStrTok = str_tokenize(newcmd, &argc, argv))) {
00138     
00139     return TRUE;
00140   }
00141 
00142   
00143   if(!strupncmp(argv[0], "default", CMDLEN)) {
00144     newMethod = DEFAULT_ATOMCOLOR;
00145   } else if(!strupncmp(argv[0], AtomColorName[NAME], CMDLEN)) {
00146     newMethod = NAME;
00147   } else if(!strupncmp(argv[0], AtomColorName[TYPE], CMDLEN)) {
00148     newMethod = TYPE;
00149   } else if(!strupncmp(argv[0], AtomColorName[ELEMENT], CMDLEN)) {
00150     newMethod = ELEMENT;
00151   } else if(!strupncmp(argv[0], AtomColorName[RESNAME], CMDLEN)) {
00152     newMethod = RESNAME;
00153   } else if(!strupncmp(argv[0], AtomColorName[RESTYPE], CMDLEN)) {
00154     newMethod = RESTYPE;
00155   } else if(!strupncmp(argv[0], AtomColorName[RESID], CMDLEN)) {
00156     newMethod = RESID;
00157   } else if(!strupncmp(argv[0], AtomColorName[CHAIN], CMDLEN)) {
00158     newMethod = CHAIN;
00159   } else if(!strupncmp(argv[0], AtomColorName[SEGNAME], CMDLEN)) {
00160     newMethod = SEGNAME;
00161   } else if(!strupncmp(argv[0], AtomColorName[CONFORMATION], CMDLEN)) {
00162     newMethod = CONFORMATION;
00163   } else if(!strupncmp(argv[0], AtomColorName[MOLECULE], CMDLEN)) {
00164     newMethod = MOLECULE;
00165   } else if(!strupncmp(argv[0], AtomColorName[STRUCTURE], CMDLEN)) {
00166     newMethod = STRUCTURE;
00167   } else if(!strupncmp(argv[0], AtomColorName[COLORID], CMDLEN) && argc > 1) {
00168     newMethod = COLORID;
00169     newIndex = atoi(argv[1]);
00170   } else if(!strupncmp(argv[0], AtomColorName[BETA], CMDLEN)) {
00171     newMethod = BETA;
00172   } else if(!strupncmp(argv[0], AtomColorName[OCCUP], CMDLEN)) {
00173     newMethod = OCCUP;
00174   } else if(!strupncmp(argv[0], AtomColorName[MASS], CMDLEN)) {
00175     newMethod = MASS;
00176   } else if(!strupncmp(argv[0], AtomColorName[CHARGE], CMDLEN)) {
00177     newMethod = CHARGE;
00178   } else if(!strupncmp(argv[0], AtomColorName[USER], CMDLEN)) {
00179     newMethod = USER;
00180   } else if(!strupncmp(argv[0], AtomColorName[USER2], CMDLEN)) {
00181     newMethod = USER2;
00182   } else if(!strupncmp(argv[0], AtomColorName[USER3], CMDLEN)) {
00183     newMethod = USER3;
00184   } else if(!strupncmp(argv[0], AtomColorName[USER4], CMDLEN)) {
00185     newMethod = USER4;
00186   } else if(!strupncmp(argv[0], AtomColorName[FRAGMENT], CMDLEN)) {
00187     newMethod = FRAGMENT;
00188   } else if(!strupncmp(argv[0], AtomColorName[POS], CMDLEN)) {
00189     newMethod = POS;
00190   } else if(!strupncmp(argv[0], AtomColorName[POSX], CMDLEN)) {
00191     newMethod = POSX;
00192   } else if(!strupncmp(argv[0], AtomColorName[POSY], CMDLEN)) {
00193     newMethod = POSY;
00194   } else if(!strupncmp(argv[0], AtomColorName[POSZ], CMDLEN)) {
00195     newMethod = POSZ;
00196   } else if(!strupncmp(argv[0], AtomColorName[INDEX], CMDLEN)) {
00197     newMethod = INDEX;
00198   } else if(!strupncmp(argv[0], AtomColorName[BACKBONE], CMDLEN)) {
00199     newMethod = BACKBONE;
00200   } else if(!strupncmp(argv[0], AtomColorName[THROB], CMDLEN)) {
00201     newMethod = THROB;
00202   } else if(!strupncmp(argv[0], AtomColorName[PHYSICALTIME], CMDLEN)) {
00203     newMethod = PHYSICALTIME;
00204   } else if(!strupncmp(argv[0], AtomColorName[TIMESTEP], CMDLEN)) {
00205     newMethod = TIMESTEP;
00206   } else if(!strupncmp(argv[0], AtomColorName[VELOCITY], CMDLEN)) {
00207     newMethod = VELOCITY;
00208   } else if(!strupncmp(argv[0], AtomColorName[VOLUME], CMDLEN) && argc > 1) {
00209     newMethod = VOLUME;
00210     newVol = atoi(argv[1]);
00211   } else {
00212     
00213     ok = FALSE;
00214   }
00215 
00216   
00217   ok = ok && ((((newMethod == COLORID) || 
00218                 (newMethod == VOLUME)) && argc < 4) || 
00219                 (newMethod != COLORID && argc < 3));
00220 
00221   
00222   if (!ok) {
00223     msgErr << "Incorrect atom color method command '" << newcmd << "'";
00224     msgErr << sendmsg;
00225   } else {
00226     
00227     colorMethod = newMethod;
00228     colIndex = newIndex;
00229     volIndex = newVol;
00230     strcpy(cmdStr, newcmd);
00231 
00232     
00233     if (colIndex < 0)
00234       colIndex = 0;
00235     else if (colIndex > VISCLRS)
00236       colIndex = VISCLRS;
00237 
00238     
00239     if (volIndex < 0)
00240       volIndex = 0;
00241 
00242     need_recalc_minmax = TRUE;
00243   }
00244 
00245   
00246   delete [] cmdStrTok;
00247   return ok;
00248 }
00249 
00250 
00252 
00253 
00254 
00255 AtomColor& AtomColor::operator=(const AtomColor &ar) {
00256 
00257   
00258   if(cmdStr != ar.cmdStr)
00259     strcpy(cmdStr, ar.cmdStr);
00260   colorMethod = ar.colorMethod;
00261   colIndex = ar.colIndex;
00262   volIndex = ar.volIndex;
00263   
00264   
00265   need_recalc_minmax = TRUE;
00266   find(mol);  
00267     
00268   return *this;
00269 }
00270 
00271 int AtomColor::current_color_use(int ccat) {
00272 
00273   if((colorMethod==MOLECULE&& ccat==molList->colorCatIndex[MLCAT_MOLECULES])
00274     || (colorMethod==CONFORMATION&& ccat==molList->colorCatIndex[MLCAT_CONFORMATIONS])
00275     || (colorMethod==NAME&& ccat==molList->colorCatIndex[MLCAT_NAMES])
00276     || (colorMethod==TYPE&& ccat==molList->colorCatIndex[MLCAT_TYPES])
00277     || (colorMethod==ELEMENT&& ccat==molList->colorCatIndex[MLCAT_ELEMENTS])
00278     || (colorMethod==RESNAME&& ccat==molList->colorCatIndex[MLCAT_RESNAMES])
00279     || (colorMethod==RESTYPE&& ccat==molList->colorCatIndex[MLCAT_RESTYPES])
00280     || (colorMethod==CHAIN&& ccat==molList->colorCatIndex[MLCAT_CHAINS])
00281     || (colorMethod==SEGNAME&& ccat==molList->colorCatIndex[MLCAT_SEGNAMES])
00282     || (colorMethod==BACKBONE&& ccat==molList->colorCatIndex[MLCAT_SPECIAL])
00283     || (colorMethod==STRUCTURE&& ccat==molList->colorCatIndex[MLCAT_SSTRUCT])
00284     ) {
00285       return TRUE;
00286   }
00287   
00288   return FALSE;
00289 }
00290 
00291 
00292 int AtomColor::find(DrawMolecule *m) {
00293   int dindex;
00294 
00295   
00296   if(!m || !molList)
00297     return FALSE;
00298     
00299   
00300   if(color && mol && nAtoms != m->nAtoms) {
00301     delete [] color;
00302     color = NULL;
00303   }
00304 
00305   
00306   mol = m;
00307   nAtoms = m->nAtoms;
00308   if(!color)
00309     color = new int[nAtoms];
00310 
00311   if (need_recalc_minmax) {
00312     
00313     minRange = 0.;  
00314     maxRange = 0.;
00315   }
00316   
00317   
00318   if(colorMethod == MOLECULE) {
00319     char buf[20];
00320     sprintf(buf, "%d", mol->id());
00321     dindex = scene->category_item_value(molList->colorCatIndex[MLCAT_MOLECULES],
00322         buf);
00323     colIndex = dindex;
00324 
00325     for(int i=0; i < nAtoms; color[i++] = dindex) ;
00326 
00327   } else if(colorMethod == COLORID) {
00328     dindex = colIndex;
00329     for(int i=0; i < nAtoms; i++) {
00330       color[i] = dindex;
00331     }
00332   } else if(colorMethod == THROB) {
00333     double timeval, t;
00334     t = time_of_day();
00335     timeval = fmod((t / 2.0), 1.0) * 254; 
00336 
00337     
00338     float scalefac = (float)(MAPCLRS-1) / 255.0f;
00339     dindex = MAPCOLOR((int)(0.5 + timeval*scalefac));
00340 
00341     for(int i=0; i < nAtoms; i++) {
00342       color[i] = dindex;
00343     }
00344   } else if(colorMethod == PHYSICALTIME) {
00345     dindex = MAPCOLOR(MAPCLRS/2);
00346 
00347     
00348     
00349     Timestep *ts = mol->current();
00350     if (ts && maxRange > minRange) {
00351       float scalefac = (float)(MAPCLRS-1) / (maxRange - minRange);
00352       int dindex2 = (int)(scalefac * ts->physical_time - minRange);
00353 
00354       
00355       
00356       if (dindex2 < 0)
00357         dindex2 = 0;
00358       else if (dindex2 >= MAPCLRS) 
00359         dindex2 = MAPCLRS-1;
00360 
00361       dindex = MAPCOLOR(dindex2);
00362     }
00363 
00364     
00365     for(int i=0; i < nAtoms; i++) {
00366       color[i] = dindex;
00367     }
00368   } else if(colorMethod == TIMESTEP) {
00369     
00370     float scalefac = (float)(MAPCLRS-1) / ((float) (mol->numframes()-1));
00371     dindex = MAPCOLOR((int)(0.5f + (mol->frame() * scalefac)));
00372     if (dindex >= MAPCLRS)
00373       dindex = MAPCLRS-1; 
00374 
00375     for(int i=0; i < nAtoms; i++) {
00376       color[i] = dindex;
00377     }
00378   } else if(colorMethod == VOLUME) {
00379     
00380     
00381     dindex = scene->nearest_index(1.0, 1.0, 1.0); 
00382     for(int i=0; i < nAtoms; i++) {
00383       color[i] = dindex;
00384     }
00385 
00386     
00387     if (need_recalc_minmax && mol->num_volume_data()) {
00388       VolumetricData *v = mol->modify_volume_data(volIndex);
00389       if (v != NULL) {
00390         v->datarange(minRange, maxRange);
00391       }
00392 
00393       need_recalc_minmax = FALSE;
00394     }
00395   } else if(colorMethod == BACKBONE) {
00396     int regc = scene->category_item_value(molList->colorCatIndex[MLCAT_SPECIAL],
00397                 "Nonback");
00398     int proc = scene->category_item_value(molList->colorCatIndex[MLCAT_SPECIAL],
00399                 "Proback");
00400     int dnac = scene->category_item_value(molList->colorCatIndex[MLCAT_SPECIAL],
00401                 "Nucback");
00402     for (int i=0; i < nAtoms; i++) {
00403       MolAtom *a = mol->atom(i);
00404       int c = regc;
00405       if (a->atomType == ATOMPROTEINBACK ||
00406           a->atomType == ATOMNUCLEICBACK) {  
00407         
00408         
00409         
00410         for(int j=0; j < a->bonds; j++) {
00411           int bondtype = mol->atom(a->bondTo[j])->atomType;
00412           if (bondtype == ATOMPROTEINBACK) {
00413             c = proc;
00414             break;
00415           } else if (bondtype == ATOMNUCLEICBACK) {
00416             c = dnac;
00417             break;
00418           }
00419         }
00420       }
00421       color[i] = c;
00422     }
00423   } else if (colorMethod == STRUCTURE) {
00424     int ind = molList->colorCatIndex[MLCAT_SSTRUCT];
00425     int alpha_helix = scene->category_item_value(ind, "Alpha Helix");
00426     int helix_3_10 = scene->category_item_value(ind, "3_10_Helix");
00427     int pi_helix = scene->category_item_value(ind, "Pi_Helix");
00428     int extended_beta = scene->category_item_value(ind, "Extended_Beta");
00429     int bridge_beta = scene->category_item_value(ind, "Bridge_Beta");
00430     int turn = scene->category_item_value(ind, "Turn");
00431     int coil = scene->category_item_value(ind, "Coil");
00432     mol->need_secondary_structure(1); 
00433     for (int i=0; i<nAtoms; i++) {
00434       Residue *a = mol->residue(mol->atom(i)->uniq_resid);
00435       switch (a->sstruct) {
00436       case SS_HELIX_ALPHA: color[i] = alpha_helix; break;
00437       case SS_HELIX_3_10: color[i] = helix_3_10; break;
00438       case SS_HELIX_PI: color[i] = pi_helix; break;
00439       case SS_BETA: color[i] = extended_beta; break;
00440       case SS_BRIDGE: color[i] = bridge_beta; break;
00441       case SS_TURN: color[i] = turn; break;
00442       case SS_COIL: 
00443       default: color[i] = coil; break;
00444       }
00445     }
00446   } else if(colorMethod == RESID) {
00447     for(int i=0; i < nAtoms; i++) {
00448       dindex = (mol->atom(i))->resid % VISCLRS;
00449       while (dindex < 0) dindex += VISCLRS;
00450       color[i] = dindex;
00451     }
00452     
00453   } else if ((colorMethod >= BETA && colorMethod <= INDEX) ||
00454               colorMethod == VELOCITY) {
00455     
00456     
00457     
00458     
00459     
00460     
00461     
00462     
00463     
00464     
00465     const Timestep *ts = mol->current();
00466     if (!ts) {
00467       
00468       ColorMethod oldMethod = colorMethod;
00469       colorMethod = MOLECULE;
00470       find(mol);
00471       colorMethod = oldMethod;
00472       return TRUE;
00473     }
00474     
00475     float data_min, data_max;
00476     float *data = new float[nAtoms];
00477     if (colorMethod == POS) { 
00478       float *pos = mol->current()->pos;
00479       float cov[3];
00480       float tmp[3];
00481 
00482       mol->cov(cov[0], cov[1], cov[2]);
00483       vec_sub(tmp, cov, pos);
00484       data[0] = data_min = data_max = norm(tmp);
00485       for (int i=1; i<nAtoms; i++) {
00486         pos += 3;
00487         vec_sub(tmp, cov, pos);
00488         data[i] = norm(tmp);
00489         if (data_min > data[i]) data_min = data[i];
00490         if (data_max < data[i]) data_max = data[i];
00491       }
00492     } else if (colorMethod == POSX) { 
00493       float *pos = mol->current()->pos;
00494       data[0] = data_min = data_max = pos[0];
00495       for (int i=1; i<nAtoms; i++) {
00496         pos += 3;
00497         data[i] = pos[0];
00498         if (data_min > data[i]) data_min = data[i];
00499         if (data_max < data[i]) data_max = data[i];
00500       }
00501     } else if (colorMethod == POSY) { 
00502       float *pos = mol->current()->pos;
00503       data[0] = data_min = data_max = pos[1];
00504       for (int i=1; i<nAtoms; i++) {
00505         pos += 3;
00506         data[i] = pos[1];
00507         if (data_min > data[i]) data_min = data[i];
00508         if (data_max < data[i]) data_max = data[i];
00509       }
00510     } else if (colorMethod == POSZ) { 
00511       float *pos = mol->current()->pos;
00512       data[0] = data_min = data_max = pos[2];
00513       for (int i=1; i<nAtoms; i++) {
00514         pos += 3;
00515         data[i] = pos[2];
00516         if (data_min > data[i]) data_min = data[i];
00517         if (data_max < data[i]) data_max = data[i];
00518       }
00519     } else if (colorMethod == FRAGMENT) {
00520       data_min = 0;
00521       data_max = (float) (mol->nFragments-1);
00522       for (int i=0; i<mol->nAtoms; i++) {
00523         data[i] = (float) mol->residue(mol->atom(i)->uniq_resid)->fragment;
00524       }
00525     } else if (colorMethod == INDEX) {
00526       data_min = 0;
00527       data_max = (float) (nAtoms-1);
00528       for (int i=0; i<nAtoms; i++) 
00529         data[i] = (float) i;
00530     } else if (colorMethod == USER || colorMethod == USER2 || 
00531                colorMethod == USER3 || colorMethod == USER4) {
00532       const float *user = NULL;
00533       if (colorMethod == USER)
00534         user = mol->current()->user;
00535       else if (colorMethod == USER2)
00536         user = mol->current()->user2;
00537       else if (colorMethod == USER3)
00538         user = mol->current()->user3;
00539       else if (colorMethod == USER4)
00540         user = mol->current()->user4;
00541 
00542       if (!user) {
00543         memset(data, 0, nAtoms*sizeof(float));
00544         data_min = data_max = 0;
00545       } else {
00546         memcpy(data, user, nAtoms*sizeof(float));
00547 #if 1
00548         minmax_1fv_aligned(data, nAtoms, &data_min, &data_max);
00549 #else
00550         data_min = data_max = data[0];
00551         for (int i=1; i<nAtoms; i++) {
00552           if (data_min > data[i]) data_min = data[i];
00553           if (data_max < data[i]) data_max = data[i];
00554         }
00555 #endif
00556       }
00557     } else if (colorMethod == VELOCITY) { 
00558       float *vel = mol->current()->vel;
00559 
00560       if (!vel) {
00561         memset(data, 0, nAtoms*sizeof(float));
00562         data_min = data_max = 0;
00563       } else {
00564         data[0] = data_min = data_max = norm(vel);
00565         for (int i=1; i<nAtoms; i++) {
00566           vel += 3;
00567           data[i] = norm(vel);
00568           if (data_min > data[i]) data_min = data[i];
00569           if (data_max < data[i]) data_max = data[i];
00570         }
00571       }
00572     } else {
00573       const float *atomfield;
00574       switch (colorMethod) {
00575         case BETA: atomfield = mol->beta(); break;
00576         case OCCUP: atomfield = mol->occupancy(); break;
00577         case MASS: atomfield = mol->mass(); break;
00578         case CHARGE: atomfield = mol->charge(); break;
00579         default: atomfield = mol->mass(); 
00580       }
00581       
00582       memcpy(data, atomfield, nAtoms*sizeof(float));
00583 #if 1
00584       minmax_1fv_aligned(data, nAtoms, &data_min, &data_max);
00585 #else
00586       data_min = data_max = data[0];
00587       for (int i=1; i < nAtoms; i++) {
00588         if (data_min > data[i]) data_min = data[i];
00589         if (data_max < data[i]) data_max = data[i];
00590       }
00591 #endif
00592     }
00593 
00594     if (need_recalc_minmax) {
00595       minRange = data_min;
00596       maxRange = data_max;
00597       need_recalc_minmax = FALSE;
00598     } else {
00599       data_min = minRange;
00600       data_max = maxRange;
00601     }
00602 
00603     
00604     if (data_min == data_max) {
00605       for (int i=0; i<nAtoms; i++) {
00606         color[i] = MAPCOLOR(MAPCLRS/2);
00607       }
00608     } else {
00609       float scalefac = (float)(MAPCLRS-1) / (data_max - data_min);
00610       int dindex2;
00611       for (int i=0; i<nAtoms; i++) {
00612         dindex2 = (int)(scalefac * (data[i] - data_min));
00613         
00614         
00615         if (dindex2 < 0)
00616           dindex2 = 0;
00617         else if (dindex2 >= MAPCLRS) 
00618           dindex2 = MAPCLRS-1;
00619         color[i] = MAPCOLOR(dindex2);
00620       }
00621     }
00622     delete [] data;
00623 
00624   } else if(colorMethod == NAME) {
00625     int ind = molList->colorCatIndex[MLCAT_NAMES];
00626     for (int i=0; i < nAtoms; i++) {
00627       dindex = scene->category_item_value(ind,
00628           (mol->atomNames).data((mol->atom(i))->nameindex));
00629       color[i] = dindex;
00630     }
00631     
00632   } else if(colorMethod == CONFORMATION) {
00633     int i;
00634     int ind = molList->colorCatIndex[MLCAT_CONFORMATIONS];
00635     int allconf = scene->category_item_value(ind, "all");
00636 
00637     NameList<int> *molnames = &(mol->altlocNames);
00638     int alltypecode = molnames->typecode("");
00639 
00640     for(i=0; i < nAtoms; i++) {
00641       int atomidx = (mol->altlocNames).data((mol->atom(i))->altlocindex);
00642       if (atomidx == alltypecode)
00643         dindex = allconf;
00644       else
00645         dindex = scene->category_item_value(ind, atomidx);
00646       color[i] = dindex;
00647     }
00648     
00649   } else if(colorMethod == TYPE) {
00650     int ind = molList->colorCatIndex[MLCAT_TYPES];
00651     for(int i=0; i < nAtoms; i++) {
00652       dindex = scene->category_item_value(ind,
00653           (mol->atomTypes).data((mol->atom(i))->typeindex));
00654       color[i] = dindex;
00655     }
00656 
00657   } else if(colorMethod == ELEMENT) {
00658     int ind = molList->colorCatIndex[MLCAT_ELEMENTS];
00659     for(int i=0; i < nAtoms; i++) {
00660       dindex = scene->category_item_value(ind, 
00661           get_pte_label(mol->atom(i)->atomicnumber));
00662       color[i] = dindex;
00663     }
00664     
00665   } else if(colorMethod == RESNAME) {
00666     int ind = molList->colorCatIndex[MLCAT_RESNAMES];
00667     for(int i=0; i < nAtoms; i++) {
00668       dindex=scene->category_item_value(ind,
00669           (mol->resNames).data((mol->atom(i))->resnameindex));
00670       color[i] = dindex;
00671     }
00672     
00673   } else if(colorMethod == RESTYPE) {
00674     int ind = molList->colorCatIndex[MLCAT_RESTYPES];
00675     for(int i=0; i < nAtoms; i++) {
00676       dindex=scene->category_item_value(ind,
00677                 (molList->resTypes).data((mol->resNames).name(mol->atom(i)->resnameindex)));
00678       color[i] = dindex;
00679     }
00680     
00681   } else if(colorMethod == CHAIN) {
00682     int ind = molList->colorCatIndex[MLCAT_CHAINS];
00683     for(int i=0; i < nAtoms; i++) {
00684       dindex=scene->category_item_value(ind,
00685           (mol->chainNames).data((mol->atom(i))->chainindex));
00686       color[i] = dindex;
00687     }
00688 
00689   } else if(colorMethod == SEGNAME) {
00690     int ind = molList->colorCatIndex[MLCAT_SEGNAMES];
00691     for(int i=0; i < nAtoms; i++) {
00692       dindex=scene->category_item_value(ind,
00693           (mol->segNames).data((mol->atom(i))->segnameindex));
00694       color[i] = dindex;
00695     }
00696 
00697   } else {
00698     msgErr << "Unknown coloring method " << (int)colorMethod << " in AtomColor.";
00699     msgErr << sendmsg;
00700     return FALSE;
00701   }
00702     
00703   return TRUE;
00704 }
00705 
00706 int AtomColor::uses_colorscale() const {
00707   return ((colorMethod >= BETA && colorMethod <= INDEX) || 
00708            colorMethod == VOLUME);
00709 }
00710