📄 density.cc
字号:
//// density.cc//// Copyright (C) 1996 Limit Point Systems, Inc.//// Author: Curtis Janssen <cljanss@limitpt.com>// Maintainer: LPS//// This file is part of the SC Toolkit.//// The SC Toolkit is free software; you can redistribute it and/or modify// it under the terms of the GNU Library General Public License as published by// the Free Software Foundation; either version 2, or (at your option)// any later version.//// The SC Toolkit 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 Library General Public License for more details.//// You should have received a copy of the GNU Library General Public License// along with the SC Toolkit; see the file COPYING.LIB. If not, write to// the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.//// The U.S. Government is granted a limited license as per AL 91-7.//#ifdef __GNUC__#pragma implementation#endif#include <util/misc/formio.h>#include <util/render/polygons.h>#include <math/scmat/local.h>#include <math/scmat/vector3.h>#include <chemistry/molecule/molecule.h>#include <chemistry/qc/wfn/density.h>using namespace std;using namespace sc;static ClassDesc ElectronDensity_cd( typeid(ElectronDensity),"ElectronDensity",1,"public Volume", 0, create<ElectronDensity>, 0);ElectronDensity::ElectronDensity(const Ref<KeyVal> &keyval): Volume(keyval){ wfn_ << keyval->describedclassvalue("wfn");}ElectronDensity::ElectronDensity(const Ref<Wavefunction>& wfn): Volume(), wfn_(wfn){}ElectronDensity::~ElectronDensity(){}voidElectronDensity::compute(){ SCVector3 r; get_x(r); // do_gradient will automatically cause the value to be computed if (gradient_needed()) { double v[3]; set_value(wfn_->density_gradient(r,v)); set_actual_value_accuracy(desired_value_accuracy()); SCVector3 d(v); set_gradient(d); set_actual_gradient_accuracy(desired_gradient_accuracy()); } else if (value_needed()) { set_value(wfn_->density(r)); set_actual_value_accuracy(desired_value_accuracy()); } if (hessian_needed()) { ExEnv::err0() << indent << "ElectronDensity::compute(): hessian isn't yet implemented\n"; abort(); }}// make a wild guess about the bounding boxvoidElectronDensity::boundingbox(double valuemin, double valuemax, SCVector3& p1, SCVector3& p2){ Molecule& mol = *wfn_->molecule(); if (mol.natom() == 0) { for (int i=0; i<3; i++) p1[i] = p2[i] = 0.0; } int i; for (i=0; i<3; i++) p1[i] = p2[i] = mol.r(0,i); for (i=1; i<mol.natom(); i++) { for (int j=0; j<3; j++) { if (mol.r(i,j) < p1[j]) p1[j] = mol.r(i,j); if (mol.r(i,j) > p2[j]) p2[j] = mol.r(i,j); } } for (i=0; i<3; i++) { p1[i] = p1[i] - 3.0; p2[i] = p2[i] + 3.0; }}/////////////////////////////////////////////////////////////////////////////// DensityColorizerstatic ClassDesc DensityColorizer_cd( typeid(DensityColorizer),"DensityColorizer",1,"public MoleculeColorizer", 0, create<DensityColorizer>, 0);DensityColorizer::DensityColorizer(const Ref<KeyVal>&keyval): MoleculeColorizer(keyval){ wfn_ << keyval->describedclassvalue("wfn"); reference_ = keyval->doublevalue("reference"); if (keyval->error() == KeyVal::OK) have_reference_ = 1; else have_reference_ = 0; scale_ = keyval->doublevalue("scale"); if (keyval->error() == KeyVal::OK) have_scale_ = 1; else have_scale_ = 0;}DensityColorizer::~DensityColorizer(){}voidDensityColorizer::colorize(const Ref<RenderedPolygons> &poly){ const double base = 0.3; int i; int nvertex = poly->nvertex(); if (nvertex) { double *data = new double[nvertex]; for (i=0; i<nvertex; i++) { SCVector3 v(poly->vertex(i)); data[i] = wfn_->density(v); } double min = data[0], max = data[0]; for (i=1; i<nvertex; i++) { if (min > data[i]) min = data[i]; if (max < data[i]) max = data[i]; } double center, scale; if (have_reference_) center = reference_; else center = (max+min)/2.0; double maxdiff = fabs(max - center); double mindiff = fabs(min - center); if (have_scale_) { scale = scale_; } else { if (maxdiff>mindiff && maxdiff>1.0e-6) scale = (1.0-base)/maxdiff; else if (mindiff>1.0e-6) scale = (1.0-base)/mindiff; else scale = (1.0-base); } ExEnv::out0() << indent << "DensityColorizer:" << scprintf(" reference=%6.5f", center) << scprintf(" scale=%8.4f",scale) << scprintf(" (%6.5f<=rho<=%6.5f)", max, min) << endl; for (i=0; i<nvertex; i++) { data[i] = (data[i]-center)*scale; } for (i=0; i<nvertex; i++) { Color c; if (data[i] < 0.0) c.set_rgb(-data[i]+base,0.3,0.3); else c.set_rgb(0.3,0.3,data[i]+base); poly->set_vertex_color(i,c); } delete[] data; }}/////////////////////////////////////////////////////////////////////////////// GradDensityColorizerstatic ClassDesc GradDensityColorizer_cd( typeid(GradDensityColorizer),"GradDensityColorizer",1,"public MoleculeColorizer", 0, create<GradDensityColorizer>, 0);GradDensityColorizer::GradDensityColorizer(const Ref<KeyVal>&keyval): MoleculeColorizer(keyval){ wfn_ << keyval->describedclassvalue("wfn"); reference_ = keyval->doublevalue("reference"); if (keyval->error() == KeyVal::OK) have_reference_ = 1; else have_reference_ = 0; scale_ = keyval->doublevalue("scale"); if (keyval->error() == KeyVal::OK) have_scale_ = 1; else have_scale_ = 0;}GradDensityColorizer::~GradDensityColorizer(){}voidGradDensityColorizer::colorize(const Ref<RenderedPolygons> &poly){ const double base = 0.3; int i; int nvertex = poly->nvertex(); if (nvertex) { double *data = new double[nvertex]; for (i=0; i<nvertex; i++) { SCVector3 v(poly->vertex(i)); SCVector3 g; wfn_->density_gradient(v,g.data()); data[i] = g.norm(); } double min = data[0], max = data[0]; for (i=1; i<nvertex; i++) { if (min > data[i]) min = data[i]; if (max < data[i]) max = data[i]; } double center, scale; if (have_reference_) center = reference_; else center = (max+min)/2.0; double maxdiff = fabs(max - center); double mindiff = fabs(min - center); if (have_scale_) { scale = scale_; } else { if (maxdiff>mindiff && maxdiff>1.0e-6) scale = (1.0-base)/maxdiff; else if (mindiff>1.0e-6) scale = (1.0-base)/mindiff; else scale = (1.0-base); } ExEnv::out0() << indent << "GradDensityColorizer:" << scprintf(" reference=%6.5f", center) << scprintf(" scale=%6.2f",scale) << scprintf(" (%6.5f<=rho<=%6.5f)", max, min) << endl; for (i=0; i<nvertex; i++) { data[i] = (data[i]-center)*scale; } for (i=0; i<nvertex; i++) { Color c; if (data[i] < 0.0) c.set_rgb(-data[i]+base,0.3,0.3); else c.set_rgb(0.3,0.3,data[i]+base); poly->set_vertex_color(i,c); } delete[] data; }}/////////////////////////////////////////////////////////////////////////////// Local Variables:// mode: c++// c-file-style: "CLJ"// End:
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -